校园春色亚洲色图_亚洲视频分类_中文字幕精品一区二区精品_麻豆一区区三区四区产品精品蜜桃

主頁 > 知識庫 > Ruby元編程技術詳解(Ruby Metaprogramming techniques)

Ruby元編程技術詳解(Ruby Metaprogramming techniques)

熱門標簽:冀州市地圖標注 個人怎么在地圖標注需要的店鋪 百度地圖標注早餐區域 清朝地圖標注哈爾濱 怎么去除地圖標注 漳州智云呼電話機器人 新岸線智能電銷機器人 武漢外呼防封系統多少錢 地圖標注大廈

我最近考慮了很多元編程(Metaprogramming)的問題,并希望看到更多這方面技術的例子和講解。無論好壞,元編程已經進入Ruby社區,并成為完成各種任務和簡化代碼的標準方式。既然找不到這類資源,我準備拋磚引玉寫一些通用Ruby技術的文章。這些內容可能對從其它語言轉向Ruby或者還沒有體驗到Ruby元編程樂趣的程序員非常有用。

1. 使用單例類 Use the singleton-class

  許多操作單個對象的方法是基于操作其單例類(singleton class),并且這樣可以使元編程更簡單。獲得單例類的經典方法是執行如下代碼:

復制代碼 代碼如下:

sclass = (class self; self; end) 

  RCR231建議這樣定義Kernel#singleton_class方法:
復制代碼 代碼如下:

module Kernel  
  def singleton_class  
    class self; self; end 
  end 
end 

  我會在下文使用這個方法。

2. DSL的使用類方法來修改子類 Write DSL's using class-methods that rewrite subclasses

  當你想創建一個DSL來定義類信息時,最常見的問題是怎樣表示信息來讓框架的其它部分使用。以定義一個ActiveRecord模型對象為例:

復制代碼 代碼如下:

class Product ActiveRecord::Base  
  set_table_name 'produce'   
end 

  在這個例子中,令人感興趣的是set_table_name的使用。這是怎么起作用的呢?好吧,這里涉及到一個小魔法。這是一種實現方法:

復制代碼 代碼如下:

module ActiveRecord  
  class Base  
    def self.set_table_name name  
      define_attr_method :table_name, name  
    end 
    def self.define_attr_method(name, value)  
      singleton_class.send :alias_method, "original_#{name}", name  
      singleton_class.class_eval do   
        define_method(name) do     
          value  
        end 
      end 
    end 
  end   
end 

  這里令人感興趣的是define_attr_method。在這個例子中我們需要獲得Product類的單例類,但又不想修改ActiveRecord::Base。通過使用單例類我們達到了這個目的。我們為原來的方法取別名,再定義新的存取器(accessor)來返回值。如果ActiveRecord需要table name就可以直接調用存取器。這種動態創建方法和存取器的技術在單例類是很常見的,特別是Rails。

3. 動態創建class和module Create classes and modules dynamically

  Ruby允許你動態創建和修改class和module。你可以在沒有凍結的class或module上做任何修改。特定情況下會很有用。Struct類可能是最好的例子:

復制代碼 代碼如下:

PersonVO = Struct.new(:name, :phone, :email)  
p1 = PersonVO.new(:name => "Ola Bini") 

  這會創建一個新類,并賦給PersonVO,然后創建一個類的實例。從草稿創建新類并定義新方法也很簡單:
復制代碼 代碼如下:

c = Class.new 
c.class_eval do 
  define_method :foo do 
    puts "Hello World" 
  end 
end 
c.new.foo    # => "Hello World" 

  除了Struct,還能在SOAP4R和Camping找到輕松創建類的例子。Camping尤其令人感興趣,因為它有專門的方法創建這些類,被你的controller和view繼承。Camping的許多有趣的功能都是用這種方式實現的:
復制代碼 代碼如下:

def R(*urls); Class.new(R) { meta_def(:urls) { urls } };   
end 

  這使得可以這樣創建controller:
class View R '/view/(\d+)' 
  def get post_id  
  end 
end 


  你也可以這樣創建module,然后在類中包含module。

4. 使用method_missing來做有趣的事 Use method_missing to do interesting things

  除了閉包(block),method_missing可能是Ruby最強大的特性,也是最容易濫用的一個。用好method_missing的話有些代碼會變得超級簡單,甚至是不能缺少。一個好的例子(Camping)是擴展Hash:

復制代碼 代碼如下:

class Hash 
  def method_missing(m,*a)  
    if m.to_s =~ /=$/  
      self[$`] = a[0]  
    elsif a.empty?    
      self[m]  
    else 
      raise NoMethodError, "#{m}" 
    end 
  end 
end 

  就可以這樣使用hash:
復制代碼 代碼如下:

x = {'abc' => 123}  
x.abc # => 123  
x.foo = :baz 
x # => {'abc' => 123, 'foo' => :baz} 

  如你所見,如果有人調用了一個hash不存在的方法,則會搜索內部集合。如果方法名以=結尾,則會賦給同名的key。

  Markaby中可以找到另一個很好的method_missing技巧。以下引用的代碼可以生成任何包含CSS class的XHTML標簽:

復制代碼 代碼如下:

body do 
  h1.header 'Blog' 
  div.content do 
    'Hellu' 
  end 
end 

會生成:
復制代碼 代碼如下:

body> 
  h1 class="header">Blog/h1> 
  div class="content"> 
    Hellu  
  /div> 
/body> 

  絕大多數這種功能,特別是CSS class名是通過method_missing設置了self的屬性然后返回self。

5. 方法模式的調度 Dispatch on method-patterns

  這對于無法預測的方法來說可以輕松的達到可擴展性。我最近創建了一個小型驗證框架,核心的驗證類會找出自身所有以check_開頭的方法并調用,這樣就可以輕松地增加新的驗證:只要往類或實例中添加新方法。
methods.grep /^check_/ do |m|  
  self.send m  
end 

  這非常簡單,并且難以置信的強大??梢钥匆幌耇est::Unit到處使用這種方法。

6. 替換方法 Replacing methods

  有時候一個方法的實現不是你要的,或者只做了一半。標準的面向對象方法是繼承并重載,再調用父類方法。僅當你有對象實例化的控制權時才有用,經常不是這種情況,繼承也就沒有價值。為得到同樣的功能,可以重命名(alias)舊方法,并添加一個新的方法定義來調用舊方法,并確保舊方法的前后條件得到保留。

復制代碼 代碼如下:

class String 
  alias_method :original_reverse, :reverse 
  def reverse   
    puts "reversing, please wait..." original_reverse  
  end 
end 

  一個極端的用法是臨時修改一個方法,然后再還原。例如:
復制代碼 代碼如下:

def trace(*mths)  
  add_tracing(*mths) # aliases the methods named, adding tracing      
  yield 
  remove_tracing(*mths) # removes the tracing aliases  
end 

  這個例子展示了編寫add_tracing和remove_tracing的一種典型方法。它依賴于第1條的單例類:
復制代碼 代碼如下:

class Object    
  def add_tracing(*mths)      
    mths.each do |m|   
      singleton_class.send :alias_method, "traced_#{m}", m   
      singleton_class.send :define_method, m do |*args|  
        $stderr.puts "before #{m}(#{args.inspect})" 
        ret = self.send("traced_#{m}", *args)  
        $stderr.puts "after #{m} - #{ret.inspect}" 
        ret  
      end 
    end    
  end 
  def remove_tracing(*mths)     
    mths.each do |m|  
      singleton_class.send :alias_method, m, "traced_#{m}" 
    end 
  end 
end 
"abc".add_tracing :reverse 

  如果這些方法是添加到module(有一點點不同,看你能不能寫出來!),你也可以在類而非實例上添加和刪除tracing。

7. 使用nil類來引入空對象的重構 Use NilClass to implement the Introduce Null Object refactoring

  在Fowler的重構中,“引入空對象”的重構是一個對象要么存在,要么為空時有一個預定義值。典型例子如下:

復制代碼 代碼如下:

name = x.nil? ? "default name" : x.name 

  目前基于Java的重構會推薦創建一個類似于null的子類。例如NullPerson會繼承Person,重載name方法總是返回"default name"。但是在Ruby中我們可以打開類,可以這樣做:
復制代碼 代碼如下:

def nil.name; "default name"; end 
x # => nil  
name = x.name # => "default name" 

8. 學習eval的不同版本 Learn the different versions of eval

  Ruby有幾種版本的執行方法(evaluation)。了解它們的區別和使用情景是很重要的。有eval、instance_eval、module_eval和class_eval幾種。首先,class_eval是module_eval的別名。其次,eval和其他的有些不同。最重要的是eval只能夠執行一個字符串,其它的可以執行block。這意味著eval是你做任何事的最后選擇,它有它的用處,但絕大多數情況下應該用instance_eval和module_eval執行block。

  eval會在當前環境執行字符串,除非環境已經提供綁定(binding)。(見第11條)

  instance_eval會在接收者(reveiver)的上下文中執行字符串或block,沒有指定的話self會作為接收者。

  module_eval會在調用的module的上下文中執行字符串或block。這個比較適合在module或單例類中定義新方法。instance_eval和module_eval的主要區別在于定義的方法會放在哪里。如果你用String.instance_eval定義foo方法會得到String.foo,如果是用module_eval會得到String.new.foo。

  module_eval幾乎總是適用;要像對待瘟疫一樣避免使用eval。遵守這些簡單的規則會對你有好處。


9. 實例變量的內省 Introspect on instance variables

  Rails使用了一個技巧來使controller中的實例變量也能用在view中,就是內省一個對象的實例變量。這會嚴重破壞封裝,然而有時候確實非常順手。可以很容易的通過instance_variables、instance_variable_get和instance_variable_set實現。要把所有實例變量從一個復制到另一個,可以這樣:

復制代碼 代碼如下:

from.instance_variables.each do |v|  
  to.instance_variable_set v, from.instance_variable_get(v)  
end 

10. 從block創建Proc并公開 Create Procs from blocks and send them around

  把一個Proc實例化保存在變量中并公開的做法使得很多API容易使用。這是Markaby用來管理CSS class定義的一種方法。很容易把block轉換成Proc:
def create_proc(p); p; end 
create_proc do 
  puts "hello" 
end       # => #Proc ...> 

  調用也很容易:
p.call(*args) 

  如果要用proc來定義方法,應該用lambda來創建,就可以用return和break:
p = lambda { puts "hoho"; return 1 }  
define_method(:a, p) 

  如果有block的話method_missing會調用block:
def method_missing(name, *args, block)  
  block.call(*args) if block_given?  
end 
thismethoddoesntexist("abc","cde") do |*args|  
  p args  
end  # => ["abc","cde"] 


11. 用綁定(binding)來控制eval Use binding to control your evaluations

  如果你確實需要用eval,你可以控制哪些變量是有效的。這時候要用kernel方法binding來獲得所綁定的對象。例如:

復制代碼 代碼如下:

def get_b; binding; end 
foo = 13  
eval("puts foo",get_b) # => NameError: undefined local variable or method `foo' for main:Object 

  ERb和Rails用這種技術來設置哪些實例變量是有效的。例如:
復制代碼 代碼如下:

class Holder  
  def get_b; binding; end 
end 
h = Holder.new 
h.instance_variable_set "@foo", 25  
eval("@foo",h.get_b) 

  希望這些技巧和技術已經為您闡明了元編程。我并不聲稱自己是Ruby或者元編程方面的專家,這只是我對這個問題的一些想法。

您可能感興趣的文章:
  • Ruby元編程的一些值得注意的地方
  • ruby元編程之創建自己的動態方法
  • ruby元編程之method_missing的一個使用細節
  • Ruby元編程之夢中情人method_missing方法詳解
  • Ruby元編程小結
  • Ruby和元編程之萬物皆為對象
  • ruby元編程實際使用實例
  • Ruby元編程基礎學習筆記整理

標簽:金昌 濰坊 天門 天門 儋州 德宏 宣城 臺灣

巨人網絡通訊聲明:本文標題《Ruby元編程技術詳解(Ruby Metaprogramming techniques)》,本文關鍵詞  Ruby,元,編程,技術,詳解,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《Ruby元編程技術詳解(Ruby Metaprogramming techniques)》相關的同類信息!
  • 本頁收集關于Ruby元編程技術詳解(Ruby Metaprogramming techniques)的相關信息資訊供網民參考!
  • 推薦文章
    校园春色亚洲色图_亚洲视频分类_中文字幕精品一区二区精品_麻豆一区区三区四区产品精品蜜桃
    中文字幕亚洲电影| 91偷拍与自偷拍精品| 色婷婷综合中文久久一本| 欧美日韩小视频| 亚洲丝袜自拍清纯另类| 国产成人小视频| 日本一区二区免费在线| 91蜜桃婷婷狠狠久久综合9色| 成人涩涩免费视频| 91黄色免费观看| 亚洲va欧美va人人爽午夜| 欧美一卡二卡在线观看| 国产精品一二三区在线| 亚洲你懂的在线视频| 欧美一区午夜精品| 91网站在线观看视频| 麻豆精品视频在线观看视频| 国产精品色在线| 欧美日韩一区二区三区免费看| 极品美女销魂一区二区三区| 成人免费在线播放视频| 欧美xfplay| 欧美三级在线看| 成人白浆超碰人人人人| 亚洲自拍偷拍麻豆| 这里是久久伊人| 成人国产视频在线观看| 蜜桃91丨九色丨蝌蚪91桃色| 欧美国产1区2区| 91精品国产一区二区三区| 国产美女精品在线| 亚洲一区二区三区四区在线观看| 精品国产91洋老外米糕| 成人18视频在线播放| 日韩高清国产一区在线| 亚洲综合小说图片| 精品久久久久久久久久久久久久久久久| 成人97人人超碰人人99| 国产在线观看一区二区| 亚洲图片欧美一区| 日本一区免费视频| 日韩一二三四区| 91黄色免费网站| 成人高清视频免费观看| 青青草91视频| 午夜精品一区二区三区电影天堂| 国产亚洲精品精华液| 欧美α欧美αv大片| 日韩午夜在线播放| 欧美日韩中文字幕一区二区| 懂色中文一区二区在线播放| 爽好多水快深点欧美视频| 亚洲色欲色欲www在线观看| 日韩一区二区三区免费观看| 91久久国产综合久久| 成人动漫在线一区| 国产成人在线视频播放| 国产一区免费电影| 免费视频最近日韩| 亚洲欧美经典视频| 日本一区二区久久| 久久人人97超碰com| 在线电影一区二区三区| 欧美精品亚洲二区| 精品视频一区 二区 三区| 欧美亚洲动漫另类| 欧美日韩中文精品| 欧美自拍偷拍一区| 色婷婷精品久久二区二区蜜臂av| 国产91精品在线观看| 成人午夜大片免费观看| 国产一区二区三区电影在线观看 | 丁香婷婷综合激情五月色| 精品亚洲欧美一区| 国产一区二三区| 国产成人av电影免费在线观看| 男女男精品视频网| 九色porny丨国产精品| 国产精品一级二级三级| 激情六月婷婷久久| 在线播放中文一区| 日韩一区二区中文字幕| 2欧美一区二区三区在线观看视频| 精品欧美一区二区久久| 久久久久亚洲蜜桃| 国产精品久久久久7777按摩| 国产女人aaa级久久久级 | 一区二区视频在线| 亚洲人一二三区| 亚洲日本护士毛茸茸| 午夜精品久久久久影视| 日韩精品视频网| 天天综合色天天综合色h| 久久国产福利国产秒拍| 岛国精品一区二区| 欧美日本不卡视频| 日韩亚洲欧美成人一区| 国产精品大尺度| 香蕉影视欧美成人| 国产成人免费9x9x人网站视频| 91热门视频在线观看| 日韩欧美精品三级| 国产精品蜜臀在线观看| 午夜久久久久久| 99精品偷自拍| 日韩亚洲欧美在线观看| 亚洲一区在线观看视频| 激情五月婷婷综合网| 欧美三级中文字| 中文字幕欧美三区| 亚洲.国产.中文慕字在线| 韩国av一区二区三区在线观看| 91美女视频网站| 91精品国产综合久久蜜臀 | 51精品视频一区二区三区| 中文欧美字幕免费| 亚洲成人777| 成人avav影音| 在线播放视频一区| 亚洲欧美日韩人成在线播放| 黄色日韩三级电影| 欧美亚洲综合另类| 国产日韩精品一区二区三区在线| 免费一级欧美片在线观看| 色94色欧美sute亚洲13| 久久精品欧美一区二区三区不卡 | 韩国午夜理伦三级不卡影院| 欧美性欧美巨大黑白大战| 国产精品伦理一区二区| 国产aⅴ综合色| 国产亚洲1区2区3区| 精品一区二区综合| 欧美日韩国产小视频| 亚洲国产一区二区在线播放| 色综合久久九月婷婷色综合| 亚洲日本va在线观看| 99久久综合国产精品| 国产精品久久久久久久久久久免费看 | 美腿丝袜亚洲综合| 欧美日韩成人在线一区| 亚洲国产一区二区a毛片| 日本精品一区二区三区高清| 亚洲激情综合网| 欧美三级视频在线观看| 日韩精品一二三| 欧美日韩在线三区| 日韩成人免费在线| 欧美一区二区三区男人的天堂| 蜜桃视频一区二区三区在线观看| 91女厕偷拍女厕偷拍高清| 久久久久久久久久久黄色| 久久久精品tv| 成人h动漫精品一区二区| 中文字幕一区二区三中文字幕| 91色乱码一区二区三区| 亚洲综合激情另类小说区| 在线不卡欧美精品一区二区三区| 美腿丝袜亚洲三区| 久久久精品综合| 色哟哟一区二区| 美女在线一区二区| 欧美激情一区二区三区四区| 99这里都是精品| 免费人成网站在线观看欧美高清| 精品国产露脸精彩对白| 99re成人精品视频| 麻豆久久一区二区| 欧美一区二区久久久| 国产麻豆9l精品三级站| 国产精品久久久久久久久搜平片| 欧美自拍丝袜亚洲| 免费一级欧美片在线观看| 日本一区二区视频在线| 欧美性受xxxx| 国产乱码精品一区二区三 | 2022国产精品视频| 一本一道综合狠狠老| 久久er99精品| 夜夜亚洲天天久久| 精品国产成人系列| 在线免费视频一区二区| 精品系列免费在线观看| 一区二区三区日本| 欧美v国产在线一区二区三区| av在线播放一区二区三区| 男男成人高潮片免费网站| 国产精品三级久久久久三级| 99re热视频精品| 激情欧美一区二区三区在线观看| 丝袜脚交一区二区| 亚洲精品福利视频网站| 欧美一级在线观看| 欧美精品乱码久久久久久| 99re66热这里只有精品3直播 | 91精品国产丝袜白色高跟鞋| 久久久亚洲综合| 成人黄色777网| 日韩精品一二区| 亚洲美女在线国产| 国内成人免费视频|