第3章methods

Ruby是动态语言,有静态语言实现不了的技巧。本章讲解代码的重构,把代码变得更简洁。

3.2Dynamic Methods

3.21Calling Methods Dynamically

Dynamic Dispatch(动态派发技巧):使用Object#send方法,方法名是send()的参数,可以在代码运行的最后一刻决定调用哪个方法。

send(symbol[,args...]) -> obj  #string也行,会自动转换为symbol

send方法可以调用任何方法,包括私有方法。

Pry的例子(Ruby的命令行解释权,功能强于默认的irb) https://github.com/pry/pry

Kernel#respond_to?   检测如Rry#memory_size= 这样的方法是否存在。

respond_to?(symbol, include_all=false) → true or false

Hash#merge!

3.22 Defining Methods Dynamically动态方法。

使用Module#define_method()方法随时定义一个方法.代替def的重要原因是可以在运行时决定方法的名字。

说明:Defines an instance method in the receiver.

define_method(symbol, method) → symbol
define_method(symbol) { block } → symbol
 实例:
class Myclass
  define_method(:my_method){|my_arg| my_arg * 3}
end
obj = Myclass.new
p obj.my_method(2)

#########复杂的实例。

class A

  def fred
    puts "In Fred"
  end
  def create_method(name, &block)
    self.class.send(:define_method, name, &block)
  end
  define_method(:wilma) { puts "Charge it!" }
end
class B < A
  define_method(:barney, instance_method(:fred))
end
a = B.new
a.barney
a.wilma
a.create_method(:betty) { p "sad" }
a.betty
##########结果:
In Fred
Charge it!
"sad"
 

3.23Refactoring the Computer Class

第一步 添加动态派发Dynamic dispatch: 把重复的代码抽出来,放到该类的一个方法中用send方法来接收method名字作为参数。

第二步 动态创建方法:define_method方法重构Computer类

第三步 内省introspection,缩减代码。

代码:

class Computer
  def initialize(computer_id, data_source)
    @id = computer_id
    @data_source = data_source

# introspection内省

    data_source.methods.grep(/^get_(.*)_info$/){Computer.define_component $1}   
  end
  def self.define_component(name)
    define_method(name) do
      info = @data_source.send "get_#{name}_info", @id 

# get_#{name}_info是传进对象的类的方法。

      price = @data_source.send "get_#{name_price}", @id
      result = "#{name.capitalize}:#{info},#{price}"

return "超支项目#{result}"  if price >= 100

      result #省略了return
    end
  end
end
my_computer = Computer.new(4, 从其他类传近来的实例对象)
my_computer.cpu



3.3method_missing方法

动态代理和幽灵方法,解决代码繁复问题。

method_missing是BasicObject的私有实例方法。

3.31 Overriding method_missing

从新写method_missing方法让你可以调用实际并不存在的方法。被处理的消息,从调用者的角度看,和普通方法没区别,但实际receiver并没有对应的方法,这称为Ghose Method。

class Lawyer
  def method_missing(method, *args)
    puts "You called: #{method}(#{args.join(', ')})"
    puts "You also passed it a block" if block_given?
  end
end
b = Lawyer.new
b.talk('a', 'b', "c") do
  # a block
end

#*args代表多个参数。block_given?是kernel方法

3.51BasicObject

拥有极少实例方法的类也叫blank Slate。BasicObject只有几个实例方法。

如果幽灵方法和真实方法的名字冲突,幽灵方法会被忽略。这时可以让创建的类继承BasicObject.

3.52删除方法

Module#undef_method: 删除所以(包含继承)的方法

Module#remove_method : 只删除receiver自己的方法。

幽灵方法的重构:动态代理和白版类。

class Computer < BasicObject
  def initialize(computer_id, data_source)
    @id = computer_id
    @data_source = data_source
  end
  def method_missing(name, *args)
    super unless @data_source.respond_to?("get_#{name}_info")
    info = @data_source.send("get_#{name}_info", @id)
    price = @data_source.send "get_#{name_price}", @id
    result = "#{name.capitalize}: #{info} ($#{price})"
    return "*#{result}" if price >= 100
    result
  end
end

3.61Dynamic Methods vs. Ghost Methods

幽灵方法不是真正的方法,只是对方法调用的拦截。

原则:在可以使用动态方法的时候,尽量使用动态方法;除非必须使用幽灵方法,否则尽量不用。

3-9《元编程》第3章Tuesday:methods的更多相关文章

  1. 3-8《Ruby元编程》第二章对象模型

    <Ruby元编程> 第二章 对象模型 类定义揭秘inside class definitions: class关键字更像一个作用域操作符,核心作用是可以在里面随时定义方法. [].meth ...

  2. 一道模板元编程题源码解答(replace_type)

    今天有一同学在群上聊到一个比较好玩的题目(本人看书不多,后面才知是<C++模板元编程>第二章里面的一道习题), 我也抱着试一试的态度去完成它, 这道题也体现了c++模板元编程的基础和精髓: ...

  3. 初识C++模板元编程(Template Mega Programming)

    前言:毕设时在开源库上做的程序,但是源码看得很晕(当时导师告诉我这是模板元编程,可以不用太在乎),最近自己造轮子时想学习STL的源码,但也是一样的感觉,大致了解他这么做要干什么,但是不知道里面的机制. ...

  4. 3-11 《Ruby元编程》第4章block块 3-12

    第4章代码块blocks 基础知识 作用域:用代码块携带variables through scopes 通过传递block给instance_eval方法来控制作用域. 把block转换为Proc, ...

  5. C++模板元编程(C++ template metaprogramming)

    实验平台:Win7,VS2013 Community,GCC 4.8.3(在线版) 所谓元编程就是编写直接生成或操纵程序的程序,C++ 模板给 C++ 语言提供了元编程的能力,模板使 C++ 编程变得 ...

  6. atitit.元编程总结 o99

    atitit.元编程总结 o99.doc 1. 元编程(Metaprogramming) 1 2. 元编程的历史and发展 1 3. 元类型and元数据 1 4. 元编程实现方式 2 4.1. 代码生 ...

  7. ES6中的元编程-Proxy & Reflect

    前言 ES6已经出来好久了,但是工作中比较常用的只有let const声明,通过箭头函数改this指向,使用promise + async 解决异步编程,还有些数据类型方法...所以单独写一篇文章学习 ...

  8. Ruby元编程:单元测试框架如何找到测试用例

    前几天看了Google Testing Blog上的一篇文章讲到C++因为没有反射机制,所以如何注册测试用例就成了一件需要各显神通的事情.从我的经验来看,无论是Google的GTest还是微软的LTM ...

  9. Groovy元编程简明教程

    同函数式编程类似,元编程,看上去像一门独派武学. 在 <Ruby元编程>一书中,定义:元编程是运行时操作语言构件的编程能力.其中,语言构件指模块.类.方法.变量等.常用的主要是动态创建和访 ...

随机推荐

  1. 20154312 曾林 Exp8 web基础

    1.基础问题回答 1.1.什么是表单 1.2.浏览器可以解析运行什么语言 1.3.WebServer支持哪些动态语言 2.实践总结与体会 3.实践过程记录 ----3.1.Web前端:HTML基础 - ...

  2. python在交互模式下直接输入对象后回车,调用的是对象的__repr__()方法,这个方法表示的是一个编码,用print+对象是调用对象的__str__方法

    交互模式下调用对象的__repr__()方法,这个方法表示的是一个编码 >>> u"国庆节快乐"u'\u56fd\u5e86\u8282\u5feb\u4e50' ...

  3. python读取剪贴板报错 pywintypes.error: (1418, 'GetClipboardData', '\xcf\xdf\xb3\xcc\xc3\xbb\xd3\xd0\xb4\xf2\xbf\xaa\xb5\x

    在封装读取剪贴板的时候,执行测试代码时遇到个错误: pywintypes.error: (1418, 'GetClipboardData', '\xcf\xdf\xb3\xcc\xc3\xbb\xd3 ...

  4. Zookeeper学习记录(二):使用以及配置

    zookeeper已经介绍了它的原理设计以及实现方式,我们接下来介绍zookeeper的使用方法以及简单配置. 下载 获取Zookeeper的发布包,从Apache下载映像中下载一个最新稳定版本. 单 ...

  5. html5 manifest 离线缓存知识点

    1.最大缓存容量为 5M. 2.manifest文件需要配置正确的MIME-type,即“text/cache-manifest”,这个是在web服务器上进行配置. ②编写.manifest文件,文件 ...

  6. Android查缺补漏(View篇)--布局文件中的“@+id”和“@id”有什么区别?

    Android布局文件中的"@+id"和"@id"有什么区别? +id表示为控件指定一个id(新增一个id),如: <cn.codingblock.vie ...

  7. Unity 使用C/C++ 跨平台终极解决方案(PC,iOS,Android,以及支持C/C++的平台)

    https://blog.csdn.net/fg5823820/article/details/47865741 PC的其实根本不用说,毕竟C#和C++交互的文章已经够多了,当然我自认为经过几次折腾后 ...

  8. JavaScript Image对象 / Tabel对象 / Select对象 / Form对象

    JavaScript Image / Tabel / Select / Form 对象 版权声明:未经授权,严禁转载! Image 对象 Image 对象,代表 <img> 元素. < ...

  9. 20145106《Java程序设计》第7周学习总结

    教材学习内容总结 使用Lambda的特性可以去除重复的信息,以取得语法的简洁,增加程序代码的表达性.Lambda表达式本身是中性的,不代表任何类型的实例,同样的Lambda表达式,可用来表示不同目标类 ...

  10. 20145304 刘钦令 Exp2 后门原理与实践

    20145304 刘钦令 Exp2 后门原理与实践 基础问题回答 (1)例举你能想到的一个后门进入到你系统中的可能方式? 浏览网页时,或许会触发网站中隐藏的下载代码,将后门程序下载到默认地址. 下载的 ...