Ruby được tạo ra từ24 tháng 2 năm1993, bởi lập trình viên ngườiNhật Bản tên làMatsumoto Yukihiro (松本 行弘), biệt danhMatz. Phiên bản chính thức 0.95 (Đây là phiên bản công khai đầu tiên của Ruby được công bố tạiNhật Bản) vào ngày 21 tháng 12 năm 1995. Ở phiên bản Ruby 0.95, nó đã có nhiều tính năng quen thuộc trong các phiên bản sau của Ruby, bao gồm thiết kếhướng đối tượng, các lớp vớikế thừa, mixin,vòng lặp, xử lý ngoại lệ vàthu gom rác.
Ruby 3.4.0[7] được phát hành vào ngày 25 tháng 12 năm 2024. Phiên bản 3.4.0 cũng cung cấp các nâng cấp về mặt hiệu năng cho các phiên bản Ruby 3.3.x. Phiên bản 3.4.0 này tập trung vào việc bổ sung các tham số chiếu trong tham số khối. Ngoài ra phiên bản cũng lựa chọn Prism là trình phân tích cú pháp măc định và hỗ trợ thêm Happy Eyeballs phiên bản 2 vào thư viện socket, cải thiện YJIT, thêm Modular GC, ...
Thay đổi lớn nhất trong phiên bản 3.4.0 của Ruby là việc xuất hiện tham sốit trong block. Tham số này chính là tham chiếu mặc định cho tham số của một block, tương tự như_1, giúp code trong các block đơn giản trở nên dễ đọc hơn. Ví dụ:
Chuyển trình phân tích cú pháp mặc định từ parse.y sang Prism.
Việc thay đổi trình phân tích cú pháp mặc định sang Prism là một thay đổi nội bộ trong ngôn ngữ Ruby nên người dùng sẽ thấy ít có sự thay đổi. Để sử dụng trình phân tích cú pháp thông thường, sử dụng đối số dòng lệnh--parser=parse.y.
YJIT tiếp tục được cải thiện, mang lại hiệu năng tốt hơn trên hầu hết các benchmark cho cả nền tàngx86-64 vàarm64. Việc sử dụngbộ nhớ cũng được giảm thiểu và có thêm các tùy chọn dòng lệnh mới để quản lý.
Ruby 3.4.0 giớ thiệu một hệ thống garbage collector (GC) dạng mô-đun, cho phép tải các trình thu gom rác khác nhau một cách linh hoạt. Đáng chú ý là sự xuất hiên của một trình Garbage Collector thử nghiệm dựa trên MMTk (viết bằngRust) hứa hẹn một hiệu năng cao.
Việc thay đổi các chuỗi (string) trong các file không có commentfrozen_string_literal giờ đây sẽ đưa ra cảnh báo không dùng nữa (deprecation warning).
Việc phân tán từ khóanil khi gọi phương thức hiện được hỗ trợ.**nil được xử lý tương tự như**{}, không truyền từ khóa và không gọi bất kỳ phương thức chuyển đổi nào.
Nguồn gốc của cái tên "Ruby" là từ một phiên chat online giữa Matsumoto và Ishitsuka Keiji vào ngày 24 tháng 2 năm 1993, trước khi bất kỳ đoạn mã nào được viết cho ngôn ngữ này. Ban đầu, "Coral" và "Ruby" là hai cái tên được đề xuất. Matsumoto chọn tên sau trongemail gửi đến Ishitsuka. Sau này Matz cũng bất ngờ khi phát hiện ra Pearl là viên đá quý tượng trưng cho những người sinh tháng 6, còn Ruby thì tượng trưng cho những người sinh tháng 7. Anh cho rằng cái tên Ruby như thế là phù hợp vì Ruby kế thừa và phát triển nhiều đặc tính từ Perl.[10][11]
Quan điểm chính trong việc thiết kế của Matz là nhằm giảm thiểu các công việc nhàm chán mà họ, các nhà lập trình, buộc phải làm; tiếp đến là nguyên tắc thiết kếgiao diện người dùng (user interface) hiệu quả.[12] Ông nhấn mạnh rằng việc thiết kế hệ thống cần phải tập trung vào con người, hơn là vào máy tính:[13]
Often people, especially computer engineers, focus on the machines. They think, "By doing this, the machine will run faster. By doing this, the machine will run more effectively. By doing this, the machine will something something something." They are focusing on machines. But in fact we need to focus on humans, on how humans care about doing programming or operating the application of the machines. We are the masters. They are the slaves.
Ngôn ngữ Ruby được thiết kế nhằm theonguyên tắc ít gây ngạc nhiên nhất (principle of least surprise-POLS), nghĩa là ngôn ngữ hoạt động theo một cách trực quan hay ít nhất đó cũng là nhận xét mà các nhà lập trình đưa ra. Nguyên tắc này không xuất phát từ Matz và, nói chung, Ruby gần với suy nghĩít ngạc nhiên nhất của 'Matz' hơn.
Ruby là một ngôn ngữ hướng đối tượng: mỗi giá trị đều là một đối tượng (object), bao gồm cáckiểu dữ liệu mà đối với các ngôn ngữ khác, chúng là kiểu cơ bản (primitive) nhưinteger. Mỗi hàm (function) là một phương thức (method). Tên biến (variables) chính là tham chiếu (references) đến các đối tượng, bản thân nó không phải là đối tượng. Ruby hỗ trợkế thừa (inheritance) vớidynamic dispatch,mixin vàsingleton method (thuộc về, và để định nghĩa cho, mộtinstance đơn hơn là định nghĩa dành cholớp). Mặc dù Ruby không hỗ trợđa kế thừa, các lớp vẫn có thể được đưa vào cácmodule dưới dạng cácmixins. Cú pháp dạng thủ tục (procedural syntax) vẫn còn được hỗ trợ, có vẻ như là ngoài tầm vực của mọi đối tượng, nhưng thực sự là thuộc mộtthể hiện của class Object tên là 'main'. Vì class này là cha của mọi class khác, nó trở trên ẩn đối với mọi lớp và đối tượng.
Ruby được xem là mộtngôn ngữ lập trình đa mẫu hình (multi-paradigm programming language): nó cho phép bạn lập trình dạng thủ tục (tạo ra các hàm/biến nằm ngoài phạm vi của các lớp và biến chúng thành một phần của đối tượng gốc, 'self' Object), với khả năng hướng đối tượng (mọi thứ đều là đối tượng) hayhàm (nó có các hàm không có tên (anonymous functions),closures, vàcontinuations; mọi câu lệnh đều có giá trị trả về, và các hàm đều trả về kết quả ước lượng cuối cùng). Nó hỗ trợ mạnh chotự định kiểu (type introspection),reflection vàmeta-programming.
Theo RubyFAQ, "Nếu bạn thíchPerl, bạn sẽ thích Ruby và sẽ thấy thoải mái với cú pháp của nó. Nếu bạn thíchSmalltalk, bạn sẽ thích Ruby và sẽ thấy thoải mái với ngữ nghĩa của nó (semantics). Nếu bạn thíchPython, bạn có thể hoặc không thể dừng lại bởi sự khác biệt lớn trong triết lý hiện thực giữa Python và Ruby/Perl."
Ruby có hai bản thông dịch chính: bộthông dịch Ruby ban đầu (viết tắt làMRI), bản được dùng phổ biến nhất, và JRuby, bộ thông dịch dựa trên ngôn ngữJava. Bộ thông dịch Ruby đã được cài đặt trên nhiều nền tảng khác nhau, bao gồmUnix,Microsoft Windows,DOS,Mac OS X,OS/2,Amiga và một số nền tảng khác. Bản chính thức của Ruby có kèm theo "IRB", là bộ thông dịch dạng dòng lệnh trực tiếp (interactive command-line interpreter) giúp cho việc kiểm tra code nhanh chóng.
Các tên bắt đầu bằng ký tự hoa được xem là hằng, vì thế biến cục bộ nên bắt đầu bằng ký tự thường.
Việc đánh giáBoolean đối với các dữ liệu không phải bool rất chặt chẽ: 0,"" và[] được xem làtrue: Trong C, biểu thức0 ? 1: 0 được xem là 0. Trong Ruby, tuy nhiên, nó lại trả về 1, vì số 0 được xem là "một cái gì đó"; chỉ cónil vàfalse mới được xem là bằngfalse. Một hệ luận đối với quy luật này là theo qui ước, các phương thức của Ruby—ví dụ,biểu thức chính quy tìm kiếm — sẽ trả về các số, chuỗi, danh sách etc. nếu thành công, nhưng lại trả vềnil nếu thất bại (ví dụ, không tìm thấy).
Để biểu diễn một số thực dấu chấm động, ta phải theo quy tắc dùng ký số zero (99.0) hay chuyển đổi tường minh (99.to_f). Việc dùng dấu chấm là không đủ (99.) vì các số vẫn có thể nhận cú pháp có phương thức.
Thiếu một kiểu dữ liệu ký tự ("char"). Điều này có thể gây ngạc nhiên khi duyệt qua chuỗi:"abc"[0] cho ra 97 (một số nguyên, biểu diễn mãASCII của ký tự đầu tiên trong chuỗi); để lấy được"a" dùng"abc"[0,1] (chuỗi con cóchiều dài 1) hay"abc"[0].chr.
Một danh sách các lỗi thường gặp ("gotchas") có thể tra trong cuốn sách của Hal FultonThe Ruby Way, trang 48–64. Tuy nhiên, vì danh sách trong cuốn sách là dựa trên phiên bản cũ của Ruby (version 1.6), một số mục đã được sửa đổi sau khi cuốn sách ấn hành. Ví dụ,retry bây giờ làm việc được vớiwhile,until vàfor, cũng như với iterators.
Lưu ý: Các ví dụ sử dụng chuỗi ký tựUnicode để chạy được cần đặt mộtbiến môi trường là $KCODE="u". Hay là chạy #ruby với tùy chọn là -K u vẫn được. Xem thêm những hạn chế của phiên bản Ruby hiện tại với Unicode tạiruby và unicodeLưu trữ ngày 18 tháng 12 năm 2005 tạiWayback Machine
# Mọi thứ, kể cả tầm thường nhất, là một đối tượng. Vì thế những ví dụ dưới đều chạy được:-199.abs# => 199, abs: giá trị tuyệt đối"ruby is cool".length# => 12"Rick".index("c")# => 2"Nice Day Isn't It?".split(//).uniq.sort.join# => " '?DINaceinsty"
Khởi tạo và xây dựng một mảng kết hợp (Ruby gọi là hash):
hash=Hash.new# Tương đương với hash = {}hash={:water=>'wet',:fire=>'hot'}# Đoạn mã đầu trở nên dư thừa khi# đưa cho hash một đối tượng hash riêng biệt mớiputshash[:fire]# Xuất ra "hot"hash.each_pairdo|key,value|# Hoặc: hash.each do |key, value|puts"#{key} is#{value}"end# Trả về {:water => 'wet',:fire => 'hot'} và xuất ra:# water is wet# fire is hothash.delete:water# Hủy cặp giá trị:water => 'wet' và trả về "wet"hash.delete_if{|key,value|value=='hot'}# Hủy cặp giá trị:fire => 'hot' và trả về {}
{puts'Xin chào, thế giới!'}# lưu ý dấu ngoặc# hoặc:doputs'Xin chào, thế giới!'end
Một block có thể được truyền đến một hàm (method) như một tham số tùy chọn. Nhiều hàm được cài sẵn có tham số như vậy:
File.open('file.txt','w')do|file|# 'w' biểu thị "chế độ ghi chép"file.puts'Viết gì đó'end# Tập tin sẽ tự động đóng ở đâyFile.readlines('file.txt').eachdo|line|putslineend# => 'Viết gì đó'
Thông số đi qua block để trở thành closure (ví dụ):
# Trong một biến thể hiện đối tượng (biểu thị bằng '@'), hãy lưu block này.defremember(&p)@block=pend# Dùng hàm trên, cho block đó một cái tên.remember{|ten|puts"Xin chào,#{ten}!"}# Dùng closure (lưu ý rằng điều này không có áp dụng tự do cho các biến khác):!@block.call("Tèo")# Xuất ra "Xin chào, Tèo!"
Tạo ra một hàm ẩn danh:
proc{|arg|putsarg}Proc.new{|arg|putsarg}lambda{|arg|putsarg}->(arg){putsarg}# được giới thiệu ở Ruby 1.9
Trả về closure từ một hàm:
defcreate_set_and_get(initial_value=0)# Lưu ý initial_value là 0closure_value=initial_value[Proc.new{|x|closure_value=x},Proc.new{closure_value}]endsetter,getter=create_set_and_get# Trả về hai giá trịsetter.call(21)getter.call# => 21# Biến tham số cũng có thể được sử dụng như là một liên kết cho closure,# Vậy nên hàm trên có thể được viết lại như sau:defcreate_set_and_get(closure_value=0)[proc{|x|closure_value=x},proc{closure_value}]end
Trong một hàm, có thể thay đổi block vào lúc gọi hàm:
Đoạn mã sau định nghĩa một lớp tên làPerson. Bên cạnh phương thức khởi tạoinitialize, là phương thức được gọi đến khi cần tạo đối tượng mới, lớp này còn có 2 phương thức khác: một là ghi đè lên toán tử so sánh <=> (vì thếArray#sort có thể sắp xếp theo tuổi) và hai là là ghi đè lên phương thứcto_s (vì thếKernel#puts có thể định dạng đầu ra của nó). Ở đây,attr_accessor là một ví dụ củameta-programming trong Ruby: nó định nghĩa các phương thức dạnggetter vàsetter của biến thực thể, trong khiattr_reader các phương thức dạng 'getter'. Và, câu lệnh cuối cùng trong một phương thức là giá trị trả về của nó, điều này cho phép bỏ qua lệnhreturn.
classPersonattr_reader:name,:agedefinitialize(name,age)@name,@age=name,ageenddef<=>(person)# Định nghĩa toán tử so sánh@age<=>person.ageenddefto_s"#{@name} (#{@age})"endendgroup=[Person.new("Bob",33),Person.new("Chris",16),Person.new("Ash",23)]putsgroup.sort.reverse
Cú pháp trong Ruby tương tự như Perl andPython. Lớp và phương thức được định nghĩa thông qua các từ khóa, ngoài ra khối (block) còn có thể được định nghĩa bằng cặp dấu ngoặc nhọn{}. Khác với Perl,biến số (variable) không nhất thiết phải bắt đầu bằng dấu$, và nếu sử dụng sẽ thay đổi phạm vi của biến số. Các kí tự xuống dòng (line breaks) cũng như dấu chấm phẩy; phân chia các mệnh đề (statement) với nhau.
Một trong những điểm khác biệt lớn nhất so với Perl và Python là Ruby ẩn hoàn toàn các biến của thực thể (instance) và chỉ có thể truy cập và chỉnh sửa chúng thông qua các phương thức (nhưattr_writer,attr_reader).
Ruby Application Archive đóng vai trò là một kho lưu trữ đủ loại ứng dụng và các thư viện viết bằng Ruby, với hàng ngàn mục. Mặc dù số lượng ứng dụng sẵn có không lớn bằng với cộng đồng củaPerl hayPython, vẫn có đủ loại công cụ và tiện ích nhằm hỗ trợ cho việc phát triển ngôn ngữ trong tương lai.
^abcCooper, Peter (2009).Beginning Ruby: From Novice to Professional. Beginning from Novice to Professional (ấn bản thứ 2). Berkeley: APress. tr. 101.ISBN1-4302-2363-4.To a lesser extent, Python, LISP, Eiffel, Ada, and C++ have also influenced Ruby.