在嵌套对象中使用自定义to_json方法

我有一个使用Ruby标准库中的Set类的数据结构。 我希望能够将我的数据结构序列化为JSON字符串。

默认情况下,将序列化设置为数组:

>> s = Set.new [1,2,3] >> s.to_json => "[1,2,3]" 

在尝试反序列化之前哪个是好的。

所以我定义了一个自定义的to_json方法:

 class Set def to_json(*a) { "json_class" => self.class.name, "data" => { "elements" => self.to_a } }.to_json(*a) end def self.json_create(o) new o["data"]["elements"] end end 

哪个效果很好:

 >> s = Set.new [1,2,3] >> s.to_json => "{\"data\":{\"elements\":[1,2,3]},\"json_class\":\"Set\"}" 

直到我把Set放入哈希或其他东西:

 >> a = { 'set' => s } >> a.to_json => "{\"set\":[1,2,3]}" 

知道为什么当Set嵌套在另一个对象中时我的自定义to_json没有被调用?

第一个块用于Rails 3.1(旧版本几乎相同); 第二个块用于标准的非Rails JSON。 如果tl;博士,请跳到最后。


你的问题是Rails这样做:

 [Object, Array, FalseClass, Float, Hash, Integer, NilClass, String, TrueClass].each do |klass| klass.class_eval <<-RUBY, __FILE__, __LINE__ # Dumps object in JSON (JavaScript Object Notation). See www.json.org for more info. def to_json(options = nil) ActiveSupport::JSON.encode(self, options) end RUBY end 

active_support/core_ext/object/to_json.rb 。 特别是,它将Hash的to_json方法更改为ActiveSupport::JSON.encode调用。

然后,看看ActiveSupport::JSON::Encoding::Encoder ,我们看到:

 def encode(value, use_options = true) check_for_circular_references(value) do jsonified = use_options ? value.as_json(options_for(value)) : value.as_json jsonified.encode_json(self) end end 

因此所有Rails JSON编码都通过as_json 。 但是,你没有为Set定义自己的as_json ,你只是设置了to_json并且当Rails忽略它不使用的东西时会感到困惑。

如果你设置了自己的Set#as_json

 class Set def as_json(options = { }) { "json_class" => self.class.name, "data" => { "elements" => self.to_a } } end end 

然后你会得到你在Rails控制台和Rails中所追求的东西:

 > require 'set' > s = Set.new([1,2,3]) > s.to_json => "{\"json_class\":\"Set\",\"data\":{\"elements\":[1,2,3]}}" > h = { :set => s } > h.to_json => "{\"set\":{\"json_class\":\"Set\",\"data\":{\"elements\":[1,2,3]}}}" 

请记住, as_json用于为JSON序列化准备对象,然后to_json生成实际的JSON字符串。 as_json方法通常返回简单的可序列化数据结构,例如Hash和Array,并且在JSON中具有直接类似物; 然后,一旦你有一些结构类似JSON的东西, to_json用于将它序列化为线性JSON字符串。


当我们查看标准的非Rails JSON库时,我们会看到如下内容:

 def to_json(*a) as_json.to_json(*a) end 

猴子修补到基本类(符号,时间,日期,......)。 所以to_json通常用to_json来实现。 在这种环境中,我们需要为Set包含标准的to_json以及上面的as_json

 class Set def as_json(options = { }) { "json_class" => self.class.name, "data" => { "elements" => self.to_a } } end def to_json(*a) as_json.to_json(*a) end def self.json_create(o) new o["data"]["elements"] end end 

我们为解码器包含了json_create类方法。 一旦完成设置,我们就会在irb得到这样的结果:

 >> s = Set.new([1,2,3]) >> s.as_json => {"json_class"=>"Set", "data"=>{"elements"=>[1, 2, 3]}} >> h = { :set => s } >> h.to_json => "{"set":{"json_class":"Set","data":{"elements":[1,2,3]}}}" 

执行摘要 :如果你在Rails中,不要担心用to_json做任何事情, as_json就是你想玩的东西。 如果您不在Rails中,请在as_json实现大部分逻辑(尽管文档中有说明)并添加标准的to_json实现( def to_json(*a);as_json.to_json(*a);end )。

这是我为自定义类获取to_json方法的方法,它最有可能不包含to_a方法(最近已从Object类实现中删除)

使用self.included在模块中有一点魔力。 这是2006年关于module具有实例和类方法的非常好的文章http://blog.jayfields.com/2006/12/ruby-instance-and-class-methods-from.html

该模块旨在包含在任何类中,以提供无缝的to_jsonfunction。 它拦截attr_accessor方法而不是使用它自己的方法,以便对现有类进行最小的更改。

 module JSONable module ClassMethods attr_accessor :attributes def attr_accessor *attrs self.attributes = Array attrs super end end def self.included(base) base.extend(ClassMethods) end def as_json options = {} serialized = Hash.new self.class.attributes.each do |attribute| serialized[attribute] = self.public_send attribute end serialized end def to_json *a as_json.to_json *a end end class CustomClass include JSONable attr_accessor :b, :c def initialize b: nil, c: nil self.b, self.c = b, c end end a = CustomClass.new(b: "q", c: 23) puts JSON.pretty_generate a { "b": "q", "c": 23 } 

寻找同一问题的解决方案,我在Rails问题跟踪器上找到了这个错误报告。 除了关闭之外,我想它仍然发生在早期版本上。 希望它可以帮助。

https://github.com/rails/rails/issues/576