i.e.
x = 10
def changeit var
var = 20
end
changeit x
puts x
==> 20
--
Posted via http://www.ruby-forum.com/.
You can easily return multiple arguments (x, y = changeit(x, y)) which
makes the inout semantics rather obsolete (with a few exceptions where
macro would be desirable). Some classes also provide destructive
methods that can be used to replace it's contents while maintaining
the object's identity.
Ruby tries to maintain scope rigidly. So your x before will not
change within the scope of the method changeit.
Your method, when called, says...
changeit 10
Then you want to do assignment as...
x = 20
It's a different x!
You can pull out of local scope with class instance variables or
globals. Your same code, just changed...
@x = 10
def changeit n
@x = n
end
..or return it directly form the method if that's the only value you need
x = 10
def changeit n
n
end
x = changeit
..in which case you would probably want to rename the method.
I have to say, I don't see this as a good design pattern using Ruby,
but I don't know what you're trying to do :)
I can't tell by your post, but if really what you want to do is send a
variable "name" into the method and have the method change the value.
That's different and might require some brainiacs on the list to help
you (my guess is that you would have use some form of #eval).
Todd
In ruby you pass the object to the method, not a reference to the variable.
This means that, if you change the object, changes will affect every variable
which contains the object. On the other hand, changing the object contained in
the argument of the method (that is, writing var = 20) won't affect any other
variable. This means that if some classes are unchangeable (such as all kind
of numbers and symbols), you can't do what you want. Other classes provide
methods which change them in place. As I said, such changes will affect every
variable which points to the object. Here are some examples which should make
things clearer:
def m1 str
str.upcase
end
def m2 str
str = "a string"
end
def m3 str
str.upcase!
end
x = "test"
m1 x
puts "After m1 x is: #{x}"
m2 x
puts "After m2 x is: #{x}"
m3 x
puts "After m3 x is: #{x}"
The output is:
After m1 x is: test
After m2 x is: test
After m3 x is: TEST
m1 calls String#upcase, which is a non-destructive method (that is, a method
which doesn't change its receiver) and does nothing else. Of course, this
doesn't change x.
m2 sets the local variable str, which initially contains the argument passed
to the method, to a different value, "a string". In C++, if the argument was a
reference, this would have also set x to "a string". In ruby it doesn't.
m3 calls a destructive method on str. A destructive method changes its
receiver, so the contents of the object contained by str change (notice that
this is different from the previous case, where a new object was put into
str). Since str and x contain the same object, x also changes. A method like
this can be used to achieve the result you wanted, but it can be written only
if the argument has a destructive method which does what you need (for
example, you couldn't have done the same for an Integer, since it doesn't have
destructive methods).
I hope this helps
Stefano
Perhaps something here can help you get your head around this aspect
of how Ruby isn't C++:
http://talklikeaduck.denhaven2.com/articles/2006/09/13/on-variables-values-and-objects
http://talklikeaduck.denhaven2.com/articles/2008/02/08/whose-variable-is-it-anyway
--
Rick DeNatale
My blog on Ruby
http://talklikeaduck.denhaven2.com/
Well actually my function changes 2 values. Functions can only pass one
back so I thought I`d pass one as an argument as well. Is that bad
programming design? What are other options?
Many thanks
You can easily return multiple arguments (x, y = changeit(x, y)) which
makes the inout semantics rather obsolete (with a few exceptions where
macro would be desirable). Some classes also provide destructive
methods that can be used to replace it's contents while maintaining
the object's identity.
which answers my question. Between reading the posts and replying I had
a good nights sleep and still a bit groggy.
You could have the function return multiple values as an array:
def foo(x, y)
[ x + y, x * y ]
end
sum, product = foo(2, 3)
puts sum
puts product
-mental