До сих пор мы видели несколько различных видов, или классов, объектов: строки, целые числа, дробные числа, массивы, а также несколько особых объектов (true, false и nil), о которых мы поговорим позже. В Ruby эти классы всегда записываются с заглавной буквы: String, Integer, Float, Array… и т. д. В общем случае, если мы хотим создать новый объект определённого класса, мы используем new:
a = Array.
new +
[12345]
#
Сложение
массивов.
b = String
. new +
'hello'
#
Сложение
строк.
c = Time.new
puts 'a =
'+a.to
s
puts 'b =
1 o t b . +
s
puts 'c =
'+c.to
s
a = 12345 b = hello
c = Wed Jun 28 02:11:24 GMT 2006
Так как мы можем создавать массивы и строки с помощью[…] и '…' соответственно, мы редко создаём их с помощью new. (Хотя это и не совсем очевидно из предыдущего примера, String.new создаёт пустую строку, а Array.new создаёт пустой массив.) Кроме того, числа являются особыми исключениями: вы не можете создать целое число с помощью Integer.new. Вам придётся просто записать число.
Класс TIME
И что же особенного в этом классе Time? Объекты класса Time представляют моменты времени. Вы можете прибавлять числа к (или вычитать из) объектов времени, чтобы получить новые моменты времени: прибавление 1.5 к моменту времени создаст новый момент, который на полторы секунды позже первого:
time = Time.new
#
Момент, когда вы получили эту web–страницу.
time2 = time + 60
#
Одной минутой позже.
puts time
puts time2
Wed
Jun
28
02
4 2 1 1
GMT
2006
Wed
Jun
28
02
12:24
GMT
2006
Вы также можете создавать объект времени, соответствующий определённому моменту, используя т^е.тк-^те:
puts
Time.mktime(2000,
1,
1)
# Двухтысячный год (Y2K).
puts
Time.mktime(1976,
8,
3, 10, 11)
# Когда я родился.
Sat
Jan
01
00:00:00
GMT
2000
Tue
u <
3 0
10:11:00
T M G
6 7 9 1
Обратите внимание: момент моего рождения задан по Тихоокеанскому летнему времени (Pacific Daylight Savings Time, PDT). Хотя, когда наступил 2000-й год, было Тихоокеанское стандартное время (Pacific Standard Time, PST), по крайней мере для нас, жителей Западного берега. Скобки здесь нужны, чтобы сгруппировать параметры метода mktime. Чем больше параметров вы указываете, тем более точным становится ваше время.
Вы можете сравнивать время с помощью методов сравнения (более раннее время меньше, чем более позднее время), а если вы вычтете одно время из другого, вы получите разницу между ними в секундах. Поиграйте с этим немного!
Попробуйте ещё кое-что
• Один миллиард секунд… Выясните точно, в какую секунду вы родились (если сможете). Вычислите, когда вы достигнете (или, возможно, уже достигли?) возраста в один миллиард секунд. А потом отметьте это в вашем календаре.
• С днём рождения! Спросите, в каком году человек родился, затем в каком месяце, потом в какой день. Вычислите, сколько ему лет, и выдайте ему большой ХЛОП! на каждый его день рождения.
Класс Hash
Другой полезный класс – это класс Hash. Хэши во многом похожи на массивы: в них имеется набор слотов, которые могут указывать на различные объекты. Однако в массиве слоты выстроены в ряд, и каждый из них пронумерован (начиная с нуля). В хэше слоты не располагаются подряд (они просто как-то беспорядочно свалены вместе), и вы можете использовать для обращения к слоту любой объект, а не только число. Использовать хэши хорошо тогда, когда у вас есть набор каких-нибудь вещей, которые вы хотите обрабатывать, но они совсем не укладываются в нумерованный список. Например, цвета, которые я использую для различных частей кода, положенного в основу этого учебника:
colorArray =
[]
# то же, что Array.new
colorHash =
{ }
# то же, что Hash.new
colorArray[0]
= 'красный'
colorArray[1]
= 'зелёный'
colorHash['строки']
colorHash['числа']
colorHash['^Ki4eBbie слова'] = 'синий' colorArray.each do |color| puts color end
colorHash.each do |codeType, color| puts codeType + ': ' + color end
красный
зелёный
синий
строки: красный ключевые слова: синий числа: зелёный
Если я использую массив, мне нужно помнить, что слот 0 предназначен для строк, слот 1 – для чисел и т. д. Но если я использую хэш, то всё просто! Слот 'строки', конечно, содержит цвет строк. Ничего не нужно запоминать. Вы, должно быть, заметили, что когда мы применяли each, объекты из хэша выдавались не в том порядке, в котором мы их в него помещали. (По крайней мере, так было, когда я это писал. Возможно, сейчас будет по-другому… С этими хэшами никогда ничего не знаешь наперёд.) Для содержания чего-то в определённом порядке предназначены массивы, а не хэши.
Хотя для именования слотов в хэше обычно используются строки, вы могли бы использовать объект любого типа, даже массивы и другие хэши (хотя не могу представить себе, зачем бы вам захотелось это делать…):
weirdHash = Hash.new weirdHash[12] = 'обезьян' weirdHash[[]] = 'пустота'
weirdHash[Time.new] = 'текущее время и никакое другое'
Хэши и массивы хороши для разных применений; вам решать, что из них лучше всего подходит для конкретной задачи.
В конце предыдущей главы вы написали метод, выдающий английскую фразу для заданного целого числа. Однако, это не был метод целых чисел; это был просто метод «программы вообще». Но разве не было бы прекрасно, если вы могли бы написать что- то вроде 22. – Ьо_епд вместо englishNumber 22? Вот как вы бы это сделали:
Ну вот, я его протестировал; кажется, он работает. ;)
Вот так мы определили метод для целых чисел: «заскочили» в класс Integer, описали там метод и «выскочили» из него обратно. Теперь у всех целых чисел есть этот (хотя и немного недоделанный) метод. Фактически, если вам не нравится, как работает какой- нибудь встроенный метод, например to_s, вы могли бы просто переопределить его примерно таким же образом… но я не советую это делать! Лучше всего оставить старые методы в покое и создавать новые, когда вам хочется сделать что-нибудь новенькое.
Ну что… всё ещё непонятно? Давайте, я ещё раз пройдусь по последней программе. До сих пор, когда мы выполняли какой-нибудь код или определяли какие-то методы, мы делали это в объекте по умолчанию под названием «программа». В нашей последней программе мы впервые покинули этот объект и проникли в класс Integer. Мы определили в нём метод (поэтому он стал методом для целых чисел), и все целые числа могут его использовать. Внутри этого метода мы использовали self, чтобы ссылаться на объект (целое число), использующий этот метод.
Создание классов
Мы уже видели достаточно много объектов различных классов. Однако, легко предоставить себе такие объекты, которых в Ruby нет. К счастью, создать новый класс так же просто, как расширить существующий. Скажем, мы бы хотели сделать на Ruby игральные кости. Вот как мы могли бы создать класс Die:
class Die # игральная кость def roll
1 + rand(6) end
end
# Давайте создадим пару игральных костей… dice =[Die.new, Die.new]
# …и бросим их. dice.each do |die|
puts die.roll end
5
6
(Если вы пропустили раздел о случайных числах: rand(6) просто возвращает случайное число между 0 и 5.)
Вот так! Наши собственноручно созданные объекты. Бросьте кости несколько раз (нажимая на кнопку «Обновить») и понаблюдайте, что при этом появится.
Мы можем определить для наших объектов самые разные методы… но здесь чего-то явно не хватает. Работа с этими объектами сильно напоминает программирование до того, как мы узнали о переменных. Взгляните на наши кости, например. Мы можем бросать их, и каждый раз при этом они выдают нам другое число. Но если мы хотели бы задержаться на этом числе, нам бы пришлось создать переменную, указывающую на это число. Кажется, любая порядочная игральная кость должна иметь возможность хранить число, а бросание кости должно изменять это число. Если мы уже отслеживаем состояние самой кости, то нам уже не нужно отслеживать где-то ещё число, которое она показывает.
Однако, если мы попытаемся сохранить полученное число в (локальной) переменной метода roll, оно исчезнет, как только закончится roll. Нам нужно хранить число в переменной другого типа –
Переменные экземпляра
Обычно, когда мы хотим что-то сказать о строке, мы просто называем её строкой. Однако, мы могли также назвать её строковым объектом. Некоторые программисты могли бы назвать её экземпляром класса String, но это просто причудливый и длинный (так что можно запыхаться) способ сказать: «строка». Экземпляр класса – это просто объект этого класса.
Так что переменные экземпляра – это просто переменные объекта. Локальные переменные метода действуют до завершения метода. С другой стороны, переменные экземпляра каждого объекта будут действительны, пока существует объект. Чтобы отличить переменные экземпляра от локальных переменных, перед их именами ставится символ @:
class Die # игральная кость def roll
@numberShowing = 1 + rand(6) end
def showing
@numberShowing
end
end
die = Die.new die.roll
puts die.showing puts die.showing die.roll
puts die.showing puts die.showing
6
6
3
Очень хорошо! Так, метод roll бросает кость, а showing сообщает нам, какое число выпало. Однако, что же будет, если мы попытаемся посмотреть, что выпало прежде, чем мы бросили кость (прежде, чем мы задали значение @numberShowing)?
class Die # игральная кость def roll
@numberShowing = 1 + rand(6) end
def showing
@numberShowing
end
end
# Поскольку я не собираюсь снова использовать эту кость,
# мне не нужно сохранять её в переменной.
puts Die.new.showing
nil
Хммм… ладно, по крайней мере, она не выдала нам ошибку. Однако, в самом деле нет никакого смысла в том, что кость «не была брошена», или что бы там ни означало значение nil в этом случае. Было бы хорошо, если бы мы могли задать значение для нашего нового объекта «кость» сразу после того, как он был создан. Вот зачем нужен метод initialize:
class
Die # игральная кость
def
initialize
#
я просто
брошу эту кость, хотя мы
#
могли бы
сделать что-нибудь ещё, если бы хотели,
#
например,
задать, что выпало число 6.
roll
end
def roll
@numberShowing = 1 + rand(6) end
def showing
@numberShowing
end
end
puts Die.new.showing
2
Когда объект создаётся, всегда вызывается его метод initialize (если он у него определён).
Наши игральные кости теперь почти безупречны. Может быть, единственное, чего не хватает, так это способа задать, какой стороной выпала кость… Почему бы вам не написать метод cheat, который как раз это и делает! Вернётесь к чтению, когда закончите его (и, конечно, когда проверите, что он работает). Убедитесь, что невозможно задать, чтобы на кости выпало 7!
Итак, мы только что прошли весьма крутой материал. Однако же, он довольно сложный, поэтому позвольте мне дать вам другой, более интересный пример. Ну, скажем, мы хотим сделать простое виртуальное домашнее животное – дракончика. Как большинство детей, он должен быть способен есть, спать и «гулять», что означает, что нам нужно будет иметь возможность кормить его, укладывать спать и выгуливать. Внутри себя нашему дракону понадобится отслеживать, когда он голоден, устал или ему нужно на прогулку; но у нас не будет возможности узнать это, когда мы будем общаться с нашим драконом: точно так же вы не можете спросить человеческого младенца: «Ты хочешь есть?». Мы также предусмотрим несколько других забавных способов для общения с нашим дракончиком, а когда он родится, мы дадим ему имя. (Что бы вы ни передали в метод new, для вашего удобства будет передано в метод initialize.) Ладно, давайте попробуем:
class Dragon
def initialize name
@name = name
@asleep = false
@stuffInBelly = 10
# Он сыт.
@stuffInIntestine = 0
# Ему не надо гулять.
puts @name + ' родился.'
end
def passageOfTime # проходит некоторое время if @stuffInBelly > 0
# Переместить пищу из желудка в кишечник.
@stuffInBelly = @stuffInBelly – 1
@stuffInIntestine = @stuffInIntestine + 1 else # Наш дракон страдает от голода! if @asleep
@asleep = false
puts 'Он внезапно просыпается!' end
puts @name + ' проголодался! Доведённый до крайности, он съедает
ВАС ! '
exit # Этим методом выходим из программы. end
if @stuffInIntestine >= 10 @stuffInIntestine = 0
puts 'Опаньки! ' + @name + ' сделал нехорошо…' end
if hungry? if @asleep
@asleep = false
puts 'Он внезапно просыпается!' end
puts 'В желудке у ' + @name + '(а) урчит…' end
if poopy? if @asleep
@asleep = false
puts 'Он внезапно просыпается!'
ешЗ
puts @пате
+ ' подпрыгивает, потому что хочет на горшок…'
етЗ
етЗ
end
pet = Dragon.new
'Норберт'
pet.feed
pet. toss
pet.walk
pet.putToBed
pet.rock
pet.putToBed
pet.putToBed
pet.putToBed
pet.putToBed
Норберт родился.
Вы кормите Норберт(а).
Вы подбрасываете Норберт(а) в воздух.
Он хихикает, обжигая при этом вам брови.
Вы выгуливаете Норберт(а).
Вы укладываете Норберт(а) спать.
Норберт храпит, наполняя комнату дымом.
Норберт храпит, наполняя комнату дымом.
Норберт храпит, наполняя комнату дымом.
Норберт медленно просыпается.
Вы нежно укачиваете Норберт(а).
Он быстро задрёмывает…
…но просыпается, как только вы перестали качать. Вы укладываете Норберт(а) спать.
Он внезапно просыпается!
В желудке у Норберт(а) урчит…
Вы укладываете Норберт(а) спать.
Он внезапно просыпается!
В желудке у Норберт(а) урчит…
Вы укладываете Норберт(а) спать.
Он внезапно просыпается!
В желудке у Норберт(а) урчит…
Норберт подпрыгивает, потому что хочет на горшок…
Вы укладываете Норберт(а) спать.
Он внезапно просыпается!
Норберт проголодался! Доведённый до крайности, он съедает ВАС!
Вот тебе и раз! Конечно, было бы лучше, если бы это была интерактивная программа, но эти изменения вы можете сделать попозже. Я просто попытался показать те части программы, которые непосредственно относятся к созданию нового класса Dragon.
В этом примере мы увидели несколько новых конструкций. Первая достаточно проста: exit заканчивает программу «здесь и сейчас». Вторая – это ключевое слово private, которое мы вставили прямо в середину описания нашего класса. Я мог бы обойтись без него, но я хотел подчеркнуть мысль о том, что одни методы – это то, что вы можете делать с драконом, а другие – то, что просто происходит внутри дракона. Вы можете считать, что эти методы скрыты «под капотом»: если вы не работаете автомехаником, всё, что на самом деле вам нужно знать, это педаль газа, педаль тормоза и рулевое колесо. Программист назвал бы их открытым интерфейсом вашей машины. Однако то, каким образом ваша аварийная подушка знает, когда наполниться воздухом, является внутренним поведением машины; обычному пользователю (водителю) не нужно знать об этом.
А сейчас в качестве более конкретного примера, иллюстрирующего эти строки, давайте поговорим о том, как вы могли бы представить автомобиль в видео-игре (чем я как раз, по случайному совпадению, и занимаюсь). Во-первых, вы захотели бы решить, каким должен выглядеть ваш внешний интерфейс; другими словами, какие методы смогут вызывать пользователи у ваших объектов-автомобилей? Ну, им понадобится нажимать на педаль газа и педаль тормоза, но им также понадобится указывать, с какой силой они нажимают на педаль. (Есть большая разница между «вдавить в пол» и «дотронуться».) Им также понадобится управлять, и снова потребуется возможность сказать, насколько сильно они поворачивают руль. Я полагаю, вы могли бы продолжить и добавить сцепление, сигналы поворота, реактивную установку, форсаж, конденсатор временного потока [главная деталь машины времени – Прим. перев.] и так далее… Это зависит от того, какую разновидность игры вы делаете.
Однако, нужно, чтобы внутри объекта-автомобиля происходило много чего другого; автомобилю нужны будут такие вещи, как скорость, направление и положение (и это только самые основные). Эти атрибуты могут изменяться нажатием на педали газа и тормоза и, конечно, поворачиванием руля, но пользователь не должен иметь возможности непосредственно устанавливать положение (что было бы подобно сверхсветовому перемещению). Вы, должно быть, также пожелаете отслеживать боковые заносы и повреждения, отрыв всех колёс от земли и так далее. Всё это будет внутренностями вашего автомобильного объекта.
Попробуйте ещё кое-что
• Сделайте класс для апельсинового дерева – OrangeTree. У него должен быть метод height, возвращающий его высоту, и метод oneYearPasses, который при вызове увеличивает возраст дерева на один год. Каждый год дерево становится выше (как по-вашему, на какую высоту в год должно вырастать апельсиновое дерево?), а после определённого числа лет (опять же, как вы считаете) дерево должно умереть. Первые несколько лет оно не должно плодоносить, но через некоторое время должно, и мне кажется, что более старые деревья приносят каждый год больше плодов, чем молодые… настолько, насколько вы считаете это разумным. И, конечно, вам нужно иметь возможность сосчитать апельсины методом countTheOranges (который возвращает число апельсинов на дереве) и собирать их методом pickAnOrange (который уменьшает @orangeCount на единицу и возвращает строку с описанием, насколько вкусен был апельсин, или же он просто сообщает вам, что больше нет апельсинов для сбора в этом году). Удостоверьтесь, что те апельсины, что вы не собрали в этом году, опадут до следующего года.
• Напишите программу, в которой вы смогли бы взаимодействовать с вашим дракончиком. У вас должна быть возможность вводить такие команды, как feed и walk, и чтобы при этом вызывались нужные методы вашего дракона. Конечно, поскольку то, что вы вводите, это просто строки, вам понадобится некое подобие диспетчера методов, где ваша программа проверяет, какая строка была введена и затем вызывает соответствующий метод.
Вот почти что и всё, что можно сказать об этом! Нет, подождите секундочку… Я же не рассказал вам обо всех этих классах, которые выполняют самые разнообразные вещи: отправляют электронную почту, сохраняют и загружают файлы на ваш компьютер, или же создают окна и кнопки (и даже 3-хмерные миры) и всё прочее! Что ж, попросту имеется настолько много классов, которые вы можете использовать, что мне никак невозможно показать вам их все; я даже не знаю, что большинство из них из себя представляют! А что я могу сказать вам о них, так это то, где можно разузнать о них поподробнее, чтобы вы смогли изучить те из них, которые вы захотите применить в ваших программах. Однако, прежде, чем вы отправитесь на самостоятельное изучение, вам ещё следует узнать о других важных особенностях Ruby, которых нет в большинстве других языков, но без которых я просто не смог бы жить: о блоках и процедурных объектах.