Нам часто необходимо предоставить переменную экземпляра как с помощью геттера, так и сеттера. Для этого у Crystal есть макрос
class Person
property name
end
Это то же самое, что написать следующее:
class Person
def name
@name
end
def name=(new_name)
@name = new_name
end
end
Как обычно, объявления типов или начальные значения могут использоваться для очень удобного синтаксиса. Существуют и другие полезные макросы, например макрос
Наследование
Классы могут основываться на других классах, обеспечивая более специализированное поведение. Когда класс наследует от другого, он получает все существующие методы и переменные экземпляра и может добавлять новые или перезаписывать существующие. Например, давайте расширим ранее определенный класс Person
:
class Person
property name : String
def initialize(@name)
end
end
class Employee < Person
property salary = 0
end
Экземпляр Employee
может находиться в любом месте, где требуется экземпляр Person
, поскольку, по сути, employee
– это человек:
person = Person.new("Alan")
employee = Employee.new("Helen")
employee.salary = 10000
p person.is_a? Person # => true
p employee.is_a? Person # => true
p person.is_a? Employee # => false
В этом примере родительским классом является Person
, а дочерним - Employee
. Для создания иерархии классов можно создать несколько классов. При наследовании от существующего класса дочерний класс может не только расширять, но и переопределять части своего родительского класса. Давайте посмотрим на это на практике:
class Employee
def yearly_salary
12 * @salary
end
end
class SalesEmployee < Employee
property bonus = 0
def yearly_salary
12 * @salary + @bonus
end
end
В этом примере мы видим, что ранее определенный класс Employee
повторно открывается для добавления нового метода. При повторном открытии класса не следует указывать его родительский класс (в данном случае Person
). Метод yearly_salary
добавляется к Employee
, а затем создается новый специализированный тип Employee
, наследуемый от него (и, в свою очередь, также наследуемый от Person
). Добавляется новое свойство и переопределяется yearly_ salary
, чтобы учесть его. Переопределение затрагивает только объекты типа SalesEmployee
, но не объекты типа Employee
.
При наследовании от класса и переопределении метода ключевое слово super
может использоваться для вызова переопределенного определения из родительского класса. yearly_salary
можно было бы записать следующим образом:
def yearly_salary
super + @bonus
end
Поскольку метод initialize
используется для подготовки начального состояния объекта, ожидается, что он всегда будет выполнен раньше всего остального. Таким образом, общепринятой практикой является использование ключевого слова super
для вызова конструктора родительского класса при наследовании от существующего класса.
Теперь, когда мы определили несколько классов и подклассов, мы можем воспользоваться еще одной мощной концепцией: объекты типа подкласса могут храниться в переменной, типизированной для хранения одного из его базовых классов.
Полиморфизм
SalesEmployee
наследуется от Employee
, чтобы определить более специализированный тип сотрудника, но это не меняет того факта, что сотрудник отдела продаж является сотрудником и может рассматриваться как таковой. Это называется полиморфизмом. Давайте посмотрим пример этого в действии:
employee1 = Employee.new("Helen")
employee1.salary = 5000
employee2 = SalesEmployee.new("Susan")
employee2.salary = 4000
employee2.bonus = 20000
employee3 = Employee.new("Eric")
employee3.salary = 4000
employee_list = [employee1, employee2, employee3]