如果有多个采用以下形式的 BinData 记录,下面是一些示例:
class DebugInfo < BinData::Record
endian :little
int32 :num
array :data, :type => :debug_log, initial_length: :num
end
class GoalInfo < BinData::Record
endian :little
int32 :num
array :data, :type => :goal, initial_length: :num
end
class PackageInfo < BinData::Record
endian :little
int32 :num
array :data, :type => :package, initial_length: :num
end
所有这些本质上都是相同的,它们只是使用不同类型的对象创建一个数组。有没有办法只制作其中之一并以某种方式将我希望它读取到数组中的对象类型传递给它?
module Info
@base_classes = {}
def self.[](type)
@base_classes[type] ||= Class.new(BinData::Record) {
endian :little
int32 :num
array :data, :type => type, initial_length: :num
}
end
end
然后
class DebugInfo < Info[:debug_log]
end
class GoalInfo < Info[:goal]
end
class PackageInfo < Info[:package]
end
免責聲明
未经测试。
BinData::Base.read
将除第一个参数(IO对象)之外的所有参数直接传递给#initialize
,所以我认为这样的事情会起作用:
class InfoRecord < BinData::Record
endian :little
int32 :num
array :data, type: :array_type, initial_length: :num
attr_reader :array_type
def initialize(*args)
@array_type = args.pop
super
end
end
InfoRecord.read(io, :debug_log)
然而,这在DSL(info_record :my_record, ...
)中效果不佳。
为了解决这个问题,我认为你可以这样做:
class InfoRecord < BinData::Record
# ...
array :data, type: :array_type, initial_length: :num
def array_type
@params[:array_type]
end
end
InfoRecord.read(io, array_type: :debug_log)
我对上述内容不太确定,因为BinData::Base#initialize
处理其 Hash 参数的方式有点复杂,但希望它能允许您执行以下操作,例如:
class MyRecord < BinData::Record
info_record :my_info, array_type: :debug_log
end