10
6

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 5 years have passed since last update.

組合せ最適化で本を買う

Last updated at Posted at 2017-11-21

これなに

某社では、技術書やビジネス書を買うと一定額まで補助してもらえるそうです。(本は自分で所有できます)
買いたい本が複数あって、N回 補助の機会があるとき、どの本を買ったら良いでしょうか?
組合せ最適化で計算してみましょう。

考え方

本を買ったら価値は9割になるとして、目的関数を「0.9×購入金額―自腹額」とします。
また、各機会ごとに合計購入金額が補助額を超えた分が自腹額となります。
この目的関数を最大化してみましょう。

下記は、N=1のときの目的関数のイメージです。傾きは、左側が0.9で、右側が -0.1(=0.9-1)です。

image.png

1冊の本は分割できないので、購入額候補(本の組合せ)は、離散的な点になります。

買いたい本のリスト

リストを表示します。

python
import numpy as np, pandas as pd
from pulp import *
from ortoolpy import addvars, addbinvars
a = pd.DataFrame([
    ('吾輩は猫だろう', 981),
    ('風の三四郎', 726),
    ('夏との扉', 1024),
    ('高い城の人', 1335),
    ('天王星年代記', 865),
    ('並列都市の科学', 1171),
    ('星は無慈悲な夜の王', 914),
    ('火の島かご', 463),
    ('ロボットは我', 758),
    ('未来世紀カガワ', 1507),
    ('ケイコとアベ', 689),
    ('48億円の身代金', 1412),
    ('星々の王女様', 826),
    ('猫とゆりかもめ', 649),
    ('風と共にとまる', 1083),
    ], columns=['Title', 'Price'])
a
Title Price
0 吾輩は猫だろう 981
1 風の三四郎 726
2 夏との扉 1024
3 高い城の人 1335
4 天王星年代記 865
5 並列都市の科学 1171
6 星は無慈悲な夜の王 914
7 火の島かご 463
8 ロボットは我 758
9 未来世紀カガワ 1507
10 ケイコとアベ 689
11 48億円の身代金 1412
12 星々の王女様 826
13 猫とゆりかもめ 649
14 風と共にとまる 1083

Pythonで計算する

回数を4回、1回当たりの補助額を3000円とします。

python
N = 4 # 回数
S = 3000 # 1回当たりの補助額
m = LpProblem(sense=LpMaximize)
for i in range(N):
    a[f'Var{i}'] = addbinvars(len(a))
sums = addvars(N) # 購入額
owns = addvars(N) # 自腹額
m += 0.9*lpDot(a.Price,sum(a[f'Var{i}'] for i in range(N))) - lpSum(owns)
for i in range(N):
    m += sums[i] == lpDot(a.Price,a[f'Var{i}'])
    m += owns[i] >= sums[i] - S
for _,r in a.iterrows():
    m += lpSum(r[f'Var{i}'] for i in range(N)) <= 1 # 同じ本は1冊まで
%time m.solve()
print(LpStatus[m.status])
for i in range(N):
    a[f'Val{i}'] = a[f'Var{i}'].apply(value)
    print(a[a[f'Val{i}']>0.5].iloc[:,:2])
    print(f'{i+1} 合計 {value(sums[i])}')
>>>
Wall time: 49.8 s
Optimal
      Title  Price
9   未来世紀カガワ   1507
10   ケイコとアベ    689
12   星々の王女様    826
1 合計 3022.0
       Title  Price
1      風の三四郎    726
4     天王星年代記    865
11  48億円の身代金   1412
2 合計 3003.0
      Title  Price
5   並列都市の科学   1171
8    ロボットは我    758
14  風と共にとまる   1083
3 合計 3012.0
        Title  Price
0     吾輩は猫だろう    981
6   星は無慈悲な夜の王    914
7       火の島かご    463
13    猫とゆりかもめ    649
4 合計 3007.0

結構、時間がかかりますね。モデル化の工夫の余地があるかもしれません。
1回だけなら、「天王星年代記, 星は無慈悲な夜の王, 火の島かご, ロボットは我」を買うとちょうど3000円ですが、4回買うとすると、ちょうどの組合せはなくなりました。

以上

10
6
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
10
6

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?