py_stepik/mod_oop/2.3_13_tvprogram.py

149 lines
6.9 KiB
Python
Raw Normal View History

2024-04-09 14:32:49 +00:00
"""
https://stepik.org/lesson/701985/step/13?unit=702086
Необходимо написать программу для представления и управления расписанием телевизионного вещания. Для этого нужно объявить класс TVProgram, объекты которого создаются командой:
>>> repr(TVProgram("название канала"))
"TVProgram('название канала')"
где название канала - это строка с названием телеканала.
В каждом объекте класса TVProgram должен формироваться локальный атрибут:
items - список из телепередач (изначально список пуст).
>>> TVProgram("название канала").items
[]
В самом классе TVProgram должны быть реализованы следующие методы:
add_telecast(self, tl) - добавление новой телепередачи в список items;
remove_telecast(self, indx) - удаление телепередачи по ее порядковому номеру (атрибуту __id, см. далее).
>>> {hasattr(TVProgram, m) for m in "add_telecast remove_telecast".split()}
{True}
Каждая телепередача должна описываться классом Telecast, объекты которого создаются командой:
>>> repr(Telecast(1, "название", 1111))
"Telecast(1, 'название', 1111)"
где 1 - порядковый номер - номер телепередачи в сетке вещания (от 1 и далее); "название" - наименование телепередачи; 1111 - длительность - время телепередачи (в секундах - целое число).
Соответственно, в каждом объекте класса Telecast должны формироваться локальные приватные атрибуты:
__id - порядковый номер (целое число);
__name - наименование телепередачи (строка);
__duration - длительность телепередачи в секундах (целое число).
>>> {hasattr(obj, a) for obj in (Telecast(1, "название", 1111),) for a in "__id __name __duration".split()}
{True}
Для работы с этими приватными атрибутами в классе Telecast должны быть объявлены соответствующие объекты-свойства (property):
uid - для записи и считывания из локального атрибута __id;
name - для записи и считывания из локального атрибута __name;
duration - для записи и считывания из локального атрибута __duration.
>>> {hasattr(obj, a) for obj in (Telecast(1, "название", 1111),) for a in "uid name duration".split()}
{True}
Пример использования классов (эти строчки в программе писать не нужно):
>>> def test():
... pr = TVProgram("Первый канал")
... pr.add_telecast(Telecast(1, "Доброе утро", 10000))
... pr.add_telecast(Telecast(2, "Новости", 2000))
... pr.add_telecast(Telecast(3, "Интервью с Балакиревым", 20))
... return [*map(lambda t: f"{t.name}: {t.duration}", pr.items)]
>>> test()
['Доброе утро: 10000', 'Новости: 2000', 'Интервью с Балакиревым: 20']
"""
from typing import List
def make_properties(names: List[str]):
def decorator(cls):
def prop(private_name: str):
def getter(self):
return getattr(self, private_name)
def setter(self, value):
return setattr(self, private_name, value)
return getter, setter
for name in names:
private_name = "__id" if name == "uid" else f"__{name}"
setattr(cls, name, property(*prop(private_name)))
return cls
return decorator
@make_properties("uid name duration".split())
class Telecast:
def __init__(self, uid: int, name: str, duration: int):
self.uid, self.name, self.duration = uid, name, duration
def __repr__(self):
return f"{self.__class__.__name__}{(self.uid, self.name, self.duration)!r}"
class TVProgram:
def __init__(self, name: str):
self.name = name
self.items: List[Telecast] = []
def __repr__(self) -> str:
return f"{self.__class__.__name__}('{self.name}')"
def add_telecast(self, telecast: Telecast):
self.items.append(telecast)
def remove_telecast(self, indx: int):
self.items = [item for item in self.items if item.uid != indx]
def tests():
assert hasattr(TVProgram, "add_telecast") and hasattr(
TVProgram, "remove_telecast"
), "в классе TVProgram должны быть методы add_telecast и remove_telecast"
pr = TVProgram("Первый канал")
pr.add_telecast(Telecast(1, "Доброе утро", 10000))
pr.add_telecast(Telecast(3, "Новости", 2000))
t = Telecast(2, "Интервью с Балакиревым", 20)
pr.add_telecast(t)
pr.remove_telecast(3)
assert (
len(pr.items) == 2
), "неверное число телеперач, возможно, некорректно работает метод remove_telecast"
assert (
pr.items[-1] == t
), "удалена неверная телепередача (возможно, вы удаляете не по __id, а по порядковому индексу в списке items)"
assert (
type(Telecast.uid) == property
and type(Telecast.name) == property
and type(Telecast.duration) == property
), "в классе Telecast должны быть объекты-свойства uid, name и duration"
for x in pr.items:
assert hasattr(x, "uid") and hasattr(x, "name") and hasattr(x, "duration")
assert (
pr.items[0].name == "Доброе утро"
), "объект-свойство name вернуло неверное значение"
assert (
pr.items[0].duration == 10000
), "объект-свойство duration вернуло неверное значение"
t = Telecast(1, "Доброе утро", 10000)
t.uid = 2
t.name = "hello"
t.duration = 10
if __name__ == "__main__":
import doctest
doctest.testmod()
tests()