元编程-代码块

基础知识

概念:

  1. 块由大量代码构成
  2. 块定义在{} 中,或者do...end关键字中
  3. 使用yield 语句来调用块
  4. 块与其具有相同名称的函数调用
  5. 调用一个方法时才可以定义一个块
  6. 可以使用kernel中的block_given?() 来询问当前方法的调用是否包含块
  7. 块有自己的参数,回调块时可以像调用方法一样为块提供参数参数

作用域

作用域是变量和方法可用性的范围

闭包

  1. 块是完整的,可以立即运行
  2. 块可以包含代码,也可以包含一组绑定
  3. 创建块时会获取到局部绑定,调用时会将块和其绑定传给方法
def method
    a = 1
    yield('whj')
end

a = 'hello'
method {|b| "#{a}, #{b}" }
=> "hello, whj"

这里在定义块 时 {|b| "#{a}, #{b}" } a 还是 'hello', 在yield 调用块时将 'hello'带入到方法中
  1. 块中额外的绑定会在块结束时消失(额外的绑定作用域仅限于块中)

切换作用域

全局变量可以在任何作用域中使用,但一当某处发生了修改,难以定位到,所以能不用全局变量就不用全局变量
建议用顶级变量代替全局变量,顶级变量: 对象main的实例变量
使用方法调用来代替作用域门

作用域门

程序会在三个地方切换作用域

  1. 类定义 class
  2. 模块定于 module
  3. 方法定义 def
#kernel 中的local_variables 可以看到绑定

v1 = 1                  
class MyClass           # 进入class
    v2 = 2        
    local_variables     # [:v2]
    def my_method       # 进入 def
        v3 = 3       
        local_variables # 方法未调用,此时无绑定
    end                 # 离开def 
    local_variables     # [:v2]
end                     # 离开class

obj = MyClass.new
obj.my_method           # [:v3]
local_variables         # [:v1, :obj] 绑定为最上面的v1和上面定义的obj对象

扁平化作用域(嵌套文法作用域)

使绑定穿越作用域门

  1. 将作用域门替换为非作用域门

Class.new 是class的完美替身

a = 'hello'
MyClass = Class.new do
    "#{a}"   #相当于创建了一个块,a是块的局部绑定
end

Module#define_method 方法来替代def

a = 'hello'
define_method :my_method do 
    "#{a}"
end

共享作用域

扁平作用域中定义了多个方法,将这些方法用一个作用域门保护起来,他们就可以共享绑定


def my_method
    a = 'hello'
    
    Kernel.send :define_method, :test_a do 
        a
    end 
    Kernel.send :define_method, :test_b do |x|
        a = a + x
    end
end

my_method

test_a # => 'hello'
test_b('whj') => 'hello whj'

使用send 来调用kernel的私有方法 define_method
kernel#test_a 和kernel#test_b 都可以看到a变量
其他地方因def这个作用域门的原因是看不到a的

instance_eval 方法

BasicObject#instance_eval, 在一个对象的上下文中执行块

obj.instance_eval do 
    self # => obj
end

instance_eval 会将代码块的接受者置为self,因此它可以访问接受者的私有方法和实例变量

instance_exec

instance_eval 不能够接受参数
instance_exce 可以接受参数,更灵活一些

洁净室

只是为了在其中执行块的对象,能够提供有用的方法来供代码块使用

可调用对象

块 (并不是对象,但是可以调用)
proc: 由块转换来的对象,Proc类
lambda: proc的变种,Proc类
方法

想存储一个代码块供以后使用就需要一个对象,将代码块传给一个proc对象,之后就可以用Proc#call 来执行这个代码块

str = Proc.new {|x| "#{x}, whj"}
str.call("hello") # => "hello, whj"

proc 和 lambda

proc 和lambda 是ruby中的两个内核方法,能够将代码块转为Proc对象

str = lambda {|x| "#{x}, whj"}
str = -> (x) {"#{x}, whj"} # lambda的第二种方式
str = proc {|x| "#{x}, whj"}
str = Proc.new {|x| "#{x}, whj"}
str.call("hello") # => "hello, whj"

区别

  1. lambda方法创建的Proc对象称为lambda, 其他方法创建的称为proc(Proc#lambda? 方法检测是否为lambda)
  2. return

lambda中 return仅仅表示从lambda中退出

def sum(obj)
    obj.call * 2
end

num = lambda { return 10 }
sum(num) # => 20

proc 中return 是从定义proc的定义域中返回

def sum
    obj = proc {return 10}
    num1 = obj.call # 此时直接返回,不再继续往下
    num1 * 2
end

sum # => 10

  1. 参数校验

lambda 参数校验严格,如果参数个数不对,会抛出ArgumentError

proc 会根据传入的参数调整成期望的参数形式,多的入参去掉, 少的入参补 nil

方法

通过调用kernel#method方法,可以获得一个用Method对象表示的方法,用Method#call 对其调用

Method#to_proc 可以将Method对象转化为Proc

&操作符

yield 不适用于:

  1. 把代码块传递给另外一个方法或者代码块
  2. 将代码块转为Proc

此时需要给代码块取一个名字,并将代码块附加到一个绑定上,可以给这个方法添加一个特殊的参数,这个参数必须是参数列表的最后一个,且以&开头

&操作符的含义: 这是一个Proc对象, 当作代码块来使用,去掉&操作符,就能得到一个Proc对象

def my_method(&proc)
    proc
end

p = my_method {|x| "#{x}, whj"}
p.class # => Proc
p.call("good night") # => "good night, whj"

将Proc转为代码块只需要在proc对象前面加上&

结论

lambda 更直观,像一个方法,且调用return只从代码中块返回

方法最严格,绑定于一个对象,在对象的作用域中执行

proc和块对参数较宽容

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351