自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

牛人點(diǎn)評Ruby語言十大令人喜愛的特點(diǎn)

譯文
開發(fā) 開發(fā)工具
自從Rails框架問世后,Ruby一舉成為了一個令不少人贊賞的編程語言。在這篇文章中,來自Merb與Rails核心開發(fā)團(tuán)隊的Yehuda Katz向我們分享了十大令人喜愛的Ruby語言特點(diǎn)。

【51CTO精選譯文】我每天都要用 Ruby 工作,久而久之,我現(xiàn)在真的喜歡上使用它了。(51CTO編者注:本文作者Yehuda Katz是Ruby on Rails核心開發(fā)團(tuán)隊的成員,以及Merb項目的主要推動者。)下面是一個列表,列出了我最喜歡的 Ruby 語言特點(diǎn)。一些特點(diǎn)顯而易見,一些特點(diǎn)也存在于其他語言中。分享 Ruby 這些我喜歡的特點(diǎn),并非是為了和其他語言進(jìn)行比較和對比。

51CTO編輯推薦:Ruby入門教程與技巧大全

1. 動態(tài)類型

靜態(tài)類型語言也有很不錯的功能,比如編譯時驗證和 IDE 支持。不過根據(jù)我的經(jīng)驗,動態(tài)類型對于項目啟動真的有很大幫助,并且便于進(jìn)行更改,尤其是在項目的早期到中期這些階段。

為了能夠讓我能夠輕松地繼續(xù)對象交換,我不需要為新對象創(chuàng)建正式的接口,這點(diǎn)讓人很開心。

2. Duck Typing(鴨子類型)

這只是動態(tài)類型的一個有效的擴(kuò)展。在 Ruby 中,預(yù)期能夠?qū)ψ址畬ο筮M(jìn)行操作的方法并不會檢查 is_a?(String)。它們檢查對象是否 respond_to?(:to_str),如果是,就接著調(diào)用對象的 to_str。與此類似,在 Ruby 中表示路徑(Path)的對象能夠?qū)崿F(xiàn)一個 to_path 方法為提供路徑重現(xiàn)(representation)。

在 Rails 語言中,對于具有“模型”特性的對象,我們可以使用這樣的技巧來實(shí)現(xiàn)對它們 respond_to?(:to_model) 的預(yù)期。如果這些對象能夠為我們提供一個它們自身的“模型”重現(xiàn),我們就能夠在相關(guān)語境中支持任何類型。51CTO之前曾發(fā)布過有關(guān)Ruby中鴨子類型的介紹,可以參考一二。

3. 令人嘆為觀止的模塊

Ruby 提供了一個與 Scala、Squeak 和 Perl 語言中“traits”類似的功能。事實(shí)上,Ruby 模塊可以在運(yùn)行時動態(tài)地址類等級中添加新元素。運(yùn)行時可以動態(tài)地對 super 的使用繼續(xù)評估以考慮所有添加的模塊,這樣就可以方便地按照所需多次地擴(kuò)展超類功能,而且無需指定在類聲明時確定super的加載地點(diǎn)。

此外,Ruby 模塊提供了生命周期鉤子(hook)append_features 和 included,這樣就可以使用模塊來互相隔離擴(kuò)展以及在特性包含的基礎(chǔ)上動態(tài)的擴(kuò)展類。

4. 類主體不是專用的

在 Ruby 中,類主體不是專用的語境。它們僅僅是一個對象類的自身指向點(diǎn)。如果你用過 Rails,你可能看到這樣的代碼:

  1. class Comment < ActiveRecord::Base  
  2.   validates_presence_of :post_id 
  3. end 

validates_presence_of 看起來好像是語言的一項功能,但實(shí)際上它是 Comment 上調(diào)用的方法,而 Comment 由 ActiveRecord::Base 提供。

該方法可以類中的執(zhí)行任意代碼(arbitrary code),包括創(chuàng)建新的方法,執(zhí)行代碼中其他內(nèi)容,或者更新類實(shí)例變量。與必須在編譯時運(yùn)行的 Java 標(biāo)注不同,Ruby 類主體能夠考慮到運(yùn)行時的信息,如動態(tài)提供的選項或其他代碼的評估結(jié)果。

5. 字符串求值(eval 功能)

這可能是一個不同的想法。這里我不是指任意運(yùn)行時字符串的求值,而是用于在 Ruby 程序啟動過程中創(chuàng)建方法的字符串求值。

這樣就能夠利用 Ruby 定義的結(jié)構(gòu)(如 Rails 路徑或 AOP 定義),并且能夠?qū)⑵渚幾g到 Ruby 方法中。當(dāng)然,也可以將其作為其他語言的附件(add-on)來實(shí)現(xiàn),但在純 Ruby 環(huán)境中實(shí)現(xiàn)這類功能是可能的。在很大程度上,它是一種自足執(zhí)行(self-hosting)的一種語言。

6. 區(qū)塊和 Lambda 表達(dá)式

我已經(jīng)說過多次,這里再重復(fù)一次:我認(rèn)為沒有匿名 lambda 表達(dá)式的語言還沒有足夠強(qiáng)大到讓我每天使用它。這些構(gòu)造事實(shí)上非常常見,在 Ruby、JavaScript、Scala、Clojure 和 Lisp 中也存在。51CTO之前介紹過Ruby 1.9中的Lambda表達(dá)式,有興趣的讀者可以看看。

利用它們,就能夠?qū)崿F(xiàn)看起來好像是語言功能的區(qū)塊范圍內(nèi)的構(gòu)造。最常見的使用示例是對文件的操作。在沒有 lambda 的語言中,用戶不得不在同一個語法范圍(和他們最初打開的文件一致)中使用一段“確?!眳^(qū)塊,以確保關(guān)閉了資源。

在 Java 中:

  1. static void run(String in)   
  2. throws FileNotFoundException {  
  3.   File input = new File(in);  
  4.   String line; Scanner reader = null;  
  5.   try {  
  6.     reader = new Scanner(input);  
  7.     while(reader.hasNextLine()) {  
  8.       System.out.println(reader.nextLine());  
  9.     }  
  10.   } finally { reader.close(); }  

Java 版的代碼常常需要在 try 區(qū)塊中包括 Scanner 的創(chuàng)建以保證其關(guān)閉。相反,讓我們看看 Ruby 的代碼:

  1. def run(input)  
  2.   File.open(input, "r"do |f|  
  3.     f.each_line {|line| puts line }  
  4.   end 
  5. end 

由于區(qū)塊的存在,我們可以省去在單個位置關(guān)閉文件的麻煩,將程序員錯誤減少到最小并且能夠減少重復(fù)。

#p#

7. 功能組合:自足執(zhí)行(self-hosting)語言

以上特點(diǎn)組合組合在一起,讓我們能夠在 Rails 中“擴(kuò)展”Ruby 語言。看下面這段代碼:

  1. respond_to do |format|  
  2.   if @user.save  
  3.     flash[:notice] = 'User was successfully created.' 
  4.     format.html { redirect_to(@user) }  
  5.     format.xml { render :xml => @user:status =>ted, :location => @user }  
  6.   else 
  7.     format.html { render :action => "new" }  
  8.     format.xml { render :xml => @user.errors, :status => :unprocessable_entity }  
  9.   end 
  10. end 

在這個示例中,我們可以無縫地將方法(respond_to)與正常的 Ruby 代碼(if 和 else))混合在一起,以生成一個新的區(qū)塊范圍的構(gòu)造。Ruby 的區(qū)塊語法讓我們能夠在區(qū)塊內(nèi)使用 return 和 yield,從而進(jìn)一步混合代碼區(qū)塊與語言構(gòu)造(如 if 或while 的界限)。

在 Rails 3 中,我們引入下面一段代碼:

  1. class PeopleController < ApplicationController  
  2.   respond_to :html:xml:json 
  3.    
  4.   def index  
  5.     @people = Person.find(:all)  
  6.     respond_with(@people)  
  7.   end 
  8. end 

這里,我們在類中提供 respond_to。它告訴 Rails:respond_with(在 index 中)應(yīng)接收 HTML、XML、或 JSON 作為響應(yīng)格式。如果用戶請求不同的格式,我們將自動返回一個 406 錯誤(Not Acceptable)。

如果再稍微深入挖掘一下,你會看到 respond_to 方法被定義為:

  1. def respond_to(*mimes)  
  2.   options = mimes.extract_options!  
  3.    
  4.   only_actions   = Array(options.delete(:only))  
  5.   except_actions = Array(options.delete(:except))  
  6.    
  7.   mimes.each do |mime|  
  8.     mime = mime.to_sym  
  9.     mimes_for_respond_to[mime]          = {}  
  10.     mimes_for_respond_to[mime][:only]   = only_actions   unless only_actions.empty?  
  11.     mimes_for_respond_to[mime][:except] = except_actions unless except_actions.empty?  
  12.   end 
  13. end 

這個方法在 ActionController::MimeResponds::ClassMethods 模塊上定義,而該模塊屬于 ActionController::Base。此外,在該模塊的生命周期鉤子 included 中使用 class_inheritable_reader 定義了 mimes_for_respond_to。class_inheritable_reader method (macro?)。 使用 class_eval 將方法添加到正在使用的類上,以模擬內(nèi)置的 attr_accessor 功能。

是否理解所有這些細(xì)節(jié)無關(guān)緊要。重要的是利用上述的 Ruby 功能,我們就可以創(chuàng)建抽象層,從而能夠為 Ruby 語言添加新的特性。

開發(fā)者看到 ActionController::MimeResponds,他無需去了解 class_inheritable_reader 如何運(yùn)行——他只需了解這個基本功能。而看到 API 文檔的開發(fā)者也無需了解 class-levelrespond_to 如何運(yùn)行——他只需了解這個已經(jīng)提供的功能。

這樣,剝離每一層就可以在其他抽象上構(gòu)造一個簡單的抽象。沒有必要一次剝離所有抽象層。

8. 很好的字面含義

在使用 Ruby 編程時,我常常會忘記這一點(diǎn);只有在使用一些字面意義很少或表達(dá)很差的語言時,我才會體會到 Ruby 的這一優(yōu)點(diǎn)。

Ruby 中每個詞都具有很好的字面意義:

  1. 字符串:single-line、double-line、interpolated
  2. 數(shù)字: binary、octal、decimal、hex
  3. 空值:nil
  4. 布爾瑪:true、false
  5. 數(shù)組: [1,2], %w(每個字都是元素)
  6. 哈希表(Hash): {key => value} 和{key: value}(Ruby 1.9)
  7. 正則表達(dá)式:/hello/、%r{hello/path}、%r{hello#{interpolated}}
  8. 符號::name 和 :”weird string”
  9. 區(qū)塊:{ 區(qū)塊文字 }

我想我可能會漏掉一些。雖然有些學(xué)術(shù)性,但可讀性良好的語句的確能夠增強(qiáng)開發(fā)者的編碼能力,讓他們寫出簡短而***表達(dá)性的代碼。

當(dāng)然,通過對新的 Hash 對象實(shí)例化并一個一個地輸入關(guān)鍵字和值,你也可以實(shí)現(xiàn) Hash 的功能。但這減少了 Hash 的用途,比如作為方法參數(shù)。

Hash 字面上的簡潔性讓 Ruby 程序員能夠無需經(jīng)過語言設(shè)計者的許可就能夠添加限制性關(guān)鍵字參數(shù)。這也是自足執(zhí)行的又一個實(shí)例。

9. 所有事物都是對象,所有代碼都是可執(zhí)行的并具有 self

很大程度上,類主體之所以能夠按照這樣的方式運(yùn)行,是 Ruby 語言始終如一地面向?qū)ο蟮慕Y(jié)果。在類主體內(nèi)部,Ruby 僅執(zhí)行具有指向類的 self 的代碼。此外,類內(nèi)容中沒有什么是專用的;可以在任何位置對類語境中的代碼進(jìn)行求值。

比如:

  1. module Util  
  2.   def self.evaluate(klass)  
  3.     klass.class_eval do 
  4.       def hello  
  5.         puts "#{self} says Hello!"   
  6.       end 
  7.     end 
  8.   end 
  9. end 
  10.    
  11. class PersonName < String 
  12.   Util.evaluate(self)  
  13. end 

這完全等同于:

  1. class PersonName < String 
  2.   def hello  
  3.     puts "#{self} says Hello!"   
  4.   end 
  5. end 

Ruby 移除了不同位置代碼之間的人工界限,降低了創(chuàng)建抽象的概念上的成本。這是強(qiáng)大的、始終如一的對象建模的結(jié)果。

有關(guān)該主體,再提供一個示例。Ruby 常見的術(shù)語:possibly_nil && possibly_nil.method_name。由于 nil 只是 Ruby 的一個對象,向它發(fā)送一個它無法理解的信息,會造成一個 NoMethodError 錯誤。有些開發(fā)者建議使用這種句法:possibly_nil.try(:method_name)。可以在 Ruby 中通過以下代碼來實(shí)現(xiàn):

  1. class Object 
  2.   alias_method :try, :__send__  
  3. end 
  4.    
  5. class NilClass 
  6.   def try  
  7.     nil 
  8.   end 
  9. end 

本質(zhì)上,這將為每個對象添加方法 try。當(dāng) Object 是 nil 時,try 只返回 nil。但 Object 不是 nil 時,try 就調(diào)用當(dāng)前所用的方法。

使用 Ruby 開放類的目標(biāo)程序,結(jié)合 Ruby 中所有事物都是對象(包括 nil)這一事實(shí),我們就能夠創(chuàng)建新的 Ruby 功能。同樣,這沒有什么大不了的,不過是又一個示例:在語言中做出正確的選擇,我們就能夠創(chuàng)建有用的抽象。

#p#

10.  Rack

由于 Rack 不是 Ruby 語言的組成部分,所以這一點(diǎn)有點(diǎn)欺騙性。但是,它的確可以演示某些有用的功能。首先,今年早些時候,Rack 庫才發(fā)布 1.0,并且所有單個 Ruby web 框架都已經(jīng)與 Rack 兼容。如果你使用 Ruby 框架,我保證你就可以使用 Rack,并且所有標(biāo)準(zhǔn)的 Rack 中間件也可以運(yùn)行。

做到這一點(diǎn)無需犧牲任何向后的兼容性,這也說明了 Ruby 語言的靈活。Rack 本身也可以利用 Ruby 功能來完成工作。

Rack API 如下:

  1. Rack::Builder.new do 
  2.   use Some::Middleware, param  
  3.   use Some::Other::Middleware  
  4.   run Application  
  5. end 

在這個簡短的代碼片段中,包含了很多東西。首先,一個區(qū)塊被傳遞到 Rack::Builder。第二,該區(qū)塊在一個新的 Rack::Builder 實(shí)例(通過它可以訪問 use 和 run 方法)中進(jìn)行求值。第三,傳遞到 use 和 run 的參數(shù)是類的名字,在 Ruby 中它是一個簡單的對象。這樣,Rack 就能夠調(diào)用 passed_in_middleware.new(app, param),其中 new 是一個調(diào)用 Class 對象 Some::Middleware 的方法。

假如你認(rèn)為上面的實(shí)現(xiàn)可能會需要一堆你所憎惡的代碼,讓我們再看下面:

  1. class Rack::Builder  
  2.   def initialize(&block)  
  3.     @ins = []  
  4.     instance_eval(&block) if block_given?  
  5.   end 
  6.    
  7.   def use(middleware, *args, &block)  
  8.     @ins << lambda { |app| middleware.new(app, *args, &block) }  
  9.   end 
  10.    
  11.   def run(app)  
  12.     @ins << app #lambda { |nothing| app }  
  13.   end 
  14. end 

上面我演示創(chuàng)建了一個新的 Rack 程序,這里就是所需的所有代碼。對中間件鏈進(jìn)行實(shí)例化也很簡單:

  1. def to_app  
  2.   inner_app = @ins.last  
  3.   @ins[0...-1].reverse_each { |app| inner_app = app.call(inner_app) }  
  4.   inner_app  
  5. end 
  6.    
  7. def call(env)  
  8.   to_app.call(env)  
  9. end 

首先,我們從該鏈中取出***一個元素(末點(diǎn)),然后我們以相反的方向遍歷其余元素,使用鏈中的下一個元素對每個中間件進(jìn)行實(shí)例化,并返回結(jié)果對象。

***,我們在 Builder 上定義了一個調(diào)用方法(Rack 尤其要求),它調(diào)用 to_app 并將環(huán)境傳遞過去,結(jié)束這個鏈。

通過本文中描述的這些技巧,利用幾十行的代碼,我們就能夠創(chuàng)建支持 Rack 中間件、兼容 Rack 的應(yīng)用程序。

原文:My 10 Favorite Things About the Ruby Language

作者:Yehuda Katz

【編輯推薦】

  1. Python和Ruby:流行動態(tài)腳本語言之特點(diǎn)對比
  2. Ruby和Python的語法比較
  3. Ruby使用心得匯總:尋找高效的實(shí)現(xiàn)
  4. Ruby on Rails入門之道
  5. Ruby on Rails開發(fā)的五點(diǎn)建議
責(zé)任編輯:yangsai 來源: 51CTO.com
相關(guān)推薦

2009-07-21 10:04:57

Scala編程語言

2021-04-12 09:42:25

漏洞網(wǎng)絡(luò)安全網(wǎng)絡(luò)攻擊

2009-07-16 16:20:46

CMS介紹

2009-12-14 18:54:30

Ruby語言優(yōu)缺點(diǎn)

2011-11-28 09:35:21

云計算云電視

2009-01-07 09:10:59

NetBeansSun6.5

2022-08-16 14:27:56

Java開發(fā)編程

2009-04-13 11:25:28

2011-03-24 13:44:05

2016-07-12 13:45:53

51CTO學(xué)院

2012-03-02 11:30:27

Windows8用戶體驗

2010-05-17 17:54:17

微軟蘋果失敗

2015-12-07 10:22:27

2015-01-04 11:21:52

編程語言

2012-12-27 09:22:45

Win 8操作系統(tǒng)

2010-03-03 15:10:48

2009-03-25 17:41:50

NehalemIntel服務(wù)器

2010-03-03 15:38:28

Linux十大版本

2023-10-08 15:54:12

2023-04-02 13:54:52

Java編程語言開發(fā)
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號