偷天换日

class Module
# Declare an attribute accessor with an initial default return value.
#
# To give attribute <tt>:age</tt> the initial value <tt>25</tt>:

#   class Person
#     attr_accessor_with_default :age, 25
#   end
#
#   some_person.age
#   => 25
#   some_person.age = 26
#   some_person.age
#   => 26
#
# To give attribute <tt>:element_name</tt> a dynamic default value, evaluated
# in scope of self:
#
#   attr_accessor_with_default(:element_name) { name.underscore }
#
def attr_accessor_with_default(sym, default = nil, &block)
raise 'Default value or block required' unless !default.nil? || block
    define_method(sym, block_given? ? block : Proc.new { default })
module_eval(<<-EVAL, __FILE__, __LINE__)
      def #{sym}=(value)                        # def age=(value)
        class << self; attr_reader :#{sym} end  #   class << self; attr_reader :age end
        @#{sym} = value                         #   @age = value
      end                                       # end
EVAL
end
end

 

读ActiveSupport代码时看到上面这个方法,即定义有缺省值的属性。看了n遍加上动手注解掉class << self ...这句后终于理解了这个缺省属性的实现方法:

1 define_method(...)定义了一个名为sym的方法,相当于一个读属性,在对象上调用 sym方法时返回default值或者block的返回值

2 在对象上调用#{sym}=方法c时lass << self; attr_reader:#{sym} end先为对象创建一个虚拟类,在该类中用attr_reader覆盖define_method中定义的sym方法,直接返回@{#sym}变量。然后再把value赋给@#{sym}

 

来看一个例子

require 'active_support'

class NiceGirl

include ActiveSupport

attr_accessor :real_age

attr_accessor_with_default(:age) {real_age}

end

nice_girl = NiceGirl.new

nice_girl.real_age = 18

puts "here calls real_age #{nice_girl.age}"

nice_girl.age = 16

puts "here returns @age #{nice_girl.age}"

puts "here returns real_age #{nice_girl.real_age}"

下面是输出

here calls real_age 18

here returns @age 16

here returns real_age 18

即在调用属性setter时动态改变了属性getter的定义。这种做法的确相当灵活,也只有ruby这种动态语言能做得出来。

你可能感兴趣的:(偷天换日)