## 問題
商品名と支払い金額を入力すると、お釣りの「小銭の枚数」を返す関数またはメソッドを実装せよ。商品名と価格は以下の通り。
* お茶: 120円
* コーラ: 150円
* エナジードリンク: 280円
入力される通貨は日本円で、最大1000円までとする。
また、有効な貨幣は「1000円」「500円」「100円」「50円」「10円」とする。
たとえば、「1000円でお茶を購入」すると、返却すべき金額は880円だが、釣銭は「500円×1, 100円×3, 50円×1, 10円×3」となる。返り値は、任意の形式のクラス、連想配列、オブジェクト、構造体、リストなどで表現してよい。
PHPの擬似コードで書くと以下のようになる。
```php
function calcChange(string $item, int $paid_money): array {
// ...
}
$expected = [500 => 1, 100 => 3, 50 => 1, 10 => 3];
assert($expected === calcChange("お茶", 1000));
```
---
## 回答
```python
item_list: dict = { 'お茶': 120, 'コーラ': 150, 'エナジードリンク': 280 }
money_list: list = [ 1000, 500, 100, 50, 10] ## descending order
def calc_change(item_name: str, money: int) -> dict:
## item_name validation section (必要に応じて関数化して切り出す)
if item_name not in item_list.keys():
print("item_name is not found", file=sys.stderr)
exit(1)
## end: item_name validation section
## money validation section (必要に応じて関数化して切り出す)
if money % 10 != 0:
# if money % money_list[-1] != 0:
# ※1: もし24円硬貨等が投入された場合には, 例えば最大公約数を用いて、取り得る最小の貨幣の単位を算出する必要がある
print("not valid money", file=sys.stderr)
exit(1)
## end: money validation section
otsuri: int = money - item_list[item_name] #220
# otrsuri validation section
if otsuri < 0:
print("moneyが足りません", file=sys.stderr)
exit(1)
# end: otrsuri validation section
result: dict = {}
## ※1が成立すればここのロジックも怪しくなる
## money validation をここで行っても良いのでは? (どっちでも良い)
for value in money_list:
count = otsuri // value
# if count > 0:
dict[value] = count
otsuri %= value
assert(otrsuri == 0) # 必要?
return result
{
100: 2, 10: 2
}[500]
```