Dynamic Variables

Is there any way in ruby to create dynamic variables?

counter = 52
box#{counter} = “cat and dogs”

There would be then a new variable
box52

Note that this is not a question whether someone should do it or not -
this is solely whether it is doable or not. Until today I thought it is
somehow possible with eval but I failed, so I assume it is not possible.

Marc H. wrote:

somehow possible with eval but I failed, so I assume it is not possible.

irb(main):010:0> counter = 52
=> 52
irb(main):011:0> eval “box#{counter} = ‘cat and dogs’”
=> “cat and dogs”
irb(main):012:0> box52
=> “cat and dogs”

-igal

On Tue, Jul 1, 2008 at 11:52 AM, Igal K. [email protected]
wrote:

Note that this is not a question whether someone should do it or not -
this is solely whether it is doable or not. Until today I thought it is
somehow possible with eval but I failed, so I assume it is not possible.

irb(main):010:0> counter = 52
=> 52
irb(main):011:0> eval “box#{counter} = ‘cat and dogs’”
=> “cat and dogs”
irb(main):012:0> box52
=> “cat and dogs”

Now try the same thing in plain ruby without irb. Never trust irb on
things like these.

Dynamic variables as created by eval live in a separate, eval-specific
scope and can only be accessed through eval. This is a consequence of
Ruby deciding at “compile” time what’s a local variable and what’s
not. So, if you have code like this:

eval “a = 5”
p a

then in “p a”, a is determined to be a method call before that code is
even run. The a is accessible through eval though:

eval “a = 5”
p eval(“a”)

Peter

On 1 Jul 2008, at 10:52, Igal K. wrote:

not -
=> “cat and dogs”

Although that doesn’t work outside of irb. (although you can change
the value of an existing local variable like that)
See also
http://groups.google.com/group/comp.lang.ruby/browse_thread/thread/a954f8aaf698a0b9/e1c761b71b63b82e?#e1c761b71b63b82e

Fred

From: Marc H. [mailto:[email protected]]

Is there any way in ruby to create dynamic variables?

note sure, but no. There’s no need :wink:

counter = 52

box#{counter} = “cat and dogs”

There would be then a new variable box52

Note that this is not a question whether someone should do it or

not -this is solely whether it is doable or not. Until today I

thought it is somehow possible with eval but I failed, so I assume

it is not possible.

variables in ruby are just references to objects. and since ruby objects
can be dynamic, variables are dynamic in a basic sense, but in an o-o
way :wink:

on your case, why not make box refer to a dynamic object, like a hash?

eg,

box={}
#=> {}
counter = 52
#=> 52
box[counter] = ‘cat and dogs’
#=> “cat and dogs”

box[counter]
#=> “cat and dogs”

store anything you want,

box[“myname”]=“botp”
#=> “botp”

even store a proc and call it

box[42]=lambda{box[counter].upcase}
#=> #Proc:0x028d31e4@:9(irb)

box
#=> {“myname”=>“botp”, 52=>“cat and dogs”,
42=>#Proc:0x028d31e4@:9(irb)}

box[42].call
#=> “CAT AND DOGS”

and you can even let box, cleanup itself :slight_smile:

box[“cleanup”]=lambda{box.clear}
#=> #Proc:0x028c141c@:19(irb)

box[“cleanup”].call
#=> {}

box
#=> {}

kind regards -botp

Frederick C. wrote:

Although that doesn’t work outside of irb. (although you can change
the value of an existing local variable like that)
See also
http://groups.google.com/group/comp.lang.ruby/browse_thread/thread/a954f8aaf698a0b9/e1c761b71b63b82e?#e1c761b71b63b82e

Thanks for catching the mistake and providing the link.

So there really is no way to get the “box52” variable directly? It’s
only available within the context of an eval?

When I run the code below directly (not from IRB), the interpreter is
clearly seeing the “box52” local variable and keeping its state around,
but won’t let me access it without wrapping an eval around it, which
seems wrong:

counter = 52
p local_variables # => [“counter”]
eval “box#{counter} = ‘cat and dogs’”
p local_variables # => [“counter”, “box52”]
p eval(“box52”) # => “cat and dogs”
p box52 # => undefined local variable or method `box52’ for main:Object
(NameError)

I say “seems wrong” because I’m used to the following behavior:

Python

counter = 52
eval(compile(‘box%s = “cat and dogs”’ % counter, ‘’, ‘exec’))
print box52 # => “cat and dogs”

Perl

$counter = 52;
eval(“$box$counter = ‘cat and dogs’”);
print $box52, “\n”; # => “cat and dogs”

-igal

On Tue, Jul 01, 2008 at 06:45:48PM +0900, Marc H. wrote:

somehow possible with eval but I failed, so I assume it is not possible.
I’m going to give you exactly what you didn’t ask for: why this is a bad
idea. This is a discussion about Perl rather than Ruby, but it remains
relevant: Why it's stupid to `use a variable as a variable name'

See the subsequent parts of the series for further discussion.

–Greg

On Jul 1, 1:50 pm, Gregory S. [email protected]
wrote:

this is solely whether it is doable or not. Until today I thought it is
somehow possible with eval but I failed, so I assume it is not possible.

I’m going to give you exactly what you didn’t ask for: why this is a bad
idea. This is a discussion about Perl rather than Ruby, but it remains
relevant:Why it's stupid to `use a variable as a variable name'

See the subsequent parts of the series for further discussion.

–Greg

Here here - I remember reading that article way back - the basic point
of it is that if you want to create a variable on the fly, it probably
means you should be using a Hash.

Igal K. wrote:

Python

counter = 52
eval(compile(‘box%s = “cat and dogs”’ % counter, ‘’, ‘exec’))
print box52 # => “cat and dogs”

Perl

$counter = 52;
eval("$box$counter = ‘cat and dogs’");
print $box52, “\n”; # => “cat and dogs”

In python, local variables are not (necessarily) determined at
parse/compile time. They’re slots in a local scope that can grow later
on.

In Ruby, this behavior only exists for eval scopes. Ruby 1.8 shares a
single eval scope under any given normal scope across all evals in that
normal scope. Ruby 1.9 instantiates a new eval scope for each eval.

I don’t know what Perl does.

  • Charlie

‘Object’ has a method to define instance variables:

irb(main):001:0> counter = 52
irb(main):002:0> self.instance_variable_set("@box#{counter}", ‘cat and
dogs’)
=> “cat and dogs”
irb(main):003:0> @box52
=> “cat and dogs”

martin

Charles Oliver N. wrote:

In python, local variables are not (necessarily) determined at
parse/compile time. They’re slots in a local scope that can grow later
on.

In Ruby, this behavior only exists for eval scopes. Ruby 1.8 shares a
single eval scope under any given normal scope across all evals in
that normal scope. Ruby 1.9 instantiates a new eval scope for each eval.

I don’t know what Perl does.
Charles, thanks for the clarification about the scoping behaviors.

I suppose if I really want to pretend like I was dynamically injecting a
local variable into a local scope, and a class/instance variable didn’t
weren’t desirable, then I could resort to:

def set(name, value)
self.class.send(:attr_accessor, name)
self.send("#{name}=", value)
end

counter = 52
set “box#{counter}”, “cat and dogs”
p box52 # => “cat and dogs”

-igal

Igal K. wrote:

set “box#{counter}”, “cat and dogs”
p box52 # => “cat and dogs”

Yup, that would work except that for an attr_accessor on self you need
to qualify it with self.box52…so it wouldn’t be quite seamless.

  • Charlie

Hello,
How to get the IP address of client in ruby?
thanks