5

以下是Why's Poignant Guide to Ruby Chapter 6中的一段 Ruby 代码片段,他试图在其中演示 Ruby 中的元编程:

# Get a metaclass for this class
def self.metaclass; class << self; self; end; end

我对 Ruby 不太熟悉,但这就是它在扩展形式中的样子吗?

def self.metaclass
    def self.self
    end
end

至少我是这么理解的。但是,它仍然不完全理解这段代码的作用。它的目的是什么?

进一步在代码中,为什么添加这个:

arr.each do |a|
   metaclass.instance_eval do
     define_method( a ) do |val|
       @traits ||= {}
       @traits[a] = val
     end
   end
 end

如果我理解正确,这段代码会使用给定的名称和值向@traits 添加一个新值。那是对的吗?

感谢您的帮助,这是给我带来麻烦的完整源代码,任何想看的人都可以:

# The guts of life force within Dwemthy's Array
class Creature

# Get a metaclass for this class
def self.metaclass; class << self; self; end; end

# Advanced metaprogramming code for nice, clean traits
def self.traits( *arr )
 return @traits if arr.empty?

 # 1. Set up accessors for each variable
 attr_accessor *arr

 # 2. Add a new class method to for each trait.
 arr.each do |a|
   metaclass.instance_eval do
     define_method( a ) do |val|
       @traits ||= {}
       @traits[a] = val
     end
   end
 end

 # 3. For each monster, the `initialize' method
 #    should use the default number for each trait.
 class_eval do
   define_method( :initialize ) do
     self.class.traits.each do |k,v|
       instance_variable_set("@#{k}", v)
     end
   end
 end

end

# Creature attributes are read-only
traits :life, :strength, :charisma, :weapon
end

在使用中:

class Dragon < Creature
    life( 1340 )     # tough scales
    strength( 451 )  # bristling veins
    charisma( 1020 ) # toothy smile
    weapon( 939 )    # fire breath
end
4

2 回答 2

5
class Foo
  def self.bar    # Create a method invoked by Foo.bar instead of Foo.new.bar
    42            # the return value of this method (value of last expression)
  end
end


class Foo
  def self.jim    # Another method on the class itself
    class << self # Change the 'self' to be the metaclass of the current object
      self        # Evaluate the current 'self' as the 'return value' of 
    end           # class<<self…end; and since this is the last expression in
  end             # the method, its value is the return value for the method
end

简而言之:您所看到的定义了一个以类本身命名的方法metaclassCreature而不是实例)。当您运行此方法时,它会找到元类Creature并返回它。

阅读“网络”以了解对象的“元类”是什么。

于 2012-04-11T03:25:08.393 回答
2

在扩展形式中,它看起来完全一样:

def self.metaclass
  class << self
    self
  end
end

注意它只是返回self,因为它是在元类的上下文中评估的,实际上class << selfself元类。

随着define_singleton_method之类的引入,您需要直接访问元类的情况变得非常少。

这一切都非常复杂,似乎是“一切都是对象”设计原则的结果。

于 2012-04-11T03:30:49.697 回答