给定一个复杂的对象层次结构(幸运地不包含循环引用),我如何实现支持各种格式的序列化?我不是来讨论实际实施的。相反,我正在寻找可能派上用场的设计模式的提示。
更确切地说:我使用Ruby,我希望解析XML和JSON数据来构建一个复杂的对象层次结构。此外,应该可以将这个层次结构序列化为JSON、XML,可能还有HTML。
为此,我可以使用Builder模式吗?在上述任何一种情况下,我都有某种结构化数据--无论是在内存中还是在文本中--我想用这些数据来构建其他的东西。
我认为最好将序列化逻辑与实际业务逻辑分开,这样以后我就可以轻松地支持多种XML格式。
发布于 2011-07-18 09:07:43
最后,我创建了一个基于Builder和策略模式的解决方案。我使用Builder模式提取解析,并将逻辑构建到自己的类中。这允许我分别轻松地添加新的解析器和构建器。我使用策略模式来实现单独的解析和构建逻辑,因为这个逻辑取决于我的输入和输出格式。
下图显示了我的解决方案的UML图。
下面的清单显示了我的Ruby实现。实现有点琐碎,因为我正在构建的对象相当简单。对于那些认为这段代码臃肿和Java-ish的人来说,我认为这实际上是一个很好的设计。我承认,在这种微不足道的情况下,我可以直接将构建方法构建到我的业务对象中。然而,我并不是在我的其他应用程序中构建成果,而是相当复杂的对象,因此分离解析、构建和业务逻辑似乎是个好主意。
require 'nokogiri'
require 'json'
class Fruit
attr_accessor :name
attr_accessor :size
attr_accessor :color
def initialize(attrs = {})
self.name = attrs[:name]
self.size = attrs[:size]
self.color = attrs[:color]
end
def to_s
"#{size} #{color} #{name}"
end
end
class FruitBuilder
def self.build(opts = {}, &block)
builder = new(opts)
builder.instance_eval(&block)
builder.result
end
def self.delegate(method, target)
method = method.to_sym
target = target.to_sym
define_method(method) do |*attrs, &block|
send(target).send(method, *attrs, &block)
end
end
end
class FruitObjectBuilder < FruitBuilder
attr_reader :fruit
delegate :name=, :fruit
delegate :size=, :fruit
delegate :color=, :fruit
def initialize(opts = {})
@fruit = Fruit.new
end
def result
@fruit
end
end
class FruitXMLBuilder < FruitBuilder
attr_reader :document
def initialize(opts = {})
@document = Nokogiri::XML::Document.new
end
def name=(name)
add_text_node(root, 'name', name)
end
def size=(size)
add_text_node(root, 'size', size)
end
def color=(color)
add_text_node(root, 'color', color)
end
def result
document.to_s
end
private
def add_text_node(parent, name, content)
text = Nokogiri::XML::Text.new(content, document)
element = Nokogiri::XML::Element.new(name, document)
element.add_child(text)
parent.add_child(element)
end
def root
document.root ||= create_root
end
def create_root
document.add_child(Nokogiri::XML::Element.new('fruit', document))
end
end
class FruitJSONBuilder < FruitBuilder
attr_reader :fruit
def initialize(opts = {})
@fruit = Struct.new(:name, :size, :color).new
end
delegate :name=, :fruit
delegate :size=, :fruit
delegate :color=, :fruit
def result
Hash[*fruit.members.zip(fruit.values).flatten].to_json
end
end
class FruitParser
attr_reader :builder
def initialize(builder)
@builder = builder
end
def build(*attrs, &block)
builder.build(*attrs, &block)
end
end
class FruitObjectParser < FruitParser
def parse(other_fruit)
build do |fruit|
fruit.name = other_fruit.name
fruit.size = other_fruit.size
fruit.color = other_fruit.color
end
end
end
class FruitXMLParser < FruitParser
def parse(xml)
document = Nokogiri::XML(xml)
build do |fruit|
fruit.name = document.xpath('/fruit/name').first.text.strip
fruit.size = document.xpath('/fruit/size').first.text.strip
fruit.color = document.xpath('/fruit/color').first.text.strip
end
end
end
class FruitJSONParser < FruitParser
def parse(json)
attrs = JSON.parse(json)
build do |fruit|
fruit.name = attrs['name']
fruit.size = attrs['size']
fruit.color = attrs['color']
end
end
end
# -- Main program ----------------------------------------------------------
p = FruitJSONParser.new(FruitXMLBuilder)
puts p.parse('{"name":"Apple","size":"Big","color":"Red"}')
p = FruitXMLParser.new(FruitObjectBuilder)
puts p.parse('<fruit><name>Apple</name><size>Big</size><color>Red</color></fruit>')
p = FruitObjectParser.new(FruitJSONBuilder)
puts p.parse(Fruit.new(:name => 'Apple', :color => 'Red', :size => 'Big'))
发布于 2011-07-18 06:09:12
每当有人说他们想使用不同的算法进行相同的操作,并选择在运行时使用哪种算法时,战略模式总是浮现在脑海中。不同类型的序列化(XML、JSON、二进制等等)都是将对象转换为纯数据、更可移植的结构的不同策略。这似乎适用于你的情况。
发布于 2011-07-18 11:48:19
看看面向模式的软件体系结构,反射模式;我认为它应该给出一些关于如何实现这样一个结构的好提示。
https://stackoverflow.com/questions/6733317
复制相似问题