views:

1708

answers:

4

I have the following code:

for attribute in site.device_attributes
  device.attribute
end

where I would like the code to substitute the value of "attribute" for the method name.

I have tried device."#{attribute}" and various permutations.

Is this completely impossible? Am I missing something?

I have considered overriding method_missing, but I can't figure out how that would actually help me when my problem is that I need to call an "unknown" method.

+11  A: 

The "send" method should do what you're looking for:

object = "upcase me!"
method = "upcase"
object.send(method.to_sym) # => "UPCASE ME!"
Matt Campbell
+1  A: 

you can also do

device.instance_eval(attribute)
David Nehme
+9  A: 

You can use #send method to call object's method by method's name:

object.send(:foo) # same as object.foo

You can pass arguments with to invoked method:

object.send(:foo, 1, "bar", 1.23) # same as object.foo(1, "bar", 1.23)

So, if you have attribute name in variable "attribute" you can read object's attribute with

object.send(attribute.to_sym)

and write attribute's value with

object.send("#{attribute}=".to_sym, value)

In Ruby 1.8.6 #send method can execute any object's method regardless of it's visibility (you can e.g. call private methods). This is subject to change in future versions of Ruby and you shouldn't rely on it. To execute private methods, use #instance_eval:

object.instance_eval {
  # code as block, can reference variables in current scope
}

# or

object.instance_eval <<-CODE
  # code as string, can generate any code text
CODE
Maxim Kulkin
+1 for future compatible.
Jonke
Thanks, this helped me out a bunch too!
daybreaker
+4  A: 

Matt and Maxim are both correct, but leave out a detail that might help you get your head around the #send syntax: In Ruby, calling a method is really sending a message. Softies on Rails has a relatively straightforward explanation of that.

bradheintz