Ruby中的反射(Reflection)应用实例_ruby专题

在Java语言中,提供了发射机制,通过发射机制可以通过字符串构造出这个对象,可以获取对象的所有方法(包括私有方法),可以调用私有方法,可以更改成员变量的值(包括私有的成员变量)。
Ruby也是面向对象的高级语言,当然也提供了反射机制,今天我们讨论通过类名称构造类对象的功能。

一、通过类名称构造类对象

我们先看普通的构造:

复制代码 代码如下:

module ModuleA

    #the class name, later we will use it to create the corresponding object

    CLASS_NAME_OF_WOOD = "ModuleA::Wood"

    CLASS_NAME_OF_WOODDESK = "ModuleA::WoodDesk"

    CLASS_NAME_OF_WOODCHAIR = "ModuleA::WoodChair"

    class Wood

        def initialize

            @desc = "I am a primal wood"

        end

        def say

            puts @desc

        end

    end

    class WoodDesk < Wood

        def initialize

            @desc = "I am a desk made of wood"

        end

        def say_private

            puts "actually, i have some bug but no public"

        end

        public :say

        private :say_private

    end

    class WoodChair < Wood

        def initialize

            @desc = "I am a chair made of wood"

        end

        def say_private

            puts "I Want get married with a WoodDesk..."

        end

        def smile

            puts "ha hah hah haha ...."

        end

        public :say

        private :say_private, :smile

    end

end

定义了一个基础类Wood,有两个子类:WoodDesk, WoodChair,子类有分别有一个私有方法 say_private。
我们new出对象来执行:

复制代码 代码如下:

#the normal initailze

wood = ModuleA::Wood.new

wood.say

desk = ModuleA::WoodDesk.new

desk.say

chair = ModuleA::WoodChair.new

chair.say

#try call the private method

puts "desk respond to say_private? #{desk.respond_to? :say_private}"

desk.say_private if desk.respond_to? :say_private

上面代码,执行public方法say,然后尝试执行private方法 say_private,执行先check是否能够执行,返回结果是不能执行,desk.respond_to? :say_private返回false:

复制代码 代码如下:

I am a primal wood

I am a desk made of wood

I am a chair made of wood

desk respond to say_private? false

好,现在我们通过反射机制来构造对象,并尝试执行其私有方法。

我们注意到模块的定义中有三个常量,定义的是类名称,

复制代码 代码如下:

    #the class name, later we will use it to create the corresponding object

    CLASS_NAME_OF_WOOD = "ModuleA::Wood"

    CLASS_NAME_OF_WOODDESK = "ModuleA::WoodDesk"

    CLASS_NAME_OF_WOODCHAIR = "ModuleA::WoodChair"

下面会通过这三个变量来理解Module.constants方法。

下面代码片段,基于上面的类定义:

复制代码 代码如下:

#get all module constants

obj_list = Array.new

tmp_const_sym_list = ModuleA.constants

tmp_const_sym_list.each do | sym |

    obj_list << ModuleA.const_get(sym)

    puts "calss = #{sym.class}, value = #{sym}"

end

我们注意到 ModuleA.constants,这个方法是Module模块中的,其作用是返回模块中所有常量的Symbol对象。我们看结果输出:

复制代码 代码如下:

calss = Symbol, value = CLASS_NAME_OF_WOOD

calss = Symbol, value = CLASS_NAME_OF_WOODDESK

calss = Symbol, value = CLASS_NAME_OF_WOODCHAIR

calss = Symbol, value = Wood

calss = Symbol, value = WoodDesk

calss = Symbol, value = WoodChair

从结果中看到,定义的三个常量和类名称都被返回了。所以注意:Ruby中的常量是包含定义的常量(变量)和类名称,注意他们都是Symbol对象。。

不过我们是需要根据类名称构造类对象,那么那三个常量就是没用的,需要删除。我们通过正则表达式匹配名字,来过滤。上面的代码修改一下:

复制代码 代码如下:

#get all module constants

sym_list = Array.new

tmp_const_sym_list = ModuleA.constants

tmp_const_sym_list.each do | sym |

    puts "calss = #{sym.class}, value = #{sym}"

    sym_list << ModuleA.const_get(sym) if /^Wood\w*/ =~ sym.to_s

end

sym_list << ModuleA.const_get(sym) if /^Wood\w*/ =~ sym.to_s,仅保存以Wood开头的symbol,这样我们就过滤掉了那三个常量。

找都类名称之后,开始构造对象:

复制代码 代码如下:

#create object from symbol

obj_list = Array.new

sym_list.each do | sym |

    obj = sym.new

    obj_list << obj

    puts "create the object: #{obj}"

end

begin

obj_list.each do | wood |

    wood.say

end

调用Symbol的new方法构造出次对象(sym.new),然后我们调用对象的say方法:

复制代码 代码如下:

create the object: #

create the object: #

create the object: #

I am a primal wood

I am a desk made of wood

I am a chair made of wood

达到了我们预期的结果。

二、操作成员变量和私有方法

使用过Java反射的同学们都知道,有了对象之后,操作成员变量和私有方法也就不在话下了。
Ruby中也是一样。

先看操作成员变量的例子。我们尝试更改一个成员变量的值。(接着上一片文章的代码)

复制代码 代码如下:

#manpulate instance variables

first_wood = obj_list.first

first_wood.instance_variables.each do | var |

    #get the instance variable

    puts "class of var = #{var.class}, value of var = #{var}"

    var_value = first_wood.instance_variable_get(var)

    puts "class of var_value = #{var_value.class}, value of var_value = #{var_value}"

    #set the new value of instance varialbe

    first_wood.instance_variable_set(var, var_value + "...and i was changed.")

    first_wood.say

end

1、first_wood.instance_variables.each,我们得到一个Wood对象,然后调用其instance_variables方法得到所有成员变量的名称(Symbol对象)。
2、然后,调用对象的first_wood.instance_variable_get方法,传递成员变量名称,得到成员变量对象。
3、最后,我们通过first_wood.instance_variable_set,改变这个成员变量的值。
代码运行结果:

复制代码 代码如下:

class of var = Symbol, value of var = @desc

class of var_value = String, value of var_value = I am a primal wood

I am a primal wood...and i was changed.

再看调用私有方法:

复制代码 代码如下:

#call private method

last_wood = obj_list.last

last_wood.method(:say_private).call

很简单,如果你知道方法名称,调用last_wood.method传入方法名,就可以得到一个Method对象,然后调用Method对象的call方法,结果是私有方法输出的内容:

复制代码 代码如下:

I Want get married with a WoodDesk...

普通场景下用不到修改成员变量和调用私有方法,因为这是违反了面向对象的封装原则的,那么反射在什么场景下有用呢?从我个人经验来说我觉得两个地方有用:
1)单元测试。
2)面向方面编程。
这两种场景都需要调用私有方法或替换成员变量的值。

你觉得呢?

时间: 2024-09-07 13:24:44

Ruby中的反射(Reflection)应用实例_ruby专题的相关文章

Ruby中对一元操作符重载实例_ruby专题

一元操作大家都知道,就是表达式的操作符只有一个输入值.这个在C和Java中都很常见.今天我们要探讨一下Ruby中的一元操作符重载. 一元操作符有:+ – * ! & 等,为了避免与数值的 + – 混淆,重载一元操作符,要在后面加上一个 @ 操作符. 1. 一个简单的一元操作符重载例子:-@ 操作符我们以String类为例子.String默认没有定义 – 操作符: 复制代码 代码如下: 1.9.3p125 :027 > a = "Hello" => "Hel

进一步深入Ruby中的类与对象概念_ruby专题

Ruby是纯面向对象的语言,所有项目似乎要Ruby中为一个对象.Ruby中的每个值是一个对象,即使是最原始的东西:字符串,数字甚至true和false.即使是一个类本身是一个对象,它是Class类的一个实例.本章将通过所有功能涉及到Ruby的面向对象. 类是用来指定对象的形式,它结合了数据表示和方法操纵这些数据,转换成一个整齐的包.在一个类的数据和方法,被称为类的成员.Ruby类的定义: 定义一个类,定义的数据类型的草图. 这实际上并不定义任何数据,但它定义的类名字的意思什么,即是什么类的对象将

浅析Ruby中的类对象的概念_ruby专题

 面向对象的程序涉及类和对象. 一个类是蓝本,从个别对象被创建.在面向对象的术语,我们说小明的自行车是被称为自行车类的对象实例. 任何车辆的例子.它包括轮子,马力,燃油或燃气罐容量.这些特点形成的类车辆的数据成员.可以从其他车辆区分这些特征. 车辆也有一定的功能,如停止,驾驶,超速驾驶.即使这些功能形成的类车辆的数据成员.因此,可以定义一个类作为一个组合的特点和功能. 车辆类可以被定义为: Class Vehicle { Number no_of_wheels Number horsepower

Ruby中操作文件的方法介绍_ruby专题

 Ruby提供了一套完整的I/O相关的内核模块中实现方法.所有I/O方法来自IO类. 类IO提供了所有的基本方法,如 read, write, gets, puts, readline, getc 和 printf. 本章将涵盖所有可供在Ruby中使用的基本I/O功能.如需使用更多的功能,请参考Ruby的IO类.puts 语句: 在前面的章节中,你指定值的变量和然后使用声明 puts 输出. puts 把语句指示程序显示存储在变量值.这将添加一个新行,每行末尾写出(输出). 例子: #!/usr

详解Ruby中的循环语句的用法_ruby专题

Ruby 中的循环用于执行相同的代码块若干次.本章节将详细介绍 Ruby 支持的所有循环语句.Ruby while 语句语法 while conditional [do] code end 当 conditional 为真时,执行 code.while 循环的 conditional 通过保留字 do.一个换行符.反斜线 \ 或一个分号 ; ,来与 code 分离开. 实例 #!/usr/bin/ruby $i = 0 $num = 5 while $i < $num do puts("I

Ruby中使用正则表达式的基础指引_ruby专题

正则表达式的内建支持通常只限于脚本语言如Ruby,Perl和awk等,这是一个耻辱:尽管正则表达式很神秘,但它是一个强大的文本处理工具.通过内建而不是通过程序库接口来支持它,有很大的不同. 正则表达式只是一种指定字符模式的方法,这个字符模式会在字符串中进行匹配.在Ruby中,通常在斜线之间(/pattern/)编写模式(pattern)来创建正则表达式.同时,Ruby就是Ruby,正则表达式是对象并且可以当作对象来操作. 比如,可以使用如下的正则表达式来编写模式,它会匹配包含Perl或Pytho

Ruby中遍历目录的简洁方法_ruby专题

在ruby中我们要实现遍历指定目录的方法,网上的方法也非常之多,我们可以拿来参考参考,如下边的traverse.rb文件内容所示: 复制代码 代码如下: #!/usr/bin/ruby def traverse(filepath)     if File.directory?(filepath)       puts "Dirs:" + filepath       Dir.foreach(filepath) do |filename|         if filename != &

Ruby中的Mechanize的使用教程_ruby专题

Ruby中实现网页抓取,一般用的是mechanize,使用非常简单. 安装 复制代码 代码如下: sudo gem install mechanize 抓取网页 复制代码 代码如下: require 'rubygems' require 'mechanize' agent = Mechanize.new page = agent.get('http://google.com/') 模拟点击事件 复制代码 代码如下: page = agent.page.link_with(:text => 'Ne

详解Ruby中的代码块及其参数传递_ruby专题

一,块的声明   块的声明在函数调用之后,用{..}括起来,或do..end封装.{}一般用在单行语句上,do..end用在多行语句上. (1..4).each{|v| print "#{v} "} #输出1 2 3 4   块可以带参数,与函数参数不同,块参数用||封装,当然,可以带多个参数.这些参数怎么定义,实际上是在函数内部定义好的,后面会讲到. 二,块内变量的访问   块内可以访问块外的变量,也就是块外的变量在块内是可见的,如 sum = 0 (1..5).each do |v

Ruby中的String对象学习笔记_ruby专题

1.String对象定义 String对象的定义可以使用""和'',对于单纯的字符串,推荐使用''进行定义,效率比""高,""与''的区别在于,''内的字符串定义后就是最终形态,即使如\n换行符,也会原样输出,而""更像是一个表达式,解析器会针对其中的特殊字符进行处理,然后才会输出,如下示例代码: 复制代码 代码如下: i  = 100 s1 = 'The value of i variable is #{i}!\n' #Th