!!! サイト改修中のため表示が乱れる場合があります(1月末頃まで) !!!
量子アニーリング

21-25. 量子アニーリング(QUBO)で複数の数字を均等に2組に分ける

やること

もうしばらくQUBOは見たくないのですが、今回は入試の数学でも出題されるような「複数の数字を均等に2グループに分けてください」を解いてみましょう。

【問題】
次の6つの自然数を、総和が等しくなるように2グループに分けなさい。
15, 25, 33, 41, 64, 82

おさらい

カクラスを解いた記事を予習してください。

例えば、重さ1, 2, 3, 4のボールのどれを取れば合計7の重さになりますか?の条件式を次のように設定しました。

H = (1*q0 + 2*q1 + 3*q2 + 4*q3 - 7)**2

解は [q0, q1, q2, q3] = [0, 0, 1, 1] or [1, 1, 0, 1] となり、3, 4のボール、または1, 2, 4のボールを取ればいいことが分かります。

こういった方程式的な設定はこちらの記事でも扱いました↓

その他のQUBOで設定できる条件式については過去のまとめ記事をご参照ください↓

どうやって数字分け問題を解くか?

まず、すべての合計値が260なので各グループは合計130です。

数字に6個の量子ビット(q0~q5)を割り当て、1になったらグループA、0になったらグループBとします。

グループAの量子ビットは1であることを利用し、重みとして数字をかけた合計が130になるような式を設定します。「量子ビットが1になったときにその数字の重みが有効」になります。

#グループA(=1になった量子ビット)は合計130
H = (15*q0 + 25*q1 + 33*q2 + 41*q3 + 64*q4 + 82*q5 - 130)**2

反対に、グループBの量子ビットは0であることから、こちらも重みをかけたものの合計が130になるように設定します。ここで大切なのは、q0 を (1 – q0) と置き換えることで0,1の条件をひっくり返し「0になったときに重みが有効」を表現できることです。

#グループB(=0になった量子ビット)は合計130
H = (15*(1-q0) + 25*(1-q1) + 33*(1-q2) + 41*(1-q3) + 64*(1-q4) + 82*(1-q5) - 130)**2

コードと結果

これだけです。

from pyqubo import Binary
from dwave_qbsolv import QBSolv

#量子ビットを用意する
q0 = Binary('q0')
q1 = Binary('q1')
q2 = Binary('q2')
q3 = Binary('q3')
q4 = Binary('q4')
q5 = Binary('q5')

#グループA(=1になった量子ビット)は合計130
H = 0
H += (15*q0 + 25*q1 + 33*q2 + 41*q3 + 64*q4 + 82*q5 - 130)**2

#グループB(=0になった量子ビット)は合計130
H += (15*(1-q0) + 25*(1-q1) + 33*(1-q2) + 41*(1-q3) + 64*(1-q4) + 82*(1-q5) - 130)**2


#コンパイル
model = H.compile()
qubo, offset = model.to_qubo()

#分割サンプリング
response = QBSolv().sample_qubo(qubo, num_repeats=500)

#レスポンスの確認
print('response')
for (sample, energy, occurrence) in response.data():
    print(f'Sample:{list(sample.values())} Energy:{energy} Occurrence:{occurrence}')
response
Sample:[1, 0, 1, 0, 0, 1] Energy:-33800.0 Occurrence:146
Sample:[0, 1, 0, 1, 1, 0] Energy:-33800.0 Occurrence:355

解は2通り出ましたがグループ名を入れ替えたものなので同じです。[15, 25, 33, 41, 64, 82] => [1, 0, 1, 0, 0, 1] ということで、グループAに [15, 33, 82]、グループBに [25, 41, 64] と分かりました。

さて、お気づきかもしれませんが実はグループBの設定式は無くても大丈夫です。グループAの条件を満たしていれば必然的にグループBの条件も満たすからです。

グループBの条件式をなくして実行しても

response
Sample:[1, 0, 1, 0, 0, 1] Energy:-16900.0 Occurrence:144
Sample:[0, 1, 0, 1, 1, 0] Energy:-16900.0 Occurrence:357

同じ結果が得られました。

さいごに

あまり数字が多いと解空間の離散性が高すぎてサンプリングのヒット率が絶望的になる気がしますが、いやいや、離散性が高いのどうのと考えている時点で疑似アニーリング脳なのでしょう。理想の量子アニーリングマシンなら解空間がどうであれ1回のサンプリングできっちり求まるはずです。未来のアニーリングマシンに期待していますよ!

リアクションのお願い

「参考になった!」「刺激された!」と思ったらぜひリアクションをしましょう。エンジニアの世界はGive and Takeによって成り立っています。これからも無料で良質な情報にアクセスできるよう、Giveする人への感謝をリアクションで示しましょう!

この記事をシェアする

自身のブログ等で使用する場合は引用を忘れずに!

また、寄付も受け付けています。コーヒー1杯でとても喜びます(*˘︶˘*)

 Amazonでギフト券(アマギフ)を贈る

こちらのリンク から金額を指定してお贈りください。(デフォルトで10000円になっているのでご変更ください)

配送:Eメール
受取人:staffあっとvigne-cla.com
贈り主:あなたのお名前やニックネーム
メッセージ:◯◯の記事が参考になりました。など

のようにご入力ください。見返りはありませんのでご了承ください。

 Amazonで食事券(すかいらーく優待券)を贈る

500円 1000円 2000円 5000円 からお贈りください。

配送:Eメール
受取人:staffあっとvigne-cla.com
贈り主:あなたのお名前やニックネーム
メッセージ:◯◯の記事が参考になりました。など

のようにご入力ください。見返りはありませんのでご了承ください。

 その他、ギフト券やクーポン券をメールで贈る

デジタルのギフト券/クーポン券はメールアドレス(staffあっとvigne-cla.com)までお送りください。受領の返信をいたします。
紙のギフト券/クーポン券は 「郵便物はこちらへ」の住所 まで送付してください。名刺やメールアドレスを同封していただければ受領の連絡をいたします。
余った株主優待券等の処理におすすめです。
いずれも見返りはありませんのでご了承ください。

不明点はSNSでお気軽にご連絡ください

ビネクラのTwitter・Youtubeでコメントをください!


Slack・Discordの場合はこちらの公開グループに参加してShoya YasudaまでDMをください!


※当ブログに関することは何でもご相談・ご依頼可能です。

この記事を書いた人
Yasuda

博士(理学)。専門は免疫細胞、数理モデル、シミュレーション。米国、中国で研究に携わった。遺伝的アルゴリズム信者。物価上昇のため半額弁当とともに絶滅寸前。

タイトルとURLをコピーしました