在Ruby On Rails中,有一个OR映射层,就是动态的从一张关系表映射到一个对象,这主要由ActiveRecord类来实现。在OR映射模型中,将关系数据库中的关系表表映射到对象模型时,将关系表的表名映射到类名,表中的每一个元组映射到对应于这个类的一个对象,元组的一个字段对应于对象的一个属性。
假如我们有一个保存职员基本信息的文件,文件的格式是这样的:第一行是文件内容的每个字段的名称,从第二行开始,则是每个职员的基本信息。现在我们有一个文件名为“employee.txt”的文件,其内容如下所示:
name,age,gender
"John", 23, "male"
"Linclon", 25, "male"
假设我们就 要从这个文本文件中读取数据,并进行一定的处理。如果是使用C++编程,你首先一定会想到应该定义一个Employee类,然后这个类中有name, age, gender这些成员变量。但是是采用这种方法的话,可以发现,如果想在职员信息中加入一个字段,比如部门(department),就不得不修改 Employee类的代码,在Employee类中增加一个“department”成员变量,所以我们的代码是高度依赖于文件的具体格式,这当然不是一 个好的现象。我们希望有一种更简单和优雅的方案,还有,Ruby动态性提高给我们一个解决方案,但是,我们应该从和下手呢,这就需要Ruby的元编程能 力。
首先,我们想应该有一个职员类,在Rails中,每个关系表的名称会成为类的名称,在这里,采用类似的方法,将文本文件的名称作为类的名称,在Ruby中,类名同时也是一个常量名,所以第一个字母必须为大写,我们使用如下的代码来生成类名。
class_name = File.basename(file_name, ".txt").capitalize
# "employee.txt" => "Employee"
klass = Object.const_set(class_name, Class.new)
Class.new生成一个新的类,这个类的名称是匿名的,所以采用const_set操作来绑定一个类名,变量klass是新类型的引用。
生成了这个类以后,需要想这个类添加姓名,年龄和性别这些属性,这些属性的名称是在文本文件的的第一行中给出的。
data = File.new(file_name)
header = data.gets.chomp
data.close
names = header.split(",")
下面的代码给出了如何生成这些属性,以及初始化这些属性值。
klass.class_eval do
attr_accessor *names
define_method(:initialize) do |*values|
names.each_with_index do |name, i|
instance_variable_set("@" + name, values)
end
end
#...
end
现在,有了一系列的访问子(可读和可写),通过instance_variable_set方法,又给每个属性做了初始化。
变量names是在块外部定义的,由于块的闭合性,所以变量names在块中也是有效的。当然,为了程序的演示,又定义的了一个to_s方法,代码如下所示:
define_method(:to_s) do
str = "<#{self.class}: "
names.each {|name| str << "#{name}=#{self.send(name)} "}
str + ">"
end
alias_method :inspect, :to_s
完成了这些以后,对于类的构造已经基本结束了,现在就需要真正的从文本文件中读取数据了。从文本文件读数据应该是一个类方法,而不是一个实例的方法,其实现代码如下:
def klass.read
array = []
data = File.new(self.to_s.downcase + ".txt")
data.gets #throw away header
data.each do |line|
line.chomp!
values = eval("[#{line}]")
array << self.new(*values)
end
data.close
return array
end
在这个方法中,使用字的类名来匹配相关的文件,比如将Employee类映射到“employee。txt”。
然后,从文件中读取职员信息,由于第一行是字段定义,所以要舍弃第一行数据。从第二行开始读取数据,每读取一行数据,则构造一个Employee实例。通过上面这个简单的例子,我们可以看出元编程的功能是相当之强大的,使用元编程技术,可以构造相当简单,优雅的解决方案。
class DataRecord
def self.make(file_name)
data = File.new(file_name)
header = data.gets.chomp
data.close
class_name = File.basename(file_name, ".txt").capitalize
# "employee.txt" => "Employee"
klass = Object.const_set(class_name, Class.new)
names = header.split(",")
klass.class_eval do
attr_accessor *names
define_method(:initialize) do |*values|
names.each_with_index do |name, i|
instance_variable_set("@" + name, values)
end
end
define_method(:to_s) do
str = "<#{self.class}: "
names.each {|name| str << "#{name}=#{self.send(name)} "}
str + ">"
end
alias_method :inspect, :to_s
end
def klass.read
array = []
data = File.new(self.to_s.downcase + ".txt")
data.gets #throw away header
data.each do |line|
line.chomp!
values = eval("[#{line}]")
array << self.new(*values)
end
data.close
return array
end
return klass
end
end
DataRecord.make("employee.txt")
list = Employee.read
puts list
<ouput></ouput>