Generate accessors for an hash

A

Andrea Gianarro

Hello, I need to generate read and write accessors for all the keys in
an hash.
Say I have an attribute named my_attribute, I need "my_attribute" and
"my_attribute=" methods to read and write it to
@attributes[my_attribute].

I tried this code in the class constructor:

for a in all_attributes
method_name = a.name
self.class.class_eval do
define_method method_name do
@attributes[method_name]
end
define_method method_name+"=" do |val|
@attributes[method_name] = val
end
end
end

It generates the method for the class, but all the methods read and
write to the last attribute in the all_attributes list. That's
strange...
Thank you,

Andrea Gianarro
 
J

James Coglan

[Note: parts of this message were removed to make it a legal post.]
I tried this code in the class constructor:

for a in all_attributes
method_name = a.name
self.class.class_eval do
define_method method_name do
@attributes[method_name]
end
define_method method_name+"=" do |val|
@attributes[method_name] = val
end
end
end

It generates the method for the class, but all the methods read and
write to the last attribute in the all_attributes list. That's
strange...



Ah, the strange joy that is closures. Basically, those 'define_method' calls
allow the code block to retain access to whatever variables were in scope
when the methods are defined. So, when those methods actually run, the
variable 'method_name' will have the last value in the loop -- each closure
references a single variable, it does not copy its value.

You might be better off with method_missing

class Hash
def method_missing(name, value = nil)
name = name.to_s
if name =~ /=$/
self[name.sub(/=$/, '')] = value
else
return self[name]
end
end
end
 
V

Vassilis Rizopoulos

Andrea said:
Hello, I need to generate read and write accessors for all the keys in
an hash.
Wouldn't a simpler solution be to use OpenStruct?

require 'ostruct'

hash={"my_attribute"=>"value"}
s=OpenStruct.new(hash)
puts s.my_attribute
s.my_attribute="new"
=> "new"
puts s.my_attribute
=> nil

And you get the freebie of creating new keys from the setter:

s.new_attribute="better"
=> "better"
puts s.new_attribute

Cheers,
V.-
 
A

Andrea Gianarro

James said:
You might be better off with method_missing

class Hash
def method_missing(name, value = nil)
name = name.to_s
if name =~ /=$/
self[name.sub(/=$/, '')] = value
else
return self[name]
end
end
end


Thank you very much, that's exactly what I did. Strange ruby behaviour
anyway. :p
bye,

Andrea Gianarro
 

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

Forum statistics

Threads
473,969
Messages
2,570,161
Members
46,710
Latest member
bernietqt

Latest Threads

Top