Ruby,一種為簡單快捷物件導向編程而創的腳本語言,在20世紀90年代由日本人松本行弘開發,遵守GPL協議和Ruby License[1]。它的靈感與特性來自於Perl、Smalltalk、Eiffel、Ada以及Lisp語言。由Ruby語言本身還發展出了JRuby(Java平台)、IronRuby(.NET平台)等其他平台的Ruby語言替代品。
Ruby相較之下比其他類似的程式語言(如Perl或Python)年輕,又因為Ruby是日本人發明的,所以早期的非日文資料和程式都比較貧乏,在網上仍然可以找到早期對Ruby的資料太少之類的批評。約於2000年,Ruby開始進入美國,英文的資料開始發展。
2004年,RoR框架誕生,Ruby更加廣為人知,Ruby並於2006年為TIOBE獲選為年度程式語言。
遵循上述的理念,Ruby語言通常非常直觀,按照編程人認為它應該的方式運行。
Ruby的作者認為[來源請求],表示Ruby是一個語法像Smalltalk一樣完全物件導向、腳本執行、又有Perl強大的文字處理功能的程式語言。
或者是在irb互動式命令列的模式下:
注意上面的字串是會對斜線\進行轉義,假如不希望對\進行轉義,Ruby還提供了其他的字串形式。
以下為用RoR使用元編程的範例:
在這個例子中,
下面的代碼可以在Ruby shell中運行,比如irb互動式命令列,或者保存為文件並運行命令
像inject()方法可以接收一個參數和一個區塊。迭代的注入列表的每一個成員,執行函數時保存總和。這同函數程式語言中的foldl函數相類似,比如:
首先區塊接收到了10(inject的參數)當作變數sum,並且1(數組的第一個元素)當作變數element;這會返回11。11又被當作下一步的sum變數,它加上3得到了14。14又被加上了5,最終返回結果19。
歷史
Ruby的作者--松本行弘於1993年2月24日開始編寫Ruby,直至1995年12月才正式公開發佈於fj(新聞群組)。之所以稱為Ruby是取法自Perl,因為Perl的發音與6月的誕生石pearl(珍珠)相同,Ruby選擇以7月的誕生石ruby(紅寶石)命名。Ruby相較之下比其他類似的程式語言(如Perl或Python)年輕,又因為Ruby是日本人發明的,所以早期的非日文資料和程式都比較貧乏,在網上仍然可以找到早期對Ruby的資料太少之類的批評。約於2000年,Ruby開始進入美國,英文的資料開始發展。
2004年,RoR框架誕生,Ruby更加廣為人知,Ruby並於2006年為TIOBE獲選為年度程式語言。
[編輯] Ruby的理念
減少編程時候的不必要的瑣碎時間,令編寫程序的人高興,是設計Ruby語言的Matz的一個首要的考慮;其次是良好的界面設計。他強調系統設計必須強調人性化,而不是一味從機器的角度設想[2]。「 | 人們特別是電腦工程師們,常常從機器著想。他們認為:「這樣做,機器就能運行的更快;這樣做,機器運行效率更高;這樣做,機器就會怎樣怎樣怎樣。」實際上,我們需要從人的角度考慮問題,人們怎樣編寫程序或者怎樣使用機器上應用程序。我們是主人,他們是僕人。 | 」 |
Ruby的作者認為[來源請求],表示Ruby是一個語法像Smalltalk一樣完全物件導向、腳本執行、又有Perl強大的文字處理功能的程式語言。
[編輯] Ruby的Hello World程序
下面是一個在標準輸出設備上輸出Hello World的簡單程序:#!/usr/bin/env ruby puts "Hello, world!"
>>puts "Hello, world!" Hello, world! => nil
[編輯] Ruby的特點
[編輯] 變數與函數的命名規則
乍看之下與Perl的命名規則有些類似,不過Perl的命名用來區分純量、陣列與映射;而Ruby的命名規則用來表示變數與類別的關係。Ruby的變數有以下幾種:- 一般小寫字母、底線開頭:變數(Variable)。
$
開頭:全域變數(Global variable)。@
開頭:實例變數(Instance variable)。@@
開頭:類別變數(Class variable)類別變數被共享在整個繼承鏈中- 大寫字母開頭:常數(Constant)。
=
結尾:賦值方法,相當於其他程式語言的set
開頭的方法,算是一種語法蜜糖。!
結尾:破壞性方法,呼叫這個方法會修改本來的物件,這種方法通常有個非破壞性的版本,呼叫非破壞性的版本會回傳一個物件的副本。?
結尾:表示這個函數的回傳值是個布爾值。
[編輯] 多種字串表示法
Ruby提供了多種字串的表示方法,方便撰寫有大量文字資料的程式。a = "\n這是一個雙引號的字串\n" a = %Q{\n這是一個雙引號的字串\n} a = <a = %/\t這是一個雙引號的字串\n/
a = '這是一個單引號的字串' a = %q{這是一個單引號的字串}
[編輯] 動態修改物件、類別
Ruby是動態語言,你可以在程序中修改先前定義過的類別。 也可以在某個類別的實例中定義該實例特有的方法,這叫做原型方法(prototype)。class MyClass def the_method "general method" end end mc = MyClass.new def mc.the_method "special for this instance." end mc.the_method
[編輯] 強大的反射機制與元編程
Ruby的反射功能相當驚人,甚至可以自行追蹤程式運作,或是取出private變數、攔截方法的呼叫。 常常與『可以動態的修改物件』這項特色結合,做為『元編程』的功能:程式在運行時, 可以由程式設計師提供的資訊,自行生成、修改類別或物件,這項功能大大的提高了撰寫程式碼的效率。 在RoR之中,就大量使用了這種特性。以下為用RoR使用元編程的範例:
class Project < ActiveRecord::Base belongs_to :portfolio has_one :project_manager has_many :milestones end
Project
類別繼承Base
類別,Base
類別內建的belongs_to
、has_one
、has_many
方法,便會根據參數來修改Project類別的內容,並自行建立其他相關的方法。程式設計師可以更專心處理程式的運作,而不必為每個類別重複得撰寫程式碼。[編輯] 其他特色
- 完全物件導向:任何東西都是物件,沒有基礎型別
- 變數沒有類型(動態型別)
- 任何東西都有值:不管是四則運算、邏輯表達式還是一個語句,都有回傳值。
- 運算符重載
- 垃圾回收
- 弱類型[3]
- 不必事先宣告變數
- 在Windows上,載入DLL
[編輯] 比較與批評
[編輯] 讓人驚訝之處
- 在Ruby中,只有
false
和nil
表示false,其它的所有值都表示true(包括0
、0.0
、""
、[]
)[4]。這點和C語言的『用0
代表false
』不同,常使用C的人可能會因此驚訝。 - Ruby的字串是可改變的,這與Java固定不變的字串不同,Java的字串比較像Ruby的
Symbol
。 - Ruby的繼承功能相當脆弱,儘管Ruby是一個物件導向語言,Ruby內的許多規則,卻使得子類別有可能不小心就覆寫了父類別的功能,在 《The Ruby Programming Language》一書中,建議除非程式設計師對一個類別相當瞭解,否則儘可能不要使用繼承。
[編輯] 和Perl 6比較
- CPAN上排名第一名,同時也是Perl 6的開發者的唐鳳(Autrijus / Audrey)說:「Ruby就是『沒有到處打廣告的Perl 6』」。[5]
- 松本行弘在接受歐萊禮(O'Reilly)訪問時,提到「Ruby借用了很多Perl的東西……,Python遠比Perl要少……」、「我認為Ruby這個名字作為Perl之後的一門語言的名字真是再恰當不過了。」[6]
- Perl之父拉里·沃爾(Larry Wall)說:「很多方面上我還是很喜歡Ruby的,這是因為那些部分是從Perl借過去的。:-)」、「我還喜歡Ruby的C<*>一元星號操作符,所以我把它加到Perl 6里面。」[7]
[編輯] 程式範例
|
此條目或章節包含指南或教學內容。 請藉由移除或重寫指南段落來改善條目,或在討論頁提出討論。 |
ruby
。- 一些基本的Ruby代碼:
# Everything, including a literal, is an object, so this works: -199.abs # 199 "ruby is cool".length # 12 "Rick Astley".index("c") # 2 "Nice Day Isn't It?".downcase.split(//).sort.uniq.join # " '?acdeinsty"
- 一些轉換:
puts "What's your favorite number?" number = gets.chomp outputnumber = number.to_i + 1 puts outputnumber.to_s + ' is a bigger and better favorite number.'
[編輯] 集合
- 構造和使用數組:
a = [1,'hi', 3.14, 1, 2, [4, 5]] p a[2] # 3.14 p a.[](2)# 3.14 p a.reverse # [[4, 5], 2, 1, 3.14, 'hi', 1] p a.flatten.uniq # [1, 'hi', 3.14, 2, 4, 5]
- 構造和使用關聯數組:
hash = { :water => 'wet', :fire => 'hot' } puts hash[:fire] # Prints: hot hash.each_pair do |key, value| # Or: hash.each do |key, value| puts "#{key} is #{value}" end # Prints: water is wet # fire is hot hash.delete :water # Deletes :water => 'wet' hash.delete_if {|k,value| value=='hot'} # Deletes :fire => 'hot'
[編輯] 區塊和迭代器
- 有兩個語法用於創建區塊:
{ puts "Hello, World!" } # Note the { braces } #or do puts "Hello, World!" end
- 傳參數的區塊使用閉包Closure:
# In an object instance variable (denoted with '@'), remember a block. def remember(&a_block) @block = a_block end # Invoke the above method, giving it a block which takes a name. remember {|name| puts "Hello, #{name}!"} # When the time is right (for the object) -- call the closure! @block.call("Jon") # => "Hello, Jon!"
- 從方法中返回閉包:
def create_set_and_get(initial_value=0) # Note the default value of 0 closure_value = initial_value return Proc.new {|x| closure_value = x}, Proc.new { closure_value } end setter, getter = create_set_and_get # ie. returns two values setter.call(21) getter.call # => 21
- 迭代調用調用時提供的區塊:
def use_hello yield "hello" end # Invoke the above method, passing it a block. use_hello {|string| puts string} # => 'hello'
- 使用區塊迭代數組:
array = [1, 'hi', 3.14] array.each { |item| puts item } # => 1 # => 'hi' # => 3.14 array.each_index { |index| puts "#{index}: #{array[index]}" } # => 0: 1 # => 1: 'hi' # => 2: 3.14 (3..6).each { |num| puts num } # => 3 # => 4 # => 5 # => 6
[1,3,5].inject(10) {|sum, element| sum + element} # => 19
- 區塊運行在內置的方法中:
File.open('file.txt', 'w') do |file| # 'w' denotes "write mode". file.puts 'Wrote some text.' end # File is automatically closed here File.readlines('file.txt').each do |line| puts line end # => Wrote some text.
- 使用枚舉器和塊求1到10的平方:
(1..10).collect {|x| x*x} # => [1, 4, 9, 16, 25, 36, 49, 64, 81, 100]
[編輯] 類別
下面的代碼定義一個命名為Person的類別。含有一個「initialize」方法,用於構選創建一個新對象,它還有兩個方法,一個重載了<=>比較運算符(這樣Array#sort
可以使用age排序)另一個重載了to_s
方法(這樣Kernel#puts
可以格式化輸出),attr_reader
是Ruby中元數據編程的例子:attr_accessor
為實例變數定義了getter和setter方法,attr_reader
只是一個getter方法。另外,方法中最後的聲明是它的返回值,也允許顯式的使用「return」語句。class Person attr_reader :name, :age def initialize(name, age) @name, @age = name, age end def <=>(person) # Comparison operator for sorting @age <=> person.age end def to_s "#@name (#@age)" end end group = [ Person.new("Bob", 33), Person.new("Chris", 16), Person.new("Ash", 23) ] puts group.sort.reverse
- 下面按age倒序輸出了三個名字:
Bob (33) Ash (23) Chris (16)
留言