Ruby Материал http://wikipedia.org

Ruby (англ. ruby — рубин, произносится ['ɹu:bɪ] — руби) — динамический, рефлективный, интерпретируемый высокоуровневый язык программирования для быстрого и удобного[7][8] объектно-ориентированного программирования. Язык обладает независимой от операционной системы реализацией многопоточности, строгой динамической типизацией, сборщиком мусора и многими другими возможностями. Ruby близок по особенностям синтаксиса к языкам Perl и Eiffel, по объектно-ориентированному подходу — к Smalltalk. Также некоторые черты языка взяты из Python, Lisp, Dylan и CLU (англ.).

Кроссплатформенная реализация интерпретатора языка является полностью свободной[5].

Содержание

Создатель Ruby — Юкихиро Мацумото (Matz) — интересовался языками программирования, ещё будучи студентом, но идея о разработке нового языка появилась позже. Ruby начал разрабатываться 23 февраля 1993 года и вышел в свет в 1995 году.

Название навеяно языком Perl, многие особенности синтаксиса и семантики из которого заимствованы в Ruby: англ. pearl — «жемчужина», ruby — «рубин».

Целью разработки было создание «настоящего объектно-ориентированного», лёгкого в разработке, интерпретируемого языка программирования. Из письма автора[9]:

Ruby родился 23 февраля 1993 года. В тот день я беседовал со своим коллегой о возможности существования объектно-ориентированного сценарного языка. Я знал Perl (Perl4, а не Perl5), но он мне не нравился — был в нём некий привкус игрушечного языка (да и поныне есть). А объектно-ориентированный интерпретируемый язык казался многообещающим. В то время я знал Python. Но он мне не нравился потому, что я не считал его настоящим объектно-ориентированным языком. Его OO свойства казались надстройкой над языком. Мне, как языковому маньяку и фанату объектно-ориентированного программирования с пятнадцатилетним стажем, очень, очень хотелось, чтобы был истинно объектно-ориентированный, простой в использовании язык. Я пытался найти такой язык, но его не было.

Тогда я решил его создать. Прошло несколько месяцев, прежде чем интерпретатор заработал. Я добавил в мой язык то, что мне хотелось — итераторы, обработку исключений, автоматическую сборку мусора. Затем я переорганизовал свойства Perl и реализовал их как библиотеку классов. В декабре 1995 года я опубликовал Ruby 0.95 в японских новостных группах. С тех пор появились сайты, списки рассылок. В списках рассылок идут жаркие обсуждения. Самый старый список сейчас содержит 14 789 писем.

В Японии Ruby стал популярным с момента появления первой общедоступной версии в 1995 году, однако наличие документации только на японском языке сдерживало его дальнейшее распространение. Лишь в 1997 году появилось описание Ruby на английском языке, а в 1998 году открылся форум «ruby-talk». Это положило начало росту известности языка в остальном мире. Издано несколько книг на различных языках, в том числе на русском. Сейчас Ruby входит в большинство дистрибутивов ОС Linux, доступен пользователям других операционных систем.

В списке ниже перечислены лишь наиболее крупные обновления[10].

Название версии Дата выхода Примечания
0.06 7 января 1994 Первая версия, указанная в Changelog’ах
1.0-961225 25 декабря 1996 Данная версия следовала сразу за версией 0.99.4-961224, выпущенной накануне. Номер после числа 1.0 — дата выпуска версии. Новые версии линейки 1.0 выходили ещё год (до 1.0-971225).
1.1 alpha0 13 августа 1997 Альфа-версии выходили вплоть до 7 октября 1997 (1.1 alpha9)
1.1b0 4 декабря 1997 Следующая версия после 1.1 alpha9. 27 февраля 1998 вышла версия 1.1b9, затем вплоть до середины 1998 выходили экспериментальные выпуски с обозначением вида 1.1b9_31 (версия 1.1b9_31 была выпущена, но в документации не отмечена).
1.1c0 17 июля 1998 Данная версия следовала за версией 1.1b9_31. Модификации этой версии выходили вплоть до 26 ноября 1998 (1.1c9).
1.1d0 (pre1.2) 16 декабря 1998 Данная версия следовала за версией 1.1c9. 22 декабря 1998 была выпущена экспериментальная версия 1.1d1, завершившая данную линейку.
1.2 (stable) 25 декабря 1998 В дальнейшем выходили модификации данной версии вплоть до версии 1.2.5, выпущенной 13 апреля 1999 года. 21 июня 1999 года была выпущена версия 1.2.6, объявленная как финальная версия 1.2 (1.2 final). 15 июля 1999 года вышла переупакованная (repacked) версия 1.2.6.
1.3 (development) 24 декабря 1998 Отдельная ветка модификаций, разрабатываемая независимо от линейки 1.2 (по аналогии с ядром ОС Linux). Первая версия была объявлена как версия для разработки (development version) и следовала за версией 1.1d1. В дальнейшем последовало множество промежуточных модификаций: ruby-1.3.1-990215 — ruby-1.3.4-990625, после чего от указания даты в номере отказались и выпустили 1.3.5 — 1.4 alpha (15 июля 1999), 1.3.6 — 1.4 alpha (28 июля 1999), 1.3.7 — 1.4 beta (6 августа 1999).
1.4.0 (stable) 13 августа 1999 Данная версия появилась через несколько дней после выхода 1.3.7 — 1.4 beta. В дальнейшем выходили новые модификации вплоть до версии 1.4.6, вышедшей 16 августа 2000 года.
1.5.0 (development) 20 ноября 1999 Данная линейка предназначалась исключительно для проверки различных нововведений при разработке. Модификации данной линейки доступны исключительно в репозитории проекта и, соответственно, сборки данной версии на официальный сервер не выкладывались.
1.6.0 (stable) 19 сентября 2000 В дальнейшем выпускались модификации этой версии вплоть до версии 1.6.8 (24 декабря 2002). 21 сентября 2005 года был выпущен патч для версии 1.6.8.
1.7.0 (development) 24 февраля 2001 Данная линейка предназначалась исключительно для проверки различных нововведений при разработке. Модификации данной линейки доступны исключительно в репозитории проекта и, соответственно, сборки данной версии на официальный сервер не выкладывались.
1.8.0 (stable) 4 августа 2003 В дальнейшем последовало большое число модификаций, которые выходят до сих пор (1 января 2011 года), например, промежуточная версия 1.8.7-p330 вышла 24 декабря 2010 года.
1.9.0 (development) 25 декабря 2007 Изначально экспериментальная ветка созданная для практической проверки ряда нововведений.
1.9.3 (stable) 31 октября 2011 Последняя ветка на данный момент (26 апреля 2012 года). Соответственно, модификации также выходят до сих пор. Отличия от 1.9.2 - существенны.

Мацумото, фанат объектно-ориентированного программирования, мечтал о языке, более мощном, чем Perl, и более объектно-ориентированном, чем Python. Основное назначение Ruby — создание простых и в то же время понятных программ, где важна не скорость работы программы, а малое время разработки, понятность и простота синтаксиса.

Язык следует принципу «наименьшей неожиданности»: программа должна вести себя так, как ожидает программист. Однако в контексте Ruby это означает наименьшее удивление не при знакомстве с языком, а при его основательном изучении. Сам Мацумото утверждает, что целью разработки была минимизация неожиданностей при программировании для него, но после распространения языка он с удивлением узнал, что мышление программистов похоже, и для многих из них принцип «наименьшей неожиданности» совпал с его принципом.

Ruby также унаследовал идеологию языка программирования Perl в части предоставления программисту возможностей достижения одного и того же результата несколькими различными способами. Люди различны, и им для свободы необходима возможность выбирать. «Я предпочитаю обеспечить много путей, если это возможно, но поощрять или вести пользователей, чтобы выбрать лучший путь, если это возможно» [11].

Одной из основных целей разработки было освобождение программистов от рутинной работы, которую компьютер может выполнять быстрее и качественнее. Особое внимание, в частности, уделялось будничным рутинным занятиям (обработка текстов, администрирование), и для них язык настроен особенно хорошо.

В противовес машинно-ориентированным языкам, работающим быстрее, целью этой разработки был язык, наиболее близкий к человеку. Любая работа с компьютером выполняется людьми и для людей, и необходимо заботиться в первую очередь о затрачиваемых усилиях людей. Язык позволяет максимально быстро и просто для человека выполнить задачу, хотя, возможно, это и потребует дополнительного времени работы компьютера.

Принципы программирования и дизайна языка иногда выделяются в термин «Путь Ruby» (англ. Ruby Way). Хэл Фултон выделяет[12] такие принципы, как «просто, но не слишком просто», «принцип наименьшего удивления», вторичность скорости работы программы, динамичность, простые строгие правила, выполнение которых не доходит до педантизма, потребность создавать полезные и красивые программы как причина программирования. В целом они не имеют точной формулировки и иногда этот термин используется для критики [13].

Ruby — полностью объектно-ориентированный язык. В нём все данные являются объектами, в отличие от многих других языков, где существуют примитивные типы. Каждая функция — метод.

Переменные Ruby передаются по ссылке. Для программиста, привычного к распространённым гибридным языкам программирования, некоторые эффекты такого решения могут показаться неожиданными. Например:

  a = "abcdefg"
  b = a
  b             #=> "abcdefg"
  a[3] = 'R'
  b             #=> "abcRefg"

то есть при изменении значения переменной a, неявно изменилось и значение b, так как они содержат ссылку на один объект. То есть механизм присваивания действует одинаково для всех объектов, в отличие от языков типа C, Object Pascal, где присваивание может означать как копирование значения, так и копирование ссылки на значение.

Ruby не поддерживает множественное наследование, но вместо него есть мощный механизм примесей. Все классы (напрямую или через другие классы) выведены из класса Object, следовательно, любой объект может использовать определённые в нём методы (например, class, to_s, nil?). Процедурный стиль также поддерживается, но все глобальные процедуры неявно являются закрытыми методами класса Object.

Ruby является мультипарадигменным языком: он поддерживает процедурный стиль (определение функций и переменных вне классов), объектно-ориентированный (всё — объект), функциональный (анонимные функции, замыкания, возврат значения всеми инструкциями, возврат функцией последнего вычисленного значения). Он поддерживает отражение, метапрограммирование, информацию о типах переменных на стадии выполнения (см. динамическая идентификация типа данных).

  • Имеет лаконичный и простой синтаксис, частично разработанный под влиянием Ада, Eiffel и Python.
  • Позволяет обрабатывать исключения в стиле Java и Python.
  • Позволяет переопределять операторы, которые на самом деле являются методами.
  • Полностью объектно-ориентированный язык программирования. Все данные в Ruby являются объектами в понимании Smalltalk. Единственное исключение — управляющие конструкции, которые в Ruby в отличие от Smalltalk не являются объектами. Например, число «1» — это экземпляр класса Fixnum. Также поддерживается добавление методов в класс и даже в конкретный экземпляр во время выполнения программы.
  • Не поддерживает множественное наследование, но вместо него может использоваться концепция «примесей», основанная в данном языке на механизме модулей.
  • Содержит автоматический сборщик мусора. Он работает для всех объектов Ruby, в том числе для внешних библиотек.
  • Создавать расширения для Ruby на Си очень просто частично из-за сборщика мусора, частично из-за несложного и удобного API.
  • Поддерживает замыкания с полной привязкой к переменным.
  • Поддерживает блоки кода (код заключается в {} или doend). Блоки могут использоваться в методах или преобразовываться в замыкания.
  • Целые переменные в Ruby автоматически конвертируются между типами Fixnum (32-разрядные) и Bignum (больше 32 разрядов) в зависимости от их значения, что позволяет производить целочисленные математические расчёты со сколь угодно большой точностью.
  • Не требует предварительного объявления переменных, но для интерпретатора желательно, чтобы переменным присваивалось пустое значение nil (тогда интерпретатор знает, что идентификатор обозначает переменную, а не имя метода). Язык использует простые соглашения для обозначения области видимости. Пример: просто var — локальная переменная, @var — переменная экземпляра (член или поле объекта класса), @@var — переменная класса, $var — глобальная переменная.
  • В Ruby непосредственно в языке реализованы многие шаблоны проектирования, так, например, «одиночка» (singleton) может быть (хотя и не обязан) реализован добавлением необходимых методов к одному конкретному объекту (см. ниже).
  • Может динамически загружать расширения, если это позволяет операционная система.
  • Имеет независимую от ОС поддержку невытесняющей многопоточности.
  • Перенесён на множество платформ. Он разрабатывался на Linux, но работает на многих версиях Unix, DOS, Microsoft Windows (в частности, Win32), Mac OS, BeOS, OS/2 и т. д.

В Ruby есть немало оригинальных решений, редко или вообще не встречающихся в распространённых языках программирования. Можно добавлять методы не только в любые классы, но и в любые объекты. Например, вы можете добавить к некоторой строке произвольный метод.

                   # всё от символа # и до конца строки - комментарий
                   # = является оператором присваивания,
                   # символы в «"» - строка, которой можно манипулировать средствами языка
  str = "Привет"   # здесь создаётся переменная str, типа String
                   # def - ключевое слово для объявления функции
  def str.bye      # str. указывает, кому принадлежит метод (по умолчанию Object)
                   # bye - имя метода, за ним может следовать необязательный, заключённый в
                   # круглые скобки список параметров функции
    "Пока!"        # из метода возвращается последнее вычисленное значение (здесь - строка)
  end              # ключевым словом end заканчиваются практически все инструкции Ruby
                   # puts - метод,
                   # str.bye - обращение к методу bye объекта str
                   # значение, полученное из метода bye передаётся методу puts
                   # который выводит на экран информацию
  puts str.bye     #=> Пока!

Этот пример также демонстрирует, как в Ruby можно использовать синглтон. В этом примере синглтоном является объект str.

Любая конструкция в Ruby возвращает значение. Например:

                   # конструкция if вычисляет выражение после него, и, если оно истинно,
                   # возвращает результат выражения между then и else, иначе между else и end
  puts( if 5 > 3 then "Одно" else "Другое" end )   #=> Одно
                   # происходит присваивание значения переменной var, и операция присваивания
                   # возвращает значение переменной var, которая выводится на экран
  puts( var = 5 )                                  #=> 5

Работа с массивами — одна из сильных сторон Ruby. Они автоматически изменяют размер, могут содержать любые элементы, и язык предоставляет мощные средства для их обработки.

                           # создаём массив
 a = [1, 'hi', 3.14, 1, 2, [4, 5] * 3]
                           # => [1, "hi", 3.14, 1, 2, [4, 5, 4, 5, 4, 5]]
                           # обращение по индексу
 a[2]                      # => 3.14
                           # «разворачиваем» все внутренние массивы, удаляем одинаковые элементы
 a.flatten.uniq            # => [1, 'hi', 3.14, 2, 4, 5]
                           # пытаемся найти индекс элемента со значением 4
 a.index(4)                # => nil
                           # предыдущая попытка найти элемент неудачна - все предыдущие функции
                           # возвращают копии, но Ruby почти для всех функций предоставляется
                           # аналог с тем же названием, но заканчивающийся на «!»,
                           # который модифицирует контейнер
 a.flatten!                # => [1, "hi", 3.14, 1, 2, 4, 5, 4, 5, 4, 5]
 a.index(4)                # => 5

В языке есть 2 эквивалентных способа записи блоков кода:

 { puts "Hello, World!" }
 
 do puts "Hello, World!" end

Сопрограммы применяются с большинством встроенных методов:

 File.open('file.txt', 'w') {|file| # открытие файла «file.txt» для записи («w» - write)
   file.puts 'Wrote some text.'
 } # Конструкция устраняет неопределённость с закрытием файла: закрывается здесь при любом исходе

Следующий пример показывает использование сопрограмм и итераторов для работы с массивами, который показывает краткость записи на Ruby многих достаточно сложных действий (случайно выбираем из последовательности квадратов чисел от «0» до «10» и распечатываем вместе с индексами):

# для работы требуется ruby 1.9
(0..10).collect{|v| v ** 2 }.select{ rand(2).zero? }.map.with_index {|*v| v }

Следующий пример определяет класс с именем Person, предназначенный для хранения информации об имени и возрасте некоторой персоны.

 class Person                # объявление класса начинается с ключевого слова class, за которым
                             # следует имя
   include Comparable        # подмешивается к классу
                             # и добавляет методы <, <=, ==, >=, > и between?
                             # с использованием нижеопределённого
                             # в классе <=>
                             #
   @@count_obj = 0           # переменная класса для подсчёта числа созданных объектов
                             #
                             # конструктор для создания объектов с помощью new
   def initialize(name, age) # name, age - параметры метода
                             # название переменных объекта начинается с @
     @name, @age = name, age # создаём объекты и увеличиваем счётчик на 1
     @@count_obj += 1
   end
 
   def <=>(person)           # переопределение оператора <=>
                             # (это даёт возможность использовать метод sort)
     @age <=> person.age     # из метода возвращается последнее вычисленное выражение,
   end
 
   def to_s                  # для форматированного вывода информации puts
     "#{@name} (#{@age})"    # конструкция #{x} в 2-х кавычках замещается в Ruby значением x
   end
 
   def inspect               # похож на to_s, но используется для диагностического вывода
     "<#{@@count_obj}:#{to_s}>"
   end
                             # пример метапрограммирования: добавляет методы для доступа к
                             # переменным объекта
   attr_reader :name, :age
 end
                             # создаём массив объектов
 group = [ Person.new("John", 20),
          Person.new("Markus", 63),
          Person.new("Ash", 16) ]
                      # => [<3:John (20)>, <3:Markus (63)>, <3:Ash (16)>]
                      # здесь при работе с irb автоматически вызывается метод inspect
                      # вызываем методы массива сортировка и разворачивание его в обратном порядке
 puts group.sort.reverse # Печатает:
                         # Markus (63)
                         # John (20)
                         # Ash (16)
                                    # обращаемся к функции, которая была добавлена
                                    # автоматически(используя <=>) при включении Comparable
 group[0].between?(group[2], group[1]) # => true

Исключения возбуждаются с помощью конструкции raise (или fail), опционально могут быть добавлены текст с сообщением, тип исключения и информация о стеке вызовов:

 raise ArgumentError, "Неверный аргумент", caller # caller - метод, возвращающий текущий стек выполнения

Обрабатываются исключения с использованием конструкции rescue. Опционально можно указать тип обрабатываемого исключения (по умолчанию обрабатываются все) и получение информации. Также можно добавлять блоки else (выполняется если исключения отсутствовали) и ensure (выполняется в любом случае).

 begin
   # ...
 rescue RuntimeError => e
   # обрабатываем конкретный тип ошибок
   puts e # напечатаем сообщение об ошибке
 rescue
   # можно писать rescue => e чтобы получить объект исключения
   # обрабатываем все исключения
 else
   # сработает, если исключений не было
 ensure
   # сработает в любом случае
 end

Для Ruby существуют несколько реализаций: официальный интерпретатор, написанный на Си, JRuby — реализация для Java, интерпретатор для платформы .NET IronRuby, Rubinius — написанная в основном на Ruby и базирующаяся на идеях Smalltalk-80 VM[14], MagLev — другая базирующаяся на Smalltalk разработка от компании Gemstone[15], Blue Ruby — реализация Ruby для виртуальной машины ABAP[16], MacRuby — реализация для Mac OS с фокусом на максимальную интеграцию с возможностями операционной системы[17].

Официальный интерпретатор портирован под большинство платформ, включая Unix, Microsoft Windows (в том числе Windows CE), DOS, Mac OS X, OS/2, Amiga, BeOS, Syllable, Acorn RISC OS и другие.

Для Windows существует RubyInstaller for Windows и есть возможность запуска под Cygwin для большей совместимости с Unix[18]

С официальной версией интерпретатора Ruby поставляется командная оболочка Ruby (Interactive Ruby Shell). Запускаемая командой irb в окне терминала (интерфейсе командной строки), она позволяет тестировать код программы очень быстро (построчно):

 $ irb
 irb(main):001:0> "Hello, World"
 => "Hello, World"
 irb(main):002:0> 2 ** 256             # ** - оператор возведения в степень
 => 115792089237316195423570985008687907853269984665640564039457584007913129639936

Программа irb выводит результат каждой строки после символов =>. В приведённых выше примерах для наглядности применяется аннотирование — результаты строк программы записываются в комментариях после =>. Имитацию irb можно запустить непосредственно в браузере.

В поставке дистрибутива One-Click Installer для Windows, начиная с версии 1.8.2-15, поставляется утилита fxri, которая включает в себя справочную систему (ri) и интерактивный интерпретатор (irb).

Базовые возможности редактирования добавляются к многим редакторам (Emacs, vim, jEdit, nano, SciTE и др.), здесь перечислены только IDE, то есть предоставляющие обширный набор функций.

Название Лицензия Платформы Ссылка
ActiveState Komodo IDE Проприетарная Linux, Mac OS X, Solaris, Windows http://activestate.com/Products/komodo_ide/
Arachno Ruby IDE Проприетарная Win 2000/XP, Linux http://www.ruby-ide.com
Aptana (RadRails + RDT) GPL, APL + CPL Java http://www.aptana.com/
EasyEclipse for Ruby and Rails Win 2000/XP, Linux, Mac OS X http://www.easyeclipse.org/site/distributions/ruby-rails.html
Eclipse + RDT EPL + CPL Java RDT Homepage
Embarcadero TurboRuby Проприетарная Windows, OS X, Linux http://www.embarcadero.com/products/turboruby
FreeRIDE Ruby License Windows, OS X, POSIX http://rubyforge.org/projects/freeride/
IntelliJ IDEA + Ruby plugin Проприетарная (на IDEA), Apache 2.0 (на сам plugin) Java, JRuby Ruby plugin
KDevelop GNU GPL Linux http://kdevelop.org/
Komodo IDE 6 Проприетарная Windows, Mac, Linux http://www.activestate.com/komodo-ide
Mondrian Ruby IDE MIT Ruby (+ FOX toolkit) http://www.mondrian-ide.com/
NetBeans IDE (версия 6.9.1 и более ранние) CDDL Java http://www.netbeans.org/features/ruby/
RDE Ruby License Windows http://homepage2.nifty.com/sakazuki/rde_en/
Ruby in steel Проприетарная Visual Studio 2005 http://www.sapphiresteel.com/
RubyMine Проприетарная (на базе IDEA) Java http://www.jetbrains.com/ruby
Visual Studio (реализация IronRuby) Проприетарная Windows http://www.microsoft.com/visualstudio/en-us/products
Xcode 3.1 Проприетарная Mac OS X 10.5 http://developer.apple.com/tools/xcode/

Кроме мощных возможностей, встроенных в язык, Ruby поставляется с большой стандартной библиотекой. Это, прежде всего, библиотеки для работы с различными сетевыми протоколами на стороне сервера и клиента, средства для работы с различными форматами представления данных (XML, XSLT, YAML, PDF, RSS, CSV, WSDL). Кроме встроенных в язык средств отладки, с Ruby поставляются библиотеки для юнит-тестирования, профилирования. Также есть библиотеки для работы с архивами, датами, кодировками, матрицами, средства для системного администрирования, распределённых вычислений, поддержки многопоточности и т. д.

В языке Ruby реализован простой и удобный механизм для расширения языка с помощью библиотек, написанных на Си, позволяющий легко разрабатывать дополнительные библиотеки.[20][21]

Для унифицированного доступа к базам данных разработана библиотека Ruby DBI (поддерживает SQLite, Oracle, ODBC, MySQL, DB2, MS SQL, InterBase, ADO и др.). Также существуют библиотеки для конкретных баз данных, поддерживающих специфические для них операции. Для реализации ORM существуют несколько библиотек, такие как ActiveRecord, DataMapper или Sequel.

Из графических библиотек следует отметить FxRuby — интерфейс к графической библиотеке FOX, графический пакет разработчика wxRuby (интерфейс к кроссплатформенному пакету wxWidgets на C++), QtRuby/Korundum — привязка к Qt и KDE соответственно, графические библиотеки для работы с Tk и Gtk. Также реализована библиотека для работы с OpenGL, позволяющая программировать трёхмерную графику.

Win32utils — позволяет обращаться к специфическим возможностям Win32 API.

Rmagick — библиотека для работы с изображениями, поддерживающая более 90 форматов (основана на ImageMagick и GraphicsMagick).

Библиотека Ruport (Ruby reports) предназначена для лёгкой реализации отчётов и создания диаграмм на основе данных из БД или прямо из текстовых файлов CSV. Причём результаты можно сохранять в форматах PDF, HTML, CSV и TXT.

RuTils — обработчик русского текста на Ruby. Позволяет реализовать сумму прописью и выбор числительного. Например, 231.propisju(2) => «двести тридцать одна» или 341.propisju_items(1, «чемодан», «чемодана», «чемоданов») => «триста сорок один чемодан». А также перевод в транслит и работу с датами.

Для управления библиотеками и программами Ruby в виде самодостаточных пакетов предназначена система управления пакетами RubyGems (англ. gems, gem — драгоценный камень).

Существует всемирный репозиторий программного обеспечения Ruby RAA (Ruby Application Archive) [1]. Репозиторий по состоянию на сентябрь 2007 года насчитывает более полутора тысяч проектов. Большое количество программного обеспечения, написанного на Ruby, пользуются хостингом проекта RubyForge[2], созданного специально с этой целью.

FreeRIDE — IDE для Ruby, реализованная с использованием библиотеки FxRuby.

Большинство расширений распространяются под свободными лицензиями (LGPL, лицензия Ruby) и могут быть использованы в любом проекте практически без ограничений.

Система RDoc предназначена для автоматического извлечения документации из исходных кодов и программ на Ruby и её дальнейшей обработки. Является стандартом де-факто для подготовки документации по программному обеспечению, написанному на Ruby.

Для доступа к документации Ruby из командной строки Unix разработана программа ri. С её помощью можно получить информацию о модулях, классах и методах Ruby. Онлайн-документация доступна на сайте http://www.ruby-doc.org.

Ruby используется в NASA, NOAA (национальная администрация по океану и атмосфере), Motorola и других крупных организациях.[22] Следующие программы используют Ruby как скриптовый язык для расширения возможностей программы или написаны на нём (частично или полностью).

Преобразование списка аргументов метода в массив

def my_method(*args)
  args.map {|arg| arg.reverse }
end
 
my_method('abc' , 'xyz' , '123' ) # => ["cba", "zyx", "321"]

Использование псевдоимен (aliases). Вызов старой версии метода из переопределенного метода.

class String
  alias :old_reverse :reverse
 
  def reverse
    "x#{old_reverse}x"
  end
end
 
"abc".reverse # => "xcbax"

Удаление метода из объекта, чтобы сделать его Ghost Method

class C
  def method_missing(name, *args)
    "a Ghost Method"
  end
end
 
obj = C.new
obj.to_s # => "#<C:0x357258>"
 
class C
  instance_methods.each do |m|
    undef_method m unless m.to_s =~ /object_id|method_missing|respond_to?|^__/
  end
end
 
obj.to_s # => "a Ghost Method"

Определение новых методов класса добавлением методов (mixins) из другого модуля. В качестве холдера новых методов используется eigenclass целевого класса. Особый случай Object Extention.

class C; end
  module M
    def my_method
      'a class method'
    end
  end
 
class << C
  include M
end
 
C.my_method # => "a class method"

Использование Hook методов для расширения класса.

module M
  def self.included(base)
    base.extend(ClassMethods)
  end
 
  module ClassMethods
    def my_method
      'a class method'
    end
  end
end
 
class C
  include M
end
 
C.my_method # => "a class method"

Сохранение состояния класса в экземплярных переменных объекта Class.

class C
  @my_class_instance_variable = "some value"
 
  def self.class_attribute
    @my_class_instance_variable
  end
end
 
C.class_attribute # => "some value"

Вызов методов класса в определении класса.

class C
end
 
class << C
  def my_macro(arg)
    "my_macro(#{arg}) called"
   end
end
 
class C
  my_macro :x # => "my_macro(x) called"
end

Использование специального объекта, для безопасного выполнения внешнего блока кода. Используется для создания DSL.

class CleanRoom
  def a_useful_method(x)
    x * 2
  end
end
 
CleanRoom.new.instance_eval { a_useful_method(3) }    # => 6

Выполнение строки кода из внешнего источника.

File.readlines("a_file_containing_lines_of_ruby.txt" ).each do |line|
  puts "#{line.chomp} ==> #{eval(line)}"
end
 
# >> 1 + 1 ==> 2
# >> 3 * 2 ==> 6
# >> Math.log10(100) ==> 2.0

Выполнение блока кода с доступом к внутреннему состоянию объекта.

class C
  def initialize
    @x = "приватная переменная экземпляра"
  end
end
 
obj = C.new
obj.instance_eval { @x } # => “приватная переменная экземпляра”

Сохранение блока кода вместе с контекстом в proc или lambda для выполнения в будущем.

class C
  def store(&block)
    @my_code_capsule = block
  end
 
  def execute
    @my_code_capsule.call
  end
end
 
obj = C.new
obj.store { $X = 1 }
$X = 0
obj.execute
$X # => 1

Принятие решения, о том какой именно метод будет вызван в рантайме.

method_to_call = :reverse
obj = "abc"
obj.send(method_to_call) # => "cba"

Определение нового метода на этапе исполнения.

class C
end
 
C.class_eval do
  define_method :my_method do
    "a dynamic method"
  end
end
 
obj = C.new
obj.my_method # => "a dynamic method"

Добавление дополнительного функционала (cross-cutting concerns) в методы класса.

class MyDynamicProxy
  def initialize(target)
    @target = target
  end
 
  def method_missing(name, *args, &block)
    "result: #{@target.send(name, *args, &block)}"
  end
end
 
obj = MyDynamicProxy.new("a string" )
obj.reverse # => "result: gnirts a"

Использование замыканий для шаринга переменных между двумя областями видимости.

class C
  def an_attribute
    @attr
  end
end
 
obj = C.new
a_variable = 100
 
# flat scope:
obj.instance_eval do
  @attr = a_variable
end
obj.an_attribute # => 100

Возврат результата для метода, который не найден в определении класса.

class C
  def method_missing(name, *args)
    name.to_s.reverse
  end
end
 
obj = C.new
obj.my_ghost_method # => "dohtem_tsohg_ym"

Перехват событий объектной модели ruby, например, расширение класса, добавление метода, и т. д.

$INHERITORS = []
class C
  def self.inherited(subclass)
    $INHERITORS << subclass
  end
end
 
class D < C
end
 
class E < C
end
 
class F < E
end
 
$INHERITORS # => [D, E, F]

Определение метода в модуле Kernel делает его доступным везде.

module Kernel
  def a_method
    "a kernel method"
  end
end
 
a_method # => "a kernel method"

Ленивая инициализация переменной.

class C
  def attribute
    @attribute = @attribute || "some value"
    # или
    # @attribute ||= "some value"
  end
end
 
obj = C.new
obj.attribute # => "some value"

Метод как другая конструкция языка, например, метод как класс.

def BaseClass(name)
  name == "string" ? String : Object
end
 
class C << BaseClass "string" # метод, который выглядит, как класс
  attr_accessor :an_attribute   # метод, который выглядит, как keyword
end
 
obj = C.new
obj.an_attribute = 1 # метод, который выглядит, как атрибут

Изменение функционала существующего класса.

"abc".reverse # => "cba"
 
class String
  def reverse
     "override"
  end
end
 
"abc".reverse # => "override"

Упаковка аргументов метода в хэш массив.

def my_method(args)
  args[:arg2]
end
 
my_method(:arg1 => "A" , :arg2 => "B" , :arg3 => "C" ) # => "B"

Замена nil на альтернативный объект (без проверки на nil)

x = nil
y = x || "a value" # => "a value"

Определение сингелтон-методов в конкретном объекте, путем подмешивания (mixing) методов из другого модуля.

obj = Object.new
module M
  def my_method
     'a singleton method'
  end
end
 
class << obj
  include M
end
 
obj.my_method # => "a singleton method"

Добавление новых методов в существующий класс.

class String
  def my_string_method
    "my method"
  end
end
 
"abc".my_string_method # => "my method"

Выбор метода, для вызова, основываясь на его имени.

$x = 0
 
class C
  def my_first_method
    $x += 1
  end
 
  def my_second_method
    $x += 2
  end
end
 
obj = C.new
obj.methods.each do |m|
  obj.send(m) if m.to_s =~ /^my_/
end
 
$x # => 3

Выполнение небезопасного кода в безопасном окружении.

def sandbox(&code)
  proc {
    $SAFE = 2
    yield
  }.call
end
begin
  sandbox { File.delete 'a_file' }
  rescue Exception => ex
  ex # => #<SecurityError: Insecure operation `delete' at level 2>
end

Изолирование переменных областью видимости. В ruby нет вложенных областей видимости!

a = 1
defined? a # => "local-variable"
 
module MyModule
  b = 1
  defined? a # => nil
  defined? b # => "local-variable"
end
 
defined? a # => "local-variable"
defined? b # => nil

Определение cинглотон-метода в конкретном объекте.

obj = "abc"
 
class << obj
  def my_singleton_method
    "x"
  end
end
 
obj.my_singleton_method # => "x"

или

obj = "abc"
 
def obj.my_singleton_method
  "x"
end
 
obj.my_singleton_method # => "x"

Преобразование символа в proc с последующим вызовом. Полезная техника при работе с массивами.

[1, 2, 3, 4].map(&:even?) # => [false, true, false, true]

  1. 1 2 О Ruby.
  2. Расширения файлов Ruby:
    .rb — исходный код.
    .rbw — исходный код для выполнения программ с графическим интерфейсом в Windows, файлы с таким расширением будут выполнены интерпретатором rubyw без запуска консоли.
  3. Репозиторий Ruby.
  4. Перевод на русский язык лицензии Ruby в Викитеке.
  5. 1 2 Лицензия Ruby  (англ.) (TXT). Архивировано из первоисточника 22 августа 2011. Проверено 14 августа 2006. Перевод лицензии Ruby в Викитеке.
  6. COPYING.txt на официальном сайте Ruby.
  7. Е. А. Роганов, Н. А. Роганова. Программирование на языке Ruby. Учебное пособие (PDF, 425 Кбайт). — М.: МГИУ, 2008. — 56 с. — ISBN 978-5-2760-1495-1
  8. Брюс Тэйт Практическое использование Rails: Часть 4. Стратегии тестирования в Ruby on Rails. 01.07.2008.
  9. Письмо Юкихиро Мацумото в рассылку ruby-talk ruby-talk:00382 от 4 июня 1999 года. Есть перевод всего письма на русский
  10. Информация взята из сопроводительных текстовых файлов Changelog. Документы различных версий различаются между собой, порой учитывая лишь версии, относящиеся к линейке, сопровождаемой данным документом.
  11. Интервью Юкихиро Мацумото (англ.)
  12. Хэл Фултон. Программирование на языке Ruby. — 2-е изд. — М.: ДМК Пресс, 2007. — С. 24—27.
  13. 5 things I hate about Ruby  (англ.) (25 октября 2005). Архивировано из первоисточника 22 августа 2011. Проверено 30 июля 2007.
  14. Rubinius Home
  15. MagLev /măg-lĕv/
  16. Blue Ruby — New Exploratory Research
  17. MacRuby
  18. Cygwin and Ruby Windows
  19. Из-за слабой документированности Ruby в ранних версиях информация получена напрямую из исходников (helium.ruby-lang.org/ftp://ruby  (англ.). Проверено 16 августа 2006.). Указаны только стабильные (чётные) версии.
  20. Thomas, Dave. Extending Ruby  (англ.). Programming Ruby — The Pragmatic Programmer’s Guide. Addison Wesley Longman, Inc.(недоступная ссылка) Проверено 30 октября 2006.
  21. Rooney, Garrett. Extending Ruby with C  (англ.). O’Reilly Media (18 ноября 2004). Архивировано из первоисточника 22 августа 2011. Проверено 30 октября 2006.
  22. Хэл Фултон. Программирование на языке Ruby. — 2-е изд. — М.: ДМК Пресс, 2007. — С. 21.

На русском языке

В электронном формате:

На английском языке

В электронном формате: