| |
 |
|
| Computers Forum Index » Computer Languages (Ruby) » Ruby doesn't implement x++ for Fixnum's because ???... |
|
Page 1 of 3 Goto page 1, 2, 3 Next |
|
| Author |
Message |
| RichardOnRails... |
Posted: Fri Oct 30, 2009 1:18 am |
|
|
|
Guest
|
Hi,
In The Well Grounded Rubyist (which, along with The Ruby Way, I
love), Dave Black presents a justification for the x++ omission in
Ruby. I thought I'd see whether I could implement just the
incrementing portion of C's x++ functionality (without the prefix/
postfix issues) in pure Ruby. I wasn’t willing to delve into Matz’
implementation of higher-level Ruby functions or low-level C/C++
code. Below is what I came up with.
Q1. Test 1 failed, I imagine, because the interpret thought “Ah, 7’s
a Fixnum which I ‘ve seen billions of times, so I know what to do with
it.” Thus, it ignored my over-riding definition of Fixnum. So I
call it “compiler error”, to use the vernacular, by reporting that x
had no method “pp”. Am I all wet?
Test 2 passed, but it with a couple of anomalies:
Q2. x.to_s yielded x’s class and address, suggesting x had become a
“semi-Fixed Num”, i.e, x now stored a REFERENCE to the to x’s data,
rather than the immediate storage of the data, as interpreter is wont
to do with Fixnum’s. Am I all wet?
Q3. y.inspect yielded only it’s value ( and nothing else,
suggesting the x.pp yielded a full-fledged Fixnum. Am I all wet?
Dave Black argues that since Fixnum values use immediate storage
rather than referenced storage, x=1;x++ would be akin to changing 1
into 2. This example somewhat supports that view, but the
interpreter sidestepped the matter by creating a reference for x’s
value and an immediate value for y (=x.pp).
Any ideas,
Richard
class FixNum
attr :val
def initialize (fixnum)
puts "Initializing object %d" % fixnum
raise "Initializer not a Fixnum" if
fixnum.class != Fixnum
puts "\nInitializing Fixnum to %d" % fixnum
at (no spam) val = fixnum
end
def pp
at (no spam) val+=1
end
end
# Test 1 (Failure)
x = 7; y = x.pp; z =x.class # undefined method `pp' for 7:Fixnum
(NoMethodError)
# Test 2 (Success)
x = FixNum.new(7); y = x.pp; z =x.class
puts [x.to_s, y.inspect, z.to_s].join("; ") # => <FixNum:0x2b62694>;
8; FixNum |
|
|
| Back to top |
|
|
|
| Rick DeNatale... |
Posted: Fri Oct 30, 2009 11:03 pm |
|
|
|
Guest
|
On Fri, Oct 30, 2009 at 11:05 AM, Robert Klemme
<shortcutter at (no spam) googlemail.com> wrote:
Quote: 2009/10/30 RichardOnRails <RichardDummyMailbox58407 at (no spam) uscomputergurus.com>:
I'll correct my errors and probably have a new theory subsequently.
IMHO the story goes like this: absence of postfix ++ and -- is a
consequence of the fact that numeric types are immutable in Ruby which
makes an assignment necessary for these operators. Although that
would be doable, it would not immediately be obvious when looking at
"foo++". On the other side, "foo += 1" makes the assignment obvious
while still being pretty concise (you do not have to write "foo = foo
+ 1").
I think that theres a more fundamental problem with ++ in a language
like ruby, which has to do with the difference between objects and
variables.
The c ++ and -- operators change a variable, NOT a value.
so in C
a = 1
b = a
a++
a is now 2, but b is still 1.
Now, consider not immutable objects, but defining ++ for a mutable
object. I've named the method plus_plus instead of ++ since I can do
the former, but not the latter.
class String
def plus_plus
self << " plus a plus"
end
end
a = "A non-plussed string"
b = a
puts "a is #{a.inspect}"
puts "b is #{b.inspect}"
a.plus_plus
puts "a is #{a.inspect}"
puts "b is #{b.inspect}"
When we run this we see that the result is:
a is "A non-plussed string"
b is "A non-plussed string"
a is "A non-plussed string plus a plus"
b is "A non-plussed string plus a plus"
Because in languages like Ruby with object reference variable
semantics, methods can only operate on objects, not the variables
which reference them.
So if you COULD successfully define Fixnum#++, or Fixnum#plus_plus:
a = 1
a++
could only change the singleton instance of 1 into 2, which probably
isn't something you'd really want to do.
By 'you' I'm not aiming at you Robert.
This reminds me of a very hard bug I encountered years ago when I was
first was learning to program in Fortran. In the original Fortran you
could assign a new value to a subroutine parameter inside the
subroutine. If the actual parameter value was an integer literal, you
could literally change 1 to 2, and that's why I know you probably
don't want to do this.
--
Rick DeNatale
Blog: http://talklikeaduck.denhaven2.com/
Twitter: http://twitter.com/RickDeNatale
WWR: http://www.workingwithrails.com/person/9021-rick-denatale
LinkedIn: http://www.linkedin.com/in/rickdenatale |
|
|
| Back to top |
|
|
|
| Marnen Laibow-Koser... |
Posted: Sat Oct 31, 2009 12:32 am |
|
|
|
Guest
|
|
| Back to top |
|
|
|
| Robert Klemme... |
Posted: Sat Oct 31, 2009 2:19 am |
|
|
|
Guest
|
On 10/30/2009 08:03 PM, Rick DeNatale wrote:
Quote: On Fri, Oct 30, 2009 at 11:05 AM, Robert Klemme
shortcutter at (no spam) googlemail.com> wrote:
2009/10/30 RichardOnRails <RichardDummyMailbox58407 at (no spam) uscomputergurus.com>:
I'll correct my errors and probably have a new theory subsequently.
IMHO the story goes like this: absence of postfix ++ and -- is a
consequence of the fact that numeric types are immutable in Ruby which
makes an assignment necessary for these operators. Although that
would be doable, it would not immediately be obvious when looking at
"foo++". On the other side, "foo += 1" makes the assignment obvious
while still being pretty concise (you do not have to write "foo = foo
+ 1").
I think that theres a more fundamental problem with ++ in a language
like ruby, which has to do with the difference between objects and
variables.
The c ++ and -- operators change a variable, NOT a value.
How their behavior is defined is completely up to the language designer.
You could make "a++" syntactic sugar for
(_tmp = a; a = a + 1; _tmp)
in the same way as "a += 1" is syntactic sugar for "a = a + 1" or "a ||=
10" is syntactic sugar for "a || a = 10". In fact the sequence in
brackets above would be a completely reasonable way to do it in Ruby:
you could use "a++" in a similar way as in C++ and Java without
sacrificing Fixnum's immutability.
But: you would hide the assignment in the syntactic sugared version and
this could lead to unexpected behavior, like
a = 1
b = a
a++
# now why is b still 1???
For someone unaware of the hidden assignment but aware of the fact that
the object referenced by a must be mutable to make this work it would be
surprising that b is not changed as well. You could only discover that
by looking at #object_id of both (or using #equal?).
Quote: so in C
a = 1
b = a
a++
a is now 2, but b is still 1.
Yes - and in C++ you can have it otherwise:
robert at (no spam) fussel:~$ g++ x.cc && ./a.out
a++=1
a=2
b=2
robert at (no spam) fussel:~$ cat x.cc
#include <stdio.h>
int main( int argc, const char* argv[] )
{
int a = 1;
int &b = a;
printf("a++=%d\n", a++);
printf("a=%d\n", a);
printf("b=%d\n", b);
return 0;
}
robert at (no spam) fussel:~$
Quote: Now, consider not immutable objects, but defining ++ for a mutable
object. I've named the method plus_plus instead of ++ since I can do
the former, but not the latter.
class String
def plus_plus
self << " plus a plus"
end
end
a = "A non-plussed string"
b = a
puts "a is #{a.inspect}"
puts "b is #{b.inspect}"
a.plus_plus
puts "a is #{a.inspect}"
puts "b is #{b.inspect}"
When we run this we see that the result is:
a is "A non-plussed string"
b is "A non-plussed string"
a is "A non-plussed string plus a plus"
b is "A non-plussed string plus a plus"
Because in languages like Ruby with object reference variable
semantics, methods can only operate on objects, not the variables
which reference them.
So if you COULD successfully define Fixnum#++, or Fixnum#plus_plus:
a = 1
a++
could only change the singleton instance of 1 into 2, which probably
isn't something you'd really want to do.
Not necessarily. If this was implemented in the language the sequence
"1" would be an object constructor in the same way as string "constants"
are.
robert at (no spam) fussel:~$ ruby1.8 -e '2.times { puts "foo".object_id }'
-605970738
-605970758
That would the only reasonable way to do it. Then there would be no
singleton instance which in turn would have the said performance
implications.
I would consider the example from Fortran that you presented (changing
the value of the constant 1) rather a bug either in the language
implementation or in the design. This does not mean that other
languages would have to repeat that. :-)
Kind regards
robert
--
remember.guy do |as, often| as.you_can - without end
http://blog.rubybestpractices.com/ |
|
|
| Back to top |
|
|
|
| Rick DeNatale... |
Posted: Sun Nov 01, 2009 1:29 am |
|
|
|
Guest
|
On Sat, Oct 31, 2009 at 1:35 PM, Simon Krahnke <overlord at (no spam) gmx.li> wrote:
Quote: * Robert Klemme <shortcutter at (no spam) googlemail.com> (23:19) schrieb:
Yes - and in C++ you can have it otherwise:
robert at (no spam) fussel:~$ g++ x.cc && ./a.out
a++=1
a=2
b=2
robert at (no spam) fussel:~$ cat x.cc
#include <stdio.h
int main( int argc, const char* argv[] )
{
int a = 1;
int &b = a;
printf("a++=%d\n", a++);
printf("a=%d\n", a);
printf("b=%d\n", b);
return 0;
}
No, you can't:
$ ./ab
a: 2 at (no spam) 0xbfe413f8
b: 2 at (no spam) 0xbfe413f8
$ cat ab.cc
#include <iostream
int main(void) {
int a = 1;
int &b = a;
a++;
std::cout << "a: " << a << " at (no spam) " << &a << std::endl;
std::cout << "b: " << b << " at (no spam) " << &b << std::endl;
return 0;
}
There is only one variable, b is just another name for it.
I'm not going to run the experiment, but I believe that if the
int &b = a;
were changed to
int b = a;
Then the result would be the same as in Robert's 'surprising' ruby example.
Which only points out the different in meaning between a C 'reference'
variable, which aliases another variable, and the concept of a ruby
variable as a reference to an object.
And no, I don't think that adding syntactic sugar to Ruby to make a++
the same as either
a += 1
or
a = a.succ
is a good idea.
As a matter of fact, neither of this is REALLY the same as a++, since
things like
b = a++
should assign the ORIGINAL value of a to b, as opposed to
b = ++a
Things like post and pre increment/decrement operators in C like
languages are best reserved to languages with the same basic
structure.
To my mind they are akin to the different rules for gender, and
agreement in different families of languages.
--
Rick DeNatale
Blog: http://talklikeaduck.denhaven2.com/
Twitter: http://twitter.com/RickDeNatale
WWR: http://www.workingwithrails.com/person/9021-rick-denatale
LinkedIn: http://www.linkedin.com/in/rickdenatale |
|
|
| Back to top |
|
|
|
| RichardOnRails... |
Posted: Wed Nov 04, 2009 3:24 am |
|
|
|
Guest
|
On Oct 30, 11:05 am, Robert Klemme <shortcut... at (no spam) googlemail.com> wrote:
Quote: 2009/10/30 RichardOnRails <RichardDummyMailbox58... at (no spam) uscomputergurus.com>:
I'll correct my errors and probably have a new theory subsequently.
IMHO the story goes like this: absence of postfix ++ and -- is a
consequence of the fact that numeric types are immutable in Ruby which
makes an assignment necessary for these operators. Although that
would be doable, it would not immediately be obvious when looking at
"foo++". On the other side, "foo += 1" makes the assignment obvious
while still being pretty concise (you do not have to write "foo = foo
+ 1").
Which brings us to the question: what is the advantage of immutable
numbers? First, it avoids errors that could be caused by aliasing
(two objects refer the same object, one of them changes it, it changes
for the other one as well without being expected). Then, it is quite
natural if you consider mathematical numbers: they cannot change.
Additionally, the expression "1" can always refer to the same object
(in reality it's a bit different but from a Ruby programmer's
perspective the difference is not noticeable) which in fact makes
using numeric constants very efficient (as opposed to the expression
"'foo'" which constructs a new String instance on every invocation,
albeit with a shared char array underneath which eases the pain a
bit).
This in turn makes integer math pretty efficient because if numbers
were mutable Ruby would have to create a new object for every result
of an operator evaluation. I am not saying that Ruby is ideal for
number crunching but it could be significantly slower if certain
design decisions would have been made differently.
Kind regards
robert
--
remember.guy do |as, often| as.you_can - without endhttp://blog.rubybestpractices.com/
Hi Robert,
Thank you for your thoughtful and extensive responses. I apologized
earlier for my sloppy analysis. I hope the following offers higher
quality.
(a) You say, in part:
“... immutable numbers? First, it avoids errors that could be caused
by aliasing (two objects refer the same object, one of them changes
it, it changes for the other one as well without being expected”
(b) Dave Black, in his truly excellent “The Well-Grounded Rubyist”
says, in part:
“Any object that’s represented as an immediate value is always exactly
the same object” and
“The reason [there’s no x=1; x++] is that , due to the immediate
presence of 1 in x, means you’d be changing 1 into 2,
and that makes no sense”
1. I’d rather discuss this matter in concrete terms rather than
abstractions. Please look at the following method (to generate
results) and two statements:
def show(v)
"Got #{v}, class = #{v.class}, object_id = #{v.object_id}
(v.object_id-1)/2 = #{(v.object_id-1)/2 }"
end
a = 1; show (a) => Got 1; class = Fixnum; object_id = 3; v >> 1
= 1
b = 1; show (b) => Got 1; class = Fixnum; object_id = 3; v >> 1
= 1
a == b => true
Both “a” and “b” have their respective values held as immediate values
(embedded in their object_id’s), but they are NOT the same thing
because they’re ultimately held in their respective entries in the
symbol table. So, object_id’s for Fixnum’s are synthetic. To think
that those object_id’s point to a location in a memory area that
stores the 32-bit 000...001 (in a 32-bit machine/OS) is to contradict
the meaning of immediacy and defeat the very efficiency that immediate
values offer.
2. Now please consider the following supplement to the statements
above:
a += 1; show (a) => Got 2; class = Fixnum; object_id = 5; v >> 1
= 2
show (a) => Got 2; class = Fixnum; object_id = 5; v >> 1 = 2
show (b) => Got 1; class = Fixnum; object_id = 3; v >> 1 = 1
The assignment of “a += 1” to “a” changed a’s object_id to embed a new
value: 2. Despite that change of 1 to 2 in a’s object, “b” remains
set to 1. “b” did not suffer the calamity of a universal change of
all Fixnum 1’s to 2’s.
3. Conclusion:
a += 1 is equivalent to a++’s natural meaning. In fact, we further
supplement the above statements with:
class Fixnum
def pp # We can’t define ++ because of a compiler restriction.
self + 1
end
end
a=1; show(a.pp) => Got 2; class = Fixnum; object_id = 5; v >> 1
= 2
show(b) => Got 1; class = Fixnum; object_id = 3; v >> 1 = 1
Furthermore, pp works fine on the boundary condition for Fixnum’s on a
common 32-bit cpu/OS, i.e. 2**30-1:
a = 2**30-1; show (a) => Got 1073741823; class = Fixnum; object_id
= 2147483647; v >> 1 = 1073741823
a = 2**30; show (a) => Got 1073741824; class = Bignum; object_id 22737670; v >> 1 = 11368835
So x++ works fine in the form of pp for positive Fixnum’s. I assume
it’ll work fine for non-positives, also.. A compiler change to allow
“def ++” is necessary to finally add ++ to Fixnum.
Again, thanks for your responses to my question.
Best wishes,
Richard |
|
|
| Back to top |
|
|
|
| Tony Arcieri... |
Posted: Wed Nov 04, 2009 6:17 am |
|
|
|
Guest
|
[Note: parts of this message were removed to make it a legal post.]
On Tue, Nov 3, 2009 at 10:30 PM, Michael W. Ryder <_mwryder55 at (no spam) gmail.com>wrote:
Quote: Maybe I am missing something
Yes, maybe you're missing that I support a unary ++ operator :)
I can see the wisdom in not having it there to begin with though. I see the
++ operator as something almost inextricably tied to for loops in C. Ruby
provides better looping mechanisms, like the various Enumerable methods that
take blocks! Perhaps Matz didn't include a unary ++ operator as
encouragement to use better looping constructs in your code. If that's the
case, I think he certainly succeeded.
--
Tony Arcieri
Medioh/Nagravision |
|
|
| Back to top |
|
|
|
| Michael W. Ryder... |
Posted: Wed Nov 04, 2009 6:17 am |
|
|
|
Guest
|
Tony Arcieri wrote:
Quote: [Note: parts of this message were removed to make it a legal post.]
On Fri, Oct 30, 2009 at 12:03 PM, Rick DeNatale <rick.denatale at (no spam) gmail.com>wrote:
I think that theres a more fundamental problem with ++ in a language
like ruby, which has to do with the difference between objects and
variables.
Personally I see nothing interesting about the behavior of ++ in any mutable
state language.
Now, consider not immutable objects, but defining ++ for a mutable
object. I've named the method plus_plus instead of ++ since I can do
the former, but not the latter.
How is ++ any different from << except for << taking an argument? (given
hypothetical C-like ++ behavior)
Maybe I am missing something but how would one print a variable and
increment it with the same statement? In C one would just write printf
("%i/n", x++). I haven't seen any easy way to do this with Ruby. Using
<< or +=1 you have to have a separate statement for the assignment. So
instead of having something like:
i = 1
while (i < 10)
puts i++
end
you have to have:
i = 1
while (i< 10)
puts i; i +=1
end
I can see a reason for not being able to do 2++, but not i++.
Quote: There is already extensive precedent in Ruby for destructive method calls
that mutate state, and they all lead to the same confusion.
There is nothing interesting with Ruby in this regard, except that Ruby does
seem to go out of its way to do things immutably by default, which, in my
opinion, is pretty cool. But in the end Ruby is still very much a mutable
state language.
|
|
|
| Back to top |
|
|
|
| Tony Arcieri... |
Posted: Wed Nov 04, 2009 6:17 am |
|
|
|
Guest
|
[Note: parts of this message were removed to make it a legal post.]
On Fri, Oct 30, 2009 at 12:03 PM, Rick DeNatale <rick.denatale at (no spam) gmail.com>wrote:
Quote: I think that theres a more fundamental problem with ++ in a language
like ruby, which has to do with the difference between objects and
variables.
Personally I see nothing interesting about the behavior of ++ in any mutable
state language.
Quote: Now, consider not immutable objects, but defining ++ for a mutable
object. I've named the method plus_plus instead of ++ since I can do
the former, but not the latter.
How is ++ any different from << except for << taking an argument? (given
hypothetical C-like ++ behavior)
There is already extensive precedent in Ruby for destructive method calls
that mutate state, and they all lead to the same confusion.
There is nothing interesting with Ruby in this regard, except that Ruby does
seem to go out of its way to do things immutably by default, which, in my
opinion, is pretty cool. But in the end Ruby is still very much a mutable
state language.
--
Tony Arcieri
Medioh/Nagravision |
|
|
| Back to top |
|
|
|
| Seebs... |
Posted: Wed Nov 04, 2009 6:17 am |
|
|
|
Guest
|
On 2009-11-04, Rick DeNatale <rick.denatale at (no spam) gmail.com> wrote:
Quote: Why this wierd semantic? Because it the original use case was for
pointers rather than integers, and it goes back to C originating
originally as a kind of high level assembly language for the DEC
PDP-11 which had postincrement and predecrement addressing modes used
for stepping through strings or arrays.
Apparently apocryphal -- the increment usage predates the PDP 11 port
of C. (Interestingly, there's now a ton of sources repeating the story,
but so far as I know, the canonical answer from Ritchie and Thompson
was that the increment operator predates C proper, and thus the PDP 11
work.)
Quote: Because pre-ANSI C allowed easy spoofing/overlay of types, it also
worked for integers and was used in contexts like for loops
for(i = 0; i < max;i++)
It is not because of spoofing or overlay of types that this worked,
but because the operator's defined for all types.
Quote: If you want the real semantics of the C/C++ post-increment operator,
I'm afraid you'll have to look for it in a C family language, not an
object reference semantic language like Ruby.
This part, though, I totally agree with. It's not coherent to imagine a
++ operator, especially a postincrement, working in a Ruby-like language.
-s
--
Copyright 2009, all wrongs reversed. Peter Seebach / usenet-nospam at (no spam) seebs.net
http://www.seebs.net/log/ <-- lawsuits, religion, and funny pictures
http://en.wikipedia.org/wiki/Fair_Game_(Scientology) <-- get educated! |
|
|
| Back to top |
|
|
|
| Rick DeNatale... |
Posted: Wed Nov 04, 2009 6:17 am |
|
|
|
Guest
|
On Tue, Nov 3, 2009 at 10:25 PM, RichardOnRails
<RichardDummyMailbox58407 at (no spam) uscomputergurus.com> wrote:
Quote: a += 1 is equivalent to a++’s natural meaning. In fact, we further
supplement the above statements with:
class Fixnum
def pp # We can’t define ++ because of a compiler restriction.
self + 1
end
end
Quote: So x++ works fine in the form of pp for positive Fixnum’s. I assume
it’ll work fine for non-positives, also.. A compiler change to allow
“def ++” is necessary to finally add ++ to Fixnum.
No x++ doesn't work fine if you want it to act like C's post-increment operator.
Let's leave Ruby aside for a moment and consider this snippet of C code.
int a = 1;
int b = a++
After these two lines a is 2, and b is 1, since a++ returns the
original value of a and then increments a.
Why this wierd semantic? Because it the original use case was for
pointers rather than integers, and it goes back to C originating
originally as a kind of high level assembly language for the DEC
PDP-11 which had postincrement and predecrement addressing modes used
for stepping through strings or arrays.
Because pre-ANSI C allowed easy spoofing/overlay of types, it also
worked for integers and was used in contexts like for loops
for(i = 0; i < max;i++)
where the value of the expression i++ was never used, only the side
effect that it left a as 1 higher than before was important.
Because of this, many C/C++ programmers seem to be unaware or forget
that the value of a++ is a, not (a+1). In fact Bjarne Stroustrup used
to rib himself by pointing out that the value of C++ was exactly the
same as C, because he IS certainly aware of the semantics.
Now what happens if we translate my initial snippet into Ruby using
your pp method
class Integer
# Not need to restrict to FixNums is there
# Name it pp since ++ isn't a valid Ruby method selector
def pp
self + 1
end
end
a = 1
b = a.pp
puts "a is #{a}, b is #{b}"
Produces not "a is 2, b is 1" as it would if it correctly implemented
C language semantics but
a is 1, b is 2
just the opposite of what would be expected by a C/C++ programmer who
understood the meaning of post-increment ++
The fact of the matter is that you can't write a ++ or a pp method in
Ruby which works like the C post-increment because it requires the
method to change the value of a variable which is bound to the object
and the method only knows what the object knows and the object doesn't
know which variable(s) refer to it.
consider
a = b = c = 1
b.pp
There are at least 3 variables which refer to the Fixnum object 1, The
fact that that reference is through a special form object id computed
by multiplying 1 by 2 and adding 1, as it is in several Ruby
implementations, is irrelevant. So is the fact that Fixnums happen to
be immutable. No object knows what variables reference it, and
wouldn't have a way to directly change those variables bindings
anyway.
The rebinding of a in
a = a.pp
doesn't happen in the pp method it happens in the calling context
because of the assignment expression there.
So if you just want a ++ which returns the incremented value of a
fixnum then you can write it as pp, but you really don't need to since
Integer#succ does just that.
If you want the real semantics of the C/C++ post-increment operator,
I'm afraid you'll have to look for it in a C family language, not an
object reference semantic language like Ruby.
--
Rick DeNatale
Blog: http://talklikeaduck.denhaven2.com/
Twitter: http://twitter.com/RickDeNatale
WWR: http://www.workingwithrails.com/person/9021-rick-denatale
LinkedIn: http://www.linkedin.com/in/rickdenatale |
|
|
| Back to top |
|
|
|
| Walton Hoops... |
Posted: Thu Nov 05, 2009 4:48 am |
|
|
|
Guest
|
Quote: -----Original Message-----
From: marnen at (no spam) marnen.org [mailto:marnen at (no spam) marnen.org]
Now consider the ruby way:
10.times do |i|
print "#{i},"
end
Some length as the C code, but much more readable. Heck, it's
almost English!
Well, you do have to know what Numeric#times yields to its block. But
that's easy to look up. (However, it starts from 0, so it's not quite
equivalent to the C.)
Whoop! Good point, that's what I get for not actually testing my code.
Corrected (and even closer to English).
(1..9).each do |i|
print "#{i},"
end
Michael Wrote:
Quote: Not for me it wasn't. I had to try it to see that it actually works.
My initial impression was that it would print 10 copies of i.
I still don't see where 'i' is incremented
It isn't incremented, at least not in MY code (if you must think in
terms of incrementing variables, then Ruby is incrementing it for me)
See the times method at:
http://ruby-doc.org/core/classes/Integer.html
That kind of looping is a pretty core Ruby concept.
Also the corrected version I wrote above should be a bit clearer. Also
I think your previous programming experience is hurting you here. To you
as (I'm guessing) a C programmer, to progress in a loop, you must increment
modify variable. The average English speaker doesn't think in those terms.
Looking at my most recent example, the English equivalent would be for each
'i' from 1 to 9 print 'i' followed by a comma. Sure, the words may not be
in the precise order, but it comes a darn site closer to natural language than:
int i=1;
while (i<10)
{
printf("%d,",i++);
} |
|
|
| Back to top |
|
|
|
| Michael W. Ryder... |
Posted: Thu Nov 05, 2009 5:08 am |
|
|
|
Guest
|
Walton Hoops wrote:
Quote: -----Original Message-----
From: marnen at (no spam) marnen.org [mailto:marnen at (no spam) marnen.org]
Now consider the ruby way:
10.times do |i|
print "#{i},"
end
Some length as the C code, but much more readable. Heck, it's
almost English!
Well, you do have to know what Numeric#times yields to its block. But
that's easy to look up. (However, it starts from 0, so it's not quite
equivalent to the C.)
Whoop! Good point, that's what I get for not actually testing my code.
Corrected (and even closer to English).
(1..9).each do |i|
print "#{i},"
end
That version I understand just looking at it as it is equivalent to a
for loop. Your first version seemed more "magical" since I don't know
where 'i' is getting incremented. At least with C I know where the
incrementing is occurring.
Quote: Michael Wrote:
Not for me it wasn't. I had to try it to see that it actually works.
My initial impression was that it would print 10 copies of i.
I still don't see where 'i' is incremented
It isn't incremented, at least not in MY code (if you must think in
terms of incrementing variables, then Ruby is incrementing it for me)
See the times method at:
http://ruby-doc.org/core/classes/Integer.html
That kind of looping is a pretty core Ruby concept.
Also the corrected version I wrote above should be a bit clearer. Also
I think your previous programming experience is hurting you here. To you
as (I'm guessing) a C programmer, to progress in a loop, you must increment
modify variable. The average English speaker doesn't think in those terms.
I started with Fortran in the early 1980's, followed by Basic, Pascal,
Modula 2, and C. For over 25 years I have been mostly programming in
Business Basic. While Ruby has a lot of things going for it I miss some
of the features available in the other languages, especially the built
in curses and file handling in Business Basic.
Quote: Looking at my most recent example, the English equivalent would be for each
'i' from 1 to 9 print 'i' followed by a comma. Sure, the words may not be
in the precise order, but it comes a darn site closer to natural language than:
int i=1;
while (i<10)
{
printf("%d,",i++);
}
|
|
|
| Back to top |
|
|
|
| David A. Black... |
Posted: Fri Nov 06, 2009 5:28 am |
|
|
|
Guest
|
Hi--
On Fri, 6 Nov 2009, Tony Arcieri wrote:
Quote: On Thu, Nov 5, 2009 at 3:40 AM, Gavin Sinclair <gsinclair at (no spam) gmail.com> wrote:
def ++
incrementing_logic_goes_here
end
I could see this as being handy
What's wrong with
def inc
incrementing_logic_goes_here
end
How is that any different?
What's wrong with Array#push? Why do we need Array#<<? How is that any
different?
irb(main):001:0> [].push(1,2,3)
=> [1, 2, 3]
irb(main):002:0> [].<<(1,2,3)
ArgumentError: wrong number of arguments (3 for 1)
:-) But I know that's not what you meant. The thing is, a method
called ++ that did in-place incremention would not be meaningful for
numbers (if I understand correctly that you mean it would be similar
to succ!), and having it for other objects would probably just lead to
more confusion. That's my hunch, at least.
David
--
The Ruby training with D. Black, G. Brown, J.McAnally
Compleat Jan 22-23, 2010, Tampa, FL
Rubyist http://www.thecompleatrubyist.com
David A. Black/Ruby Power and Light, LLC (http://www.rubypal.com) |
|
|
| Back to top |
|
|
|
| Michael W. Ryder... |
Posted: Sat Nov 07, 2009 2:24 am |
|
|
|
Guest
|
Walton Hoops wrote:
Quote: -----Original Message-----
From: marnen at (no spam) marnen.org [mailto:marnen at (no spam) marnen.org]
Now consider the ruby way:
10.times do |i|
print "#{i},"
end
Some length as the C code, but much more readable. Heck, it's
almost English!
But if you wanted to do something like:
i = 10;
while (i > 0)
{
printf("%d/n", i--);
}
in Ruby you would have to do something like:
i = 10
while (i > 0)
puts i
i -= 1
end
As far as I can tell there is no way in Ruby to use .each or .times to
go backwards. While I realize this thread is about the ++ operator the
-- operator is complementary.
<snip> |
|
|
| Back to top |
|
|
|
|
|
All times are GMT
The time now is Sun Nov 22, 2009 7:45 am
|
|