在define_method定义的方法上调用super

时间:2010-01-03 13:37:04

标签: ruby metaprogramming super superclass

我创建了一个Model类,我根据在User中调用的方法(属性)定义方法(继承自Model)。问题是我无法覆盖define_method定义的方法,并调用super来传递给定义的方法。我想这是因为定义的方法被添加到User本身,而不是Model,因此它实际上没有超类中的方法(即Model)。

我想这样做的原因是因为大多数属性应该直接保存到数据库中,而某些属性(如密码)需要一些额外的处理。

class Model
  def self.attribute(name)
    define_method(name) do
      self
    end
  end  
end

class User < Model
  attribute :password
end

class User2 < Model
  attribute :password

  def password
    super
  end
end

@user = User.new
puts @user.password # => <User:0x00000100845540>

@user2 = User2.new
puts @user2.password
# define_super.rb:17:in `password': super: no superclass method 
# `password' for #<User2:0x00000100845578> (NoMethodError)
# from define_super.rb:25:in `<main>'

有什么办法可以改变代码来实现这个目的吗?我需要一种方法来覆盖动态创建的方法。

3 个答案:

答案 0 :(得分:10)

superclass上定义方法:

class Model
  def self.attribute(name)
    superclass.send :define_method, name do
      self
    end
  end  
end

答案 1 :(得分:3)

Rails处理这个问题的方法是有多种获取属性的方法。其中一个(按惯例)从未被覆盖,因此可以在您定义的方法中使用它:

# This method is never overridden, but also rarely used as a public method
def[](key)
  # Returns value of `key` attribute
end

# This is the effective default implementation of an attribute
def att1
  self[:att1]
end

# This shows how you can have custom logic but still access the underlying value
def att2
  self[:att2] unless self[:att2].empty?
end

答案 2 :(得分:0)

使用superclass.send :define_method可能会导致某些意外行为,因为该方法将可用于每个Model子类。

本文https://thepugautomatic.com/2013/07/dsom/

对此进行了详细说明

因此,在本文之后,您可以像这样定义attribute方法

class Model
  MODULE_NAME = :DynamicAttributes

  def self.attribute(name)
    if const_defined?(MODULE_NAME, _search_ancestors = false)
      mod = const_get(MODULE_NAME)
    else
      mod = const_set(MODULE_NAME, Module.new)
      include mod
    end

    mod.module_eval do
      define_method(name) do
        self
      end
    end
  end
end