How Jbuilder Works

jbuilder是Rails开发者最常用的gem之一了,自不必多说,它可是 API 开发中的利器,灵活的DSL语法与Rails和Ruby本身很相配。本文就要探知一下,Jbuilder的实现原理是什么,以便我们日后,更加深入的使用和开发Rails API应用。

结构

首先打开Jbuilder的lib目录,也就是主目录,我们会看到以下文件结构:

├── generators
├── jbuilder
└── jbuilder.rb

我们可以看到,lib下由 一个 jbuilder.rb的入口文件和jbuildergenerators 两个目录构成的。其中generators是注册在Rails中的生成器,因为本文主要介绍的是jbuilder的工作原理,所以我们就把目光放在 jbuilder目录中。

├── jbuilder
│   ├── dependency_tracker.rb
│   ├── errors.rb
│   ├── jbuilder.rb
│   ├── jbuilder_template.rb
│   ├── key_formatter.rb
│   └── railtie.rb
└── jbuilder.rb
jbuilder类图

在Jbuilder 的实现中,我们基本上可以将它的功能部分分为:Jbuilder模块,template模块和dependency模块。

下面我们就来依次介绍它们。

Jbuilder

# lib/jbuilder/jbuiler.rb
Jbuilder = Class.new(begin
  require 'active_support/proxy_object'
  ActiveSupport::ProxyObject
rescue LoadError
  require 'active_support/basic_object'
  ActiveSupport::BasicObject
end)

# lib/jbuilder.rb
require 'jbuilder/jbuilder'
....
class Jbuilder
  @@key_formatter = KeyFormatter.new
  @@ignore_nil    = false
  .....
end

Jbulder类本身是继承自 ActiveSupport::ProxyBasic类,同时jbuilder使用了打开类的方式,去扩充现有jbuilder类的方法。 继承ProxyBasic的主要作用就是作为一个洁净室,让继承自它的JbuilderTemplate对象可以通过 method_missnig 去处理非定义方法的调用。我们这也就道出了Jbuilder及JbuilderTemplate都是使用set! 方法去代理所有未定义的方法。

  alias_method :method_missing, :set!
  private :method_missing

最后在set!方法会将数据保存在 @attributes 属性中,之后的操作也都是这样的步骤,直到在template_bundler中调用了jbuilder的target方法 将@attributes转换成json数据。

set!方法最后会调用_write将键值对保存到@attributes属性中,其中Key还会经过@key_formatter进行格式化。

# lib/jbuilder.rb
def _write(key, value)
  @attributes = {} if _blank?
  @attributes[_key(key)] = value
end

def _key(key)
  @key_formatter.format(key) # 每次在调用json.key_format!是都会重新的实例化一个KeyFormatter。
end

在下面介绍的Template中你就会看到模板处理器最后会调用 json.target!方法,然后进行渲染。

# 将Hash 转换为json字符串返回。
def target!
  ::MultiJson.dump(@attributes)
end

Template

Jbuilder本身就是一个Rails的Railtie,并且它在active_view加载完成后,注册了 jbuilder 模板处理器 register_template_handler ,active_view中规定如果要注册 模板的需要一个能够响应call方法的处理类,并且call方法要接受一个template对象,返回一个字符串对象,然后action_view会将返回的字符串进行eval运行。

# lib/jbuilder/jbuilder_template.rb
class JbuilderHandler
  cattr_accessor :default_format
  self.default_format = Mime::JSON

  def self.call(template)
    # this juggling is required to keep line numbers right in the error
    %{__already_defined = defined?(json); json||=JbuilderTemplate.new(self); #{template.source}
      json.target! unless (__already_defined && __already_defined != "method")}
  end
end

call方法返回的字符串,是用";"分隔的多条语句,模板的代码也被插入在其中。其中的json是从JbuilderTemplate类中初始化出来的,这样jbuilder中 json receiver 就是 JbuilderTemplate的实例了。

JbuilderTemplate 也是继承与Jbuilder类,它在其中扩充了Jbuilder的功能方法有:

  • partial!
  • array!
  • cache!
  • cache_if!

之所以将这几个方法单独放在JbuilderTemplate中,是因为需要使用ViewContext对象的render方法,去渲染其他的模板。

#lib/jbuilder/jbuilder_template.rb
  def _render_partial(options)
    options[:locals].merge! json: self 
    @context.render options
  end

在Railtie中 定义模板处理器

# lib/jbuilder/railtie.rb
...
initializer :jbuilder do |app|
  ActiveSupport.on_load :action_view do
    # 向View中注册处理器
    ActionView::Template.register_template_handler :jbuilder, JbuilderHandler
    # 解决依赖问题
    require 'jbuilder/dependency_tracker'
  end
end

Dependency

jbuilder 注册template,同时也使用了,action view的 dependency_tracker 去管理template中对外依赖。

jbuilder/dependency_tracker.rb 类首先继承自 ::ActionView::DependencyTracker 然后对其核心的dependencies 方法进行了重载,让其支持jbuilder自己的规范方法。

具体的实现就是,使用正则表达式去在template字符串中匹配,jbuilder自己指定的规则。

# lib/jbuilder/dependency_tracker.rb      
# Matches:
      #   json.partial! "messages/message"
      #   json.partial!('messages/message')
      #
      DIRECT_RENDERS = /
        \w+\.partial!     # json.partial!
        \(?\s*            # optional parenthesis
        (['"])([^'"]+)\1  # quoted value
      /x

      # Matches:
      #   json.partial! partial: "comments/comment"
      #   json.comments @post.comments, partial: "comments/comment", as: :comment
      #   json.array! @posts, partial: "posts/post", as: :post
      #   = render partial: "account"
      #
      INDIRECT_RENDERS = /
        (?::partial\s*=>|partial:)  # partial: or :partial =>
        \s*                         # optional whitespace
        (['"])([^'"]+)\1            # quoted value
      /x

      def dependencies
        direct_dependencies + indirect_dependencies + explicit_dependencies
      end

      private

      def direct_dependencies
        source.scan(DIRECT_RENDERS).map(&:second)
      end

      def indirect_dependencies
        source.scan(INDIRECT_RENDERS).map(&:second)
      end

我们在Rails View中使用 render template 路径中不带扩展名就是因为,扩展名已经注册到register_tracker方法中了,所以在render 的时候,action view 会自动的在所以注册的tracker中寻找匹配的文件。

其他

KeyFormatter

KeyFormatter非常简单,就是将传入的key按照上一次设置好的格式进行格式化。它的具体实现方法就是。

在json对象上的key_format! 方法传入的参数,都会传入到KeyFormatter的构造方法中。

# 传入Proc对象
json.key_format! ->(key){ "_" + key }
# 或是 Symbol Hash
json.key_format! camelize: :lower
# lib/jbuilder/key_formatter.rb
class KeyFormatter
  def initialize(*args)
    @format = {}
    @cache = {}
    ...
  end
end

在经过format方法判断传入的是Proc还是Symbol ,Proc的的话就执行它,Symbol就使用send方法调用,并将参数传入。

并且还会将已经格式化过的key缓存下来,避免了相同key多次调用的开销。

def format(key)
  @cache[key] ||= @format.inject(key.to_s) do |result, args|
    func, args = args
    if ::Proc === func
      func.call result, *args
    else
      result.send func, *args
    end
  end
end

Errors

Jbuilder 仅定义了一个异常类,就是NullError 用于处理为空异常的。

#lib/jbuilder/errors.rb
class NullError < ::NoMethodError
  def self.build(key)
    message = "Failed to add #{key.to_s.inspect} property to null object"
    new(message)
  end
end

总结

Jbuilder使用上非常简洁灵活的DSL结构,其实核心就是通过 method_missing 来将数据存放在一个Hash中,最后再将中其转换成JSON数据,在配合一下Ruby元编程的技巧,比如:打开类,动态派发等。其设计上的方式还是很有借鉴意义的。不愧是Rails官方出品。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,530评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 86,403评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,120评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,770评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,758评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,649评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,021评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,675评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,931评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,659评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,751评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,410评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,004评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,969评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,042评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,493评论 2 343

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,594评论 18 139
  • Spring Boot 参考指南 介绍 转载自:https://www.gitbook.com/book/qbgb...
    毛宇鹏阅读 46,733评论 6 342
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,434评论 25 707
  • 这是一个高速发展的时代,这是一个信息网络化的时代,这是一个物质条件不断丰富的时代,这是一个需求多元和表达多元的时代...
    博雅大师兄阅读 369评论 4 4
  • 当你感到人生迷茫的时候,干货们往往想要从成功人士那里获取经验,或者从书本上找到人生哲理,就像我,今年21岁了...
    娟儿娟儿阅读 715评论 2 2