ActiveSupport::Concern 和 gem 'name_of_person'(300✨) 的内部运行机制分析

理解ActiveRecord::Concern:

参考:include和extend的区别:

https://www.cnblogs.com/chentianwei/p/9408963.html

 

传统的模块看起来像:

module M
  def self.included(base)
    # base(一个类)扩展了一个模块"ClassMethods", base的类方法就包含了"ClassMethods"模块中的方法。
    base.extend ClassMethods
    # base添加了一个:disabled方法。
    base.class_eval do
      scope :disabled, -> { where(disabled: true) }
    end
  end

  module ClassMethods
    ...
  end
end

 

使用ActiveSupport::Concern:

 

require 'active_support/concern'

module M
  # M扩展了模块Concern,可以使用Concern的方法。
  extend ActiveSupport::Concern

  # 当M被一个类包括后,这个类就可以使用块内的方法了。
  included do
    scope :disabled, -> { where(disabled: true) }
  end

  # 当M被一个类包括后,这个类的类方法就扩展了,👇的方法就作为类方法使用。
  class_methods do
    ...
  end
end


 

gem 'name_of_person'

一个小的gem,为英文网站用户的注册名字添加了很多调用的方法。

https://github.com/basecamp/name_of_person/tree/master/lib/name_of_person

  1. 加载了gem后,
  2. ActiveRecord::Base包含了模块HasPersonName, 就可以使用lib/name_of_person/has_person_name.rb中的方法:类方法has_person_name.
  3. 在Rails app中, app/model/user.rb, 使用has_person_name方法后,就include包含了模块Assignable
  4. User的实例就新增了2个实例方法,这两个方法会调用模块PersonName中的方法
    • @user.name=: 调用PersonName.full(name)方法,@user的first_name, last_name属性被分配值。
    • @user.name:  返回一个PersonName.new对象,这个对象可以使用:
      • full | initials | familiar 等定义在模块PersonName中的方法。
      • first | last

使用方法:

1 . User类必须包括first_name, last_name2个属性,添加validates :first_name, :last_name, presence: true

2. 当实例化一个@user时,代码内部调用name= 方法为first_name, last_name属性分配值!

(这里不是很理解,是否是devise这个gem,当发现必须验证first_name, last_name后,自动调用name=方法?)

3. 之后通过@user.name.xxx就可以使用不同的名和姓的组合。

 

分析:先看三张图:

图2

 

 

图3:

 

 

 

@user.name的内部运行机制:

首先是一个判断:

if  @user.first_name
  NameOfPerson::PersonName.new(@user.first_name, @user.last_name)
end

如果first_name存在,则新增一个PersonName对象,调用initialize方法

    def initialize(first, last = nil)
      raise ArgumentError, "First name is required" unless first.present?
      @first, @last = first, last
      super full
    end

然后调用full这个方法,进行if判断

    def full
      @full ||= last.present? ? "#{first} #{last}" : first
    end
分析:
如果@user.last_name存在(last.present?),则 把@user的两个name属性合并,并分配给@full对象。

最后返回一个PersonName对象实例, 内部包括@first, @full, 及@last(根据@user决定是否存在)

 

@user.name = "Dav Tom"内部运行分析:

    def name=(name)
      full_name = NameOfPerson::PersonName.full(name)
      self.first_name, self.last_name = full_name.try(:first), full_name.try(:last)
    end

 

首先:调用模块PersonName的类方法full。

  • 把传入的字符串参数分成first, last变量
  • 如果first变量存在,则新建一个PersonName对象
  • 之后的分析和@ueser.name相同。 
    def self.full(full_name)
      first, last = full_name.to_s.strip.split(/\s+/, 2)
      new(first, last) if first.present?
    end

 

posted @ 2018-10-22 11:28  Mr-chen  阅读(322)  评论(0编辑  收藏  举报