Иногда подобные прямые обращения к атрибутам допустимы, но чаще разработчик пишет вспомогательный метод, который изменяет значение за него.
Изменение значения атрибута с использованием метода
В класс можно включить методы, которые изменяют некоторые атрибуты за вас. Вместо того чтобы изменять атрибут напрямую, вы передаете новое значение методу, который берет обновление атрибута на себя.
В следующем примере в класс включается метод update_odometer() для изменения показаний одометра:
class Car():
. ....
. . . .
(1) . .def update_odometer(self, mileage):
. . . ."""Устанавливает заданное значение на одометре."""
. . . .self.odometer_reading = mileage
. .
my_new_car = Car('audi', 'a4', 2016)
print(my_new_car.get_descriptive_name())
(2)my_new_car.update_odometer(23)
my_new_car.read_odometer()
Класс Car почти не изменился, в нем только добавился метод update_odometer()(1) . Этот метод получает пробег в милях и сохраняет его в self.odometer_reading. В точке (2) мы вызываем метод update_odometer() и передаем ему значение 23 в аргументе (соответствующем параметру mileage в определении метода). Метод устанавливает на одометре значение 23, а метод read_odometer() выводит текущие показания:
2016 Audi A4
This car has 23 miles on it.
Метод update_odometer() можно расширить так, чтобы при каждом изменении показаний одометра выполнялась некоторая дополнительная работа. Добавим проверку, которая гарантирует, что никто не будет пытаться сбрасывать показания одометра:
class Car():
. ....
. .def update_odometer(self, mileage):
. . . ."""
. . . .Устанавливает на одометре заданное значение.
. . . .При попытке обратной подкрутки изменение отклоняется.
. . . ."""
(1) . . . .if mileage >= self.odometer_reading:
. . . . . .self.odometer_reading = mileage
. . . .else:
(2) . . . . . .print("You can't roll back an odometer!")
Теперь update_odometer() проверяет новое значение перед изменением атрибута. Если новое значение mileage больше или равно текущего, self.odometer_reading, показания одометра можно обновить новым значением (1) . Если же новое значение меньше текущего, вы получите предупреждение о недопустимости обратной подкрутки (2).
Изменение значения атрибута с приращением
Иногда значение атрибута требуется изменить с заданным приращением (вместо того чтобы присваивать атрибуту произвольное новое значение). Допустим, вы купили подержанную машину и проехали на ней 100 миль. Следующий метод получает величину приращения и прибавляет ее к текущим показаниям одометра:
class Car():
...
def update_odometer(self, mileage):
--snip--
. .
(1) . .def increment_odometer(self, miles):
. . . ."""Увеличивает показания одометра с заданным приращением."""
. . . .self.odometer_reading += miles
. .
(2)my_used_car = Car('subaru', 'outback', 2013)
print(my_used_car.get_descriptive_name())
(3)my_used_car.update_odometer(23500)
my_used_car.read_odometer()
(4)my_used_car.increment_odometer(100)
my_used_car.read_odometer()
Новый метод increment_odometer() в точке (1) получает расстояние в милях и прибавляет его к self.odometer_reading. В точке (2) создается экземпляр my_used_car. Мы инициализируем показания его одометра значением 23 500; для этого вызывается метод update_odometer(), которому передается значение 23500 (3). В точке (4) вызывается метод increment_odometer(), которому передается значение 100, чтобы увеличить показания одометра на 100 миль, пройденные с момента покупки:
2013 Subaru Outback
This car has 23500 miles on it.
This car has 23600 miles on it.
При желании можно легко усовершенствовать этот метод, чтобы он отклонял отрицательные приращения; тем самым вы предотвратите обратную подкрутку одометра.
примечание
Подобные методы управляют обновлением внутренних значений экземпляров (таких, как показания одометра), однако любой пользователь, имеющий доступ к программному коду, сможет напрямую задать атрибуту любое значение. Эффективная схема безопасности должна уделять особое внимание таким подробностям, не ограничиваясь простейшими проверками.
Упражнения
9-4. Посетители: начните с программы из упражнения 9-1 (с. 165). Добавьте атрибут number_served со значением по умолчанию 0; он представляет количество обслуженных посетителей. Создайте экземпляр с именем restaurant. Выведите значение number_served, потом измените и выведите снова.
Добавьте метод с именем set_number_served(), позволяющий задать количество обслуженных посетителей. Вызовите метод с новым числом, снова выведите значение.
Добавьте метод с именем increment_number_served(), который увеличивает количество обслуженных посетителей на заданную величину. Вызовите этот метод с любым числом, которое могло бы представлять количество обслуженных клиентов — скажем, за один день.
9-5. Попытки входа: добавьте атрибут login_attempts в класс User из упражнения 9-3 (с. 165). Напишите метод increment_login_attempts(), увеличивающий значение login_attempts на 1. Напишите другой метод с именем reset_login_attempts(), обнуляющий значение login_attempts.