Sobes.tech
Назад к вопросам
Senior
2

# Банкомат, который заряжается кассетами с купюрами, с нашим приложением на борту # должен уметь выдавать купюры для заданной суммы или отвечать отказом # При выдаче купюры списываются с баланса банкомата. # Допустимые номиналы: 50Р, 100Р, 500Р, 1000Р, 5000Р. class ATM: pass # место для кода # Устройство банкомата: # - деньги расположены в кассетах внутри банкомата, которые загружает инкассатор и перезагружает банкомат; # - в каждой кассете лежат купюры своего номинала; # - банкомат может подсчитать оставшиеся в кассетах банкноты, но эта операция занимает продолжительное время - её стоит вызывать как можно реже. # API для взаимодействия с аппаратурой банкомата. # интерфейс SDK может быть изменён/расширен по договорённости сторон, если это необходимо class SDK(ABC): @abstractmethod def count_banknotes(self, banknote: int) -> int: pass @abstractmethod def move_banknote_to_dispenser(self, banknote: int, count: int) -> None: pass @abstractmethod def open_dispenser(self) -> None: pass

Компании, где спрашивали
Яндекс

Ответ от нейросети

sobes.tech AI

python from abc import ABC, abstractmethod

class ATM: def init(self, sdk: SDK): self.sdk = sdk self.banknotes_cache = {}

def _update_cache(self):
    # Обновляем кэш количества банкнот для каждого номинала
    for banknote in [50, 100, 500, 1000, 5000]:
        self.banknotes_cache[banknote] = self.sdk.count_banknotes(banknote)

def _can_dispense(self, amount: int) -> bool:
    # Проверяем, можно ли выдать сумму с текущим кэшем
    # Используем жадный алгоритм по убыванию номиналов
    remaining = amount
    temp_cache = self.banknotes_cache.copy()
    for banknote in sorted(temp_cache.keys(), reverse=True):
        needed = remaining // banknote
        available = temp_cache[banknote]
        used = min(needed, available)
        remaining -= used * banknote
        temp_cache[banknote] -= used
        if remaining == 0:
            return True
    return False

def dispense(self, amount: int) -> bool:
    # Обновляем кэш перед выдачей
    self._update_cache()
    if amount <= 0 or amount % 50 != 0:
        return False  # Неверная сумма
    if not self._can_dispense(amount):
        return False  # Невозможно выдать сумму

    remaining = amount
    for banknote in sorted(self.banknotes_cache.keys(), reverse=True):
        needed = remaining // banknote
        available = self.banknotes_cache[banknote]
        used = min(needed, available)
        if used > 0:
            self.sdk.move_banknote_to_dispenser(banknote, used)
            self.banknotes_cache[banknote] -= used
            remaining -= used * banknote
        if remaining == 0:
            break

    self.sdk.open_dispenser()
    return True

**Объяснение:**
- Класс `ATM` принимает объект `SDK` для взаимодействия с аппаратной частью.
- Для оптимизации вызовов `count_banknotes` используется кэш, который обновляется перед выдачей.
- Метод `dispense` проверяет возможность выдать сумму, используя жадный алгоритм по номиналам.
- Если выдача возможна, банкноты перемещаются в диспенсер, после чего он открывается.
- Если сумма не кратна 50 или не может быть выдана, возвращается отказ.