3

以下是代码片段:

模块:ActiveSupport::关注

    module ActiveSupport
      module Concern
        def self.extended(base)
          base.instance_variable_set("@_dependencies", [])
        end

        def append_features(base)
          if base.instance_variable_defined?("@_dependencies")
            base.instance_variable_get("@_dependencies") << self
            return false
          else
            return false if base < self
            @_dependencies.each { |dep| base.send(:include, dep) }
            super
            base.extend const_get("ClassMethods") if const_defined?("ClassMethods")
            base.send :include, const_get("InstanceMethods") if const_defined?("InstanceMethods")
            base.class_eval(&@_included_block) if instance_variable_defined?("@_included_block")
          end
        end

        def included(base = nil, &block)
          if base.nil?
            @_included_block = block
          else
            super
          end
        end
      end
    end

自定义模块:GeneralScopes

    module GeneralScopes
        extend ActiveSupport::Concern

        included do
          scope :reverse_order_by_date, :order => "date DESC"
          scope :recent, lambda { |count| reverse_order_by_date.limit(count) }
        end
    end

自定义类:用户

    class User < ActiveRecord::Base
      include GeneralScopes
    end

用法:

    User.recent(5) => Prints the recent five User instances available in database

在上面显示的代码片段的上下文中,我有以下问题:

1)包括(MODULE_NAME(s))include 是 Module 的私有实例方法,该方法由self隐式调用。那么self在类体中代表什么?在类的实例方法中, self代表当前对象。在类的类方法中, self代表类的Class对象。

include <MODULE_NAME>类主体中的这种语法是如何工作的?

2) 扩展(MODULE_NAME(s))

引自 David Flanagan 和 Yukihiro Matsumoto(Oreilly)的“The Ruby Programming Language”,第 7 章类和模块

Object.extend.This方法将指定模块的实例方法或模块的实例方法变成接收者对象的单例方法。(如果接收者对象是Class实例,则接收者的方法成为该类的类方法)

在上面显示的GeneralScopes模块的上下文中,我无法理解这一点。

3)

    module GeneralScopes
        extend ActiveSupport::Concern

        included do
          scope :reverse_order_by_date, :order => "date DESC"
          scope :recent, lambda { |count| reverse_order_by_date.limit(count) }
        end
    end

我假设模块 GeneralScopes 正在调用 ActiveSupport::Concern 模块的实例方法“included(base = nil, &block)”。从模块主体调用实例方法是如何工作的?哪个对象在什么时候充当接收者

     included do
          scope :reverse_order_by_date, :order => "date DESC"
          scope :recent, lambda { |count| reverse_order_by_date.limit(count) }
     end

被执行?

4) ActiveSupport::Concern 模块的实例方法 "included(base = nil, &block)"

        def included(base = nil, &block)
          if base.nil?
            @_included_block = block
          else
            super
          end
        end 

这里使用了super。这个super在 User 类或 ActiveSupport::Concern 模块的上下文中工作?执行super时,控件转到哪个类?

如果有人能让我理解所执行的控制流或向我指出任何相关资源来解释与我的问题相关的概念,那将是一个很大的帮助。

4

1 回答 1

5

1)include使模块的方法可用作类的实例方法extend而使模块的方法可用作类方法。所以self指的是实例或类:

module M
  def foo
    puts self
  end
end

class A
  include M
end

A.new.foo # => #<A:0x007fcefa18e270> ('self' refers to the instance)

class B
  extend M
end

B.foo # => B ('self' refers to the class)

2)extend ActiveSupport::Concern使ActiveSupport::Concern's 方法可用,GeneralScopes以便您可以调用该included方法。

3) 块在被用户包含时被调用的( ) 内分配@_included_block和评估。从文档append_featuresbase.class_eval(&@_included_block)GeneralScopes

当这个模块包含在另一个模块中时,Rubyappend_features在这个模块中调用,并将其传递给 mod 中的接收模块。

4)super指的​​是ActiveSupport::Concern' 的父母,所以Module'included被称为。

于 2012-06-28T13:21:48.427 回答