functions, arguments and changing their value permantently

A

Adam Akhtar

Im coming over to Ruby from C++ after a long break from programming. One
thing im having to get used to is how ruby references or points directly
to variables. In C++ I can pass a variable as an argument to a function
and then change the value within the function. This change will be
reflected outside of the function. How do i go about doing this in ruby?

i.e.

x = 10

def changeit var
var = 20
end

changeit x
puts x
==> 20
 
T

ThoML

x = changeit x

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.
 
T

Todd Benson

Im coming over to Ruby from C++ after a long break from programming. One
thing im having to get used to is how ruby references or points directly
to variables. In C++ I can pass a variable as an argument to a function
and then change the value within the function. This change will be
reflected outside of the function. How do i go about doing this in ruby?

i.e.

x = 10

def changeit var
var = 20
end

changeit x
puts x
==> 20

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
 
S

Stefano Crocco

Im coming over to Ruby from C++ after a long break from programming. One
thing im having to get used to is how ruby references or points directly
to variables. In C++ I can pass a variable as an argument to a function
and then change the value within the function. This change will be
reflected outside of the function. How do i go about doing this in ruby?

i.e.

x = 10

def changeit var
var = 20
end

changeit x
puts x
==> 20

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
 
R

Rick DeNatale

Im coming over to Ruby from C++ after a long break from programming. One
thing im having to get used to is how ruby references or points directly
to variables. In C++ I can pass a variable as an argument to a function
and then change the value within the function. This change will be
reflected outside of the function. How do i go about doing this in ruby?

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
 
A

Adam Akhtar

Thank you so much for all of your detailed replies. Some of you asked
why I wanted to do this and why don`t I just return the value to the
variable I want to change ala x = doit x

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
 
A

Adam Akhtar

Sorry I missed this reply

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.
 
M

MenTaLguY

Thank you so much for all of your detailed replies. Some of you asked
why I wanted to do this and why don`t I just return the value to the
variable I want to change ala x = doit x

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?

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
 

Ask a Question

Want to reply to this thread or ask your own question?

You'll need to choose a username for the site, which only take a couple of moments. After that, you can post your question and our members will help you out.

Ask a Question

Members online

No members online now.

Forum statistics

Threads
473,755
Messages
2,569,537
Members
45,021
Latest member
AkilahJaim

Latest Threads

Top