主頁 > 知識庫 > ruby元編程實(shí)際使用實(shí)例

ruby元編程實(shí)際使用實(shí)例

熱門標(biāo)簽:柯城手機(jī)地圖如何做地圖標(biāo)注 征服者企業(yè)地圖標(biāo)注 天津外呼系統(tǒng)怎么收費(fèi) 中牟外呼系統(tǒng)違法嗎 漯河電銷 AI電銷機(jī)器人 線路 外呼線路從哪里出來的 巫師3地圖標(biāo)注魔力之所 淮安自動外呼系統(tǒng)供應(yīng)商

很喜歡ruby元編程,puppet和chef用到了很多ruby的語言特性,來定義一個(gè)新的部署語言。
分享幾個(gè)在實(shí)際項(xiàng)目中用到的場景,能力有限,如果有更優(yōu)方案,請留言給我:)

rpc接口模板化——使用eval、alias、defind_method

require 'rack/rpc'

class Server  Rack::RPC::Server
 def hello_world
  "Hello, world!"
 end

 rpc 'hello_world' => :hello_world
end

上面是一個(gè)rpc server,編寫一個(gè)函數(shù),調(diào)用rpc命令進(jìn)行注冊。

采用define_method、eval、alias方法,可以實(shí)現(xiàn)一個(gè)判斷rpc/目錄下的*.rb文件,進(jìn)行加載和rpc接口注冊的功能,實(shí)現(xiàn)代碼如下:

module RPC
  require 'rack/rpc'
  #require rpc/*.rb文件
  Dir.glob(File.join(File.dirname(__FILE__), 'rpc', "*.rb")) do |file|
   require file
  end
  class Runner  Rack::RPC::Server
   #include rpc/*.rb and regsiter rpc call
   #eg. rpc/god.rb  god.hello
   @@rpc_list = []
   Dir.glob(File.join(File.dirname(__FILE__), 'rpc', "*.rb")) do |file|
    rpc_class = File.basename(file).split('.rb')[0].capitalize
    rpc_list = []
    
    #加載module下的方法到Runner這個(gè)類下面
    eval "include Frigga::RPC::#{rpc_class}"
    #獲取聲明的RPC接口
    eval "rpc_list = Frigga::RPC::#{rpc_class}::RPC_LIST"
    rpc_list.each do |rpc_name|
     #alias一個(gè)新的rpc方法,叫old_xxxx_xxxx
     eval "alias :old_#{rpc_class.downcase}_#{rpc_name} :#{rpc_name}"

     #重新定義rpc方法,添加一行日志打印功能,然后再調(diào)用old_xxxx_xxxx rpc方法
     define_method "#{rpc_class.downcase}_#{rpc_name}".to_sym do |*arg|
      Logger.info "[#{request.ip}] called #{rpc_class.downcase}.#{rpc_name} #{arg.join(', ')}"
      eval "old_#{rpc_class.downcase}_#{rpc_name} *arg"
     end 

     #注冊RPC調(diào)用
     rpc "#{rpc_class.downcase}.#{rpc_name}" => "#{rpc_class.downcase}_#{rpc_name}".to_sym

     #添加到全局變量,匯總所有的rpc方法
     @@rpc_list  "#{rpc_class.downcase}.#{rpc_name}"
    end
   end
   
   def help
    rpc_methods = (['help'] + @@rpc_list.sort).join("\n")
   end
   rpc "help" => :help

  end
 end #RPC

完成上述功能后,可以非常方便的開發(fā)rpc接口,例如下面這個(gè)IP地址增、刪、查的代碼,注冊ip.list, ip.add和ip.del方法:

module RPC
  module Ip
   #RPC_LIST used for regsiter rpc_call
   RPC_LIST = %w(list add del)

   def list
    $white_lists
   end   

   def add(ip) 
    if ip =~ /^((25[0-5]|2[0-4]\d|[0-1]?\d\d?)\.){3}(25[0-5]|2[0-4]\d|[0-1]?\d\d?)$/
     $white_lists  ip
     write_to_file
     return "succ"
    else
     return "fail"
    end
   end

   def del(ip)
    if $white_lists.include?(ip)
     $white_lists.delete ip
     write_to_file
     return "succ"
    else
     return "fail"
    end    
   end

   def write_to_file
     File.open(IP_yml, "w") do |f|
      $white_lists.uniq.each {|i| f  "- #{i}\n"}
     end
   end
  end 
 end

DSL——使用instance_eval

instance_eval是ruby語言中的瑞士軍刀,特別是支持DSL方面。
我們來看一下chef(一個(gè)開源的自動化部署工具)中設(shè)置文件模板的API:

復(fù)制代碼 代碼如下:

    template "/path/to/file.conf" do
      source "file.conf.erb"
      owner  "wilbur"
      mode   "0744"
    end

上述代碼中,source、owner、mode需要從外部block,傳遞到template內(nèi)部的block中,為了實(shí)現(xiàn)該目的,采用了instance_eval代碼如下:

  class ChefDSL
   def template(path, block)
    TemplateDSL.new(path, block)
   end
  end

  class TemplateDSL
   def initialize(path, block)
    @path = path
    instance_eval block
   end

   def source(source); @source = source; end
   def owner(owner);  @owner = owner; end
   def mode(mode);   @mode  = mode;  end
  end

上面這個(gè)小技巧使得TemplateDSL對象可以應(yīng)用block,和在自己的scope一樣。block可以訪問和調(diào)用TemplateDSL中的變量和方法。

如果沒有使用instance_eval,如下面的代碼,ruby就會拋出一個(gè)NoMethodError,因?yàn)閟ource、owner、mode無法在block中被訪問到。

復(fù)制代碼 代碼如下:

    class TemplateDSL
      def initialize(path, block)
        @path = path
        block.call
      end
    end

當(dāng)然也可以使用yeild傳遞變量的方式實(shí)現(xiàn),但沒有instance_eval簡潔和靈活。

命令行交互——使用instance_eval

命令行交互,可以采用highline這個(gè)gem.
但highline在有些方面不能滿足我的需求,比如類似上面介紹的chef template功能,達(dá)到的效果如下,大大簡化了重復(fù)代碼:

復(fù)制代碼 代碼如下:

        #檢查frigga fail,詢問是否繼續(xù)
        Tip.ask frigga_fail? do
          banner "Check some frigga failed, skip failed host and continue deploy?"
          on :yes
          on :quit do
            raise Odin::TipQuitExcption
          end
        end
        ...

        #運(yùn)行時(shí)顯示結(jié)果如下:
        Check some frigga failed, skip failed host and continue deploy? [yes/quit]
        #輸入yes繼續(xù),輸入quit退出

實(shí)現(xiàn)代碼如下:

 require 'colorize'
 class Tip
  def self.ask(stat = true, block)
   new(block).ret if stat == true
  end

  attr_reader :ret
  def initialize(block)
   @opt = []
   @caller = {}
   @banner = ""
   @ret = false
   self.instance_eval(block)
   print "#{@banner} [#{@opt.join('/')}]: ".light_yellow
   loop do
    x = gets.chomp.strip.to_sym
    if @opt.include?(x)
     @ret = ( @caller[x].call if @caller.key?(x) )
     if @ret == :retry
      print "\n#{@banner} [#{@opt.join('/')}]: ".light_yellow
      next
     else
      return @ret
     end
    else
     print "input error, please enter [#{@opt.join('/')}]: ".light_yellow
    end
   end

  end

  def on(opt, block)
   @opt  opt
   @caller[opt] = block if block_given?
  end
  def banner(str)
   @banner = str
  end
 end
您可能感興趣的文章:
  • Ruby元編程的一些值得注意的地方
  • ruby元編程之創(chuàng)建自己的動態(tài)方法
  • ruby元編程之method_missing的一個(gè)使用細(xì)節(jié)
  • Ruby元編程之夢中情人method_missing方法詳解
  • Ruby元編程技術(shù)詳解(Ruby Metaprogramming techniques)
  • Ruby元編程小結(jié)
  • Ruby和元編程之萬物皆為對象
  • Ruby元編程基礎(chǔ)學(xué)習(xí)筆記整理

標(biāo)簽:大慶 克拉瑪依 河池 西雙版納 南昌 甘孜 內(nèi)江 棗莊

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《ruby元編程實(shí)際使用實(shí)例》,本文關(guān)鍵詞  ruby,元,編程,實(shí)際,使用,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《ruby元編程實(shí)際使用實(shí)例》相關(guān)的同類信息!
  • 本頁收集關(guān)于ruby元編程實(shí)際使用實(shí)例的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章