Ruby中嵌套对象转换成json的方法
json由于其数据结构简单便利,已逐渐成为了互联网上的主流数据交换的数据格式。
在讨论嵌套对象(nested object)的json转换方法之前,我们先看简单的ruby json转换。
首先,ruby对象转换为json字符串:
class obj1
def initialize(var1)
@var1 = var1
end
def to_json(*a)
{
"json_class" => self.class,
"data" => {"var1" => @var1}
}.to_json(*a)
end
def self.json_create(json_str)
new(json_str["data"]["var1"])
end
end
obj1 = obj1.new("i am obj1")
#obj1 to json string
json_str = obj1.to_json
puts "json string of obj1 = #{json_str}"
#json string to obj1
obj11 = json.parse(json_str)
puts "ob1 from json string = #{obj11.var1}"
上面代码我们可以看到,ruby与json string之间的转换,关键有三个点:
#引入json库,才能有下面两个方法,json是通过open class的方式,给hash对象加上了to_json(*a)方法,关于ruby的open class参考支持open class特性的编程语言中的开闭原则(open-closed principle)
1)require ‘json'
#定义对象转为json string的to_json(*a)方法,其实现是使用hash对象的to_json(*a)方法
2)def to_json(*a)
#定义从json string构造对象的json_create方法,此方法是类方法
3)def self.json_create(json_str)
上面三点是ruby中实现json string互相转换的基本要求。
代码运行结果为:
json string of obj1 = {"json_class":"obj1","data":{"var1":"i am obj1"}}
ob1 from json string = i am obj1
现在我们来看嵌套对象的json string转换:
#!/usr/local/ruby/bin/ruby
require 'json'
class obj1
def initialize(var1)
@var1 = var1
end
def to_json(*a)
{
"json_class" => self.class,
"data" => {"var1" => @var1}
}.to_json(*a)
end
def self.json_create(json_str)
new(json_str["data"]["var1"])
end
attr_reader :var1
end
class obj2
def initialize(var2)
@var2 = var2
end
def to_json(*a)
{
"json_class" => self.class,
"data" => {"var2" => @var2}
}.to_json(*a)
end
def self.json_create(json_str)
new(json_str["data"]["var2"])
end
attr_reader :var2
end
class obj
def initialize(obj1, obj2)
@obj1 = obj1
@obj2 = obj2
end
def to_json(*a)
{
"json_class" => self.class,
"data" => {"obj1" => @obj1.to_json, "obj2" => @obj2.to_json}
}.to_json(*a)
end
def self.json_create(json_str)
new(json_str["data"]["obj1"], json_str["data"]["obj2"])
end
def to_s
"hi, i am obj"
end
attr_reader :obj1, :obj2
end
obj1 = obj1.new("i am obj1")
obj2 = obj2.new("i am obj2")
obj = obj.new(obj1,obj2)
obj_json_str = obj.to_json
puts "json string of obj = #{obj_json_str}"
obj_1 = json.parse(obj_json_str)
puts "obj_1 from json string , obj1.class = #{obj_1.obj1.class}, obj2.class = #{obj_1.obj2.class}"
上面代码中,嵌套对象我们惯性思维,是先将对象自己转换为json string:
"data" => {"obj1" => @obj1.to_json, "obj2" => @obj2.to_json}
上面代码输出:
json string of obj = {"json_class":"obj","data":{"obj1":"{\"json_class\":\"obj1\",\"data\":{\"var1\":\"i am obj1\"}}","obj2":"{\"json_class\":\"obj2\",\"data\":{\"var2\":\"i am obj2\"}}"}}
obj_1 from json string , obj1.class = string, obj2.class = string
我们注意到,被嵌套的对象转换为json string后,多了一个反斜杠 \ :
json string of obj = {"json_class":"obj","data":{"obj1":"{\"json_class\":\"obj1\",\"data\":{\"var1\":\"i am obj1\"}}","obj2":"{\"json_class\":\"obj2\",\"data\":{\"var2\":\"i am obj2\"}}"}}
且,json string转换后,obj对象中嵌套的对象obj1和obj2,其类型都为string,而不是期望的obj1和obj2类型
obj_1 from json string , obj1.class = string, obj2.class = string
实际上,这里是惯性思维害人,被嵌套的对象,不需要调用其to_json方法。
因此将obj类的to_json代码:
def to_json(*a)
{
"json_class" => self.class,
"data" => {"obj1" => @obj1.to_json, "obj2" => @obj2.to_json}
}.to_json(*a)
end
修正为:
def to_json(*a)
{
"json_class" => self.class,
"data" => {"obj1" => @obj1, "obj2" => @obj2}
}.to_json(*a)
end
然后,运行代码,可以看到预期的输出:
json string of obj = {"json_class":"obj","data":{"obj1":{"json_class":"obj1","data":{"var1":"i am obj1"}},"obj2":{"json_class":"obj2","data":{"var2":"i am obj2"}}}}
obj_1 from json string = {"json_class"=>"obj", "data"=>{"obj1"=>#, "obj2"=>#}}
上一篇: 21个你应该知道的Ruby编程技巧
下一篇: [C]最大公约数和最小公倍数
推荐阅读
-
php中输出json对象的值(实现方法)
-
js中json字符串转json对象的方法(提取json格式的数据)
-
C#实现json格式转换成对象并更换key的方法
-
基于python list对象中嵌套元组使用sort时的排序方法
-
PHP中把对象数组转换成普通数组的方法
-
php实现数组中索引关联数据转换成json对象的方法
-
JSON PHP中,Json字符串反序列化成对象/数组的方法
-
浅谈jQuery中的$.extend方法来扩展JSON对象
-
Python的Bottle框架中返回静态文件和JSON对象的方法
-
c# json转换成dynamic对象,然后在dynamic对象中动态获取指定字符串列表中的值