動的計画法 表の作成 制約の追加 練習問題
動的計画法とは • (組合せ最適化などの)問題を解く解法のひとつ • 問題が直線的な構造を持つとき、その構造にそって反復的に解く • 特徴は、簡単な問題をたくさん解き、その解の表から、ひとつずつ問題を解く、というもの。最終的に、解きたい大きさの問題が解ける
サブセットサム問題 • n 個の数(a1,…,an)がある。これらの数の組合せで、合計がちょうど b になるようなものがあるか? 例題) 15, 24, 65, 32, 4, 55, 54, 23 の組合せで合計が 145 となるものはあるか?
サブセットサム問題 (2) • n 番目の数だけ取り除いた問題を考える • 元の問題に合計 b の組合せがある 取り除いた問題で合計 b か b - an の組合せがある 例題) 15, 24, 65, 32, 4, 55, 54, 23 の組合せで合計が 145 となるものはあるか? 15, 24, 65, 32, 4, 55, 54 の組合せで合計が 145 か 122 となるものはあるか?
分枝限定法で解く 最後の 23 を取り除く 問題0: 15, 24, 65, 32, 4, 55, 54 合計 145 はある? • 要素 an から順番に、使う/使わないを決め、問題を分割し、再帰的に解く 最後の 23 を取り除く 問題0: 15, 24, 65, 32, 4, 55, 54 合計 145 はある? 問題1: 15, 24, 65, 32, 4, 55, 54 合計 122 はある? 同様に 問題00: 15, 24, 65, 32, 4, 55 合計 145 はある? 問題01: 15, 24, 65, 32, 4, 55 合計 91 はある? 問題10: 15, 24, 65, 32, 4, 55 合計 122 はある? 問題11: 15, 24, 65, 32, 4, 55 合計 68 はある?
分枝限定法で解く (2) • 問題11111: 15, 24, 65 合計 -23 となるものはあるか? • 問題11111: 15, 24, 65 合計 -23 となるものはあるか? • 問題000: 15, 24, 65, 32, 4 合計 145 はとなるものはあるか? • こういった明らかに実行不能な問題は、すぐに答えが分かる 最悪で O(2n) 時間かかる
すべてのb についての答えの表を作れればよい 動的計画法 アイディア: • 0 から b までの全ての数について、合計がその数になる組合せがあるかどうかを調べ、表を作る • anを取り除いた問題で、組合せてできる数を全て調べる 元の問題で組合せてできる数は、 これらに anを加えた/加えないもの 1つ数を取り除き、小さくした問題の、 すべてのb についての答えの表を作れればよい
アルゴリズム • f( i, m ) : a1 ,…, ai の組合せで、和が m となるものがあれば1、そうでなければ 0 • i = 1,…,n に対して、以下の操作を実行する。 全ての 0 ≦m≦ b に対して、 f( i, m ) を計算する • f(i,m) = 1 f(i-1,m) =1 または f(i-1,m-ai) =1 • f(i,m) = 0 f(i-1,m) =0 かつ f(i-1,m-ai) =0 • f(1,m) は簡単に計算できる 計算時間は O(nb)
動的計画法コード • a[0],…,a[n-1] に数値を格納 DP_setsum (int *a, int n, int b){ int i, j, f[b+1]; for ( j=0 ; j<=b ; j++ ) f[j] = 0; f[0] = 1; for ( i=0 ; i<n ; i++ ){ for ( j=b-a[i] ; j>=0 ; j-- ){ if ( f[j] = = 1 ) f[j+a[i]] = 1; } for ( j=b ; j>=0 ; j-- ){ if ( f[j] = = 1 ){ printf (“%d\n”, j ); break;
もっと短いやつ • a[0],…,a[n-1] に数値を格納 DP_setsum (int *a, int n, int b){ int i, j, f[b+1]; for ( j=0 ; j<=b ; j++ ) f[j] = j; for ( i=0 ; i<n ; i++ ){ for ( j=b-a[i] ; j>=0 ; j-- ){ if ( f[j] < f[j+a[i]] ) f[j+a[i]] = f[j]; } printf (“%d\n”, b-f[b] );
どっちが速い? • 分枝限定法 : O(2n) 時間 • 動的計画法 : O(nb) 時間 n = 30 ⇒ 2n ≒ 10億
ウィークポイント • 数に小数があるときは、表に当てはまらない数が出てくる ある程度の桁数までなら、小数点以下の数を ある程度の桁数までなら、小数点以下の数を 1単位とした表を作ればよい • それでも、循環小数・無理数のような数があると計算できない
誤差が n(ε-1) の範囲のものしか見つけられないが メモリを省略 • サブセットサム問題を解く動的計画法は、b が大きいと、大きなメモリと時間を消費する • 精度を犠牲にして、メモリと時間を節約する方法がある • 表のマスを、縦 ε個ずつまとめる 縦のマスの数は b/ε個に減る • (x,y) のマスに書き込む代わりに、 (x, Ly/ε」) に書き込む エラーが最大 ε-1 発生 n 反復繰り返すと、エラーの最大は n(ε-1) 誤差が n(ε-1) の範囲のものしか見つけられないが メモリと時間は 1/εにできる
サブセットサムの解の数 • n 個の数(a1,…,an)がある。これらの数の組合せで、合計がちょうど b になるようなものは、いくつあるか? • 15, 24, 65, 32, 4, 55, 54, 23 の組合せで合計 145 となるものはいくつ?
動的計画の拡張 • f( i, m ) : a1 ,…, ai の組合せで、和が m となるものの数 • i = 1,…,n 、 m = 0,…,b に対して、 f( i, m ) を計算する • f( i, m ) = f( i-1, m ) + f( i-1, m-ai ) • f( 1, m ) は簡単に計算できる 計算時間は O(nb)
練習問題 • 1,1,2,3,3,4,5,7 の組合せで、 合計 10 となる組合せはいくつあるか?
ちょうど k 個の和 • n 個の数(a1,…,an)がある。これらの数のk個の組合せで、合計がちょうど b になるようなものは、いくつあるか? 例) 15, 24, 65, 32, 4, 55, 54, 23 の中の 4個の数の組合せで 合計 145 となる組合せはいくつ?
動的計画の拡張 • f( i, j, m ) : a1 ,…, ai の j 個の組合せで、 和が m となるものの数 • i=1,…,n 、 j=0,…,k 、 m=0,…,b に対して、 f(i,j,m) を計算する • f( i, j, m ) = f( i-1, j, m ) + f( i-1, j-1, m-ai ) • f(1,j,m) は簡単に計算できる 計算時間は O(nkb)
ナップサック問題 • 最大積載量のあるナップサックに荷物を詰める問題 - 荷物はいくつかある - 荷物はそれぞれ重さが違う - 荷物はいくつかある - 荷物はそれぞれ重さが違う - 荷物はそれぞれ価値が違う - 荷物の重さの合計は、最大積載量を超えてはいけない このとき、荷物の価値の合計が最大になる詰め込み方を求めよ
ナップサック問題 (2) • 荷物 i の重さを ai 、価値を ci 、ナップサックの最大積載量を b とすると、定式化は max. Σ ci xi s.t. Σ ai xi ≦ b xi ∈ { 0,1 } • NP-hard 問題である
動的計画の拡張 max. Σ cj xj s.t. Σ aj xj = m x1,…, xi ∈ { 0,1 } max. Σ cj xj • f(i,m) : 荷物 1,…,i の組合せで、重さの和が m となるものの中での、 価値の和の最大値 max. Σ cj xj s.t. Σ aj xj = m x1,…, xi ∈ { 0,1 } • f( i, m ) = max { f( i-1, m ), f( i-1, m-ai ) + ci } • f(1,m) は簡単に計算できる max. Σ cj xj s.t. Σ aj xj = m xi = 0 x1,…, xi -1 ∈ { 0,1 } max. Σ cj xj s.t. Σ aj xj = m- ai xi = 1 x1,…, xi-1 ∈ { 0,1 } 計算時間は O(nb)
ナップサックDPコード • a[0],…,a[n-1] に重さ、 c[0],…,c[n-1] に価値を格納 DP_knapsack (int *a, int *c, int n, int b){ int i, j, m=0, mx=0, f[b+1]; for ( j=0 ; j<=b ; j++ ) f[j] = -1; f[0] = 0; for ( i=0 ; i<n ; i++ ){ for ( j=b-a[i] ; j>=0 ; j-- ){ if ( f[j] >= 0 && f[j+a[i]] < f[j]+c[i] ) f[j+a[i]] = f[j]+c[i]; } for ( j=b ; j>=0 ; j-- ) if ( f[j] > m ){ m = f[j]; mx = j; } printf (“weight = %d, profit = %d\n”, mx, m );
練習問題 • n 個の数(a1,…,an)がある。これらの数を3つのグループに分け、 1つ目のグループの合計が b1 に、2つ目のグループの合計が b2 になるようなものはあるか? • この問題を解くための動的計画法を設計しなさい 問題の例)15, 24, 65, 32, 4, 55, 54, 23 を1番目、2番目のグループの合計が 120,110 とするようななる分け方はあるか?
直線構造の利用 • 動的計画法のキモは、大きさが小さい問題の解を用いて、現在の問題が解けるかどうか。 • 言い方を変えれば、左から右に問題を解いていったときに、自分の答えが、自分の左だけを見て作れるか • これは、ある種の直線構造があるかどうか、と考えていいだろう (現時点の右側を変更しても、左側に影響がない、あるいは影響が簡単なパラメータになる(重みの合計とか、組合せ的でない)) • つまり、直線構造があれば動的計画を使えばいいし、動的計画を使えるかどうかは、直線構造があるかないか (有向非閉路的グラフとか、木も直線構造)
最長昇順列 • 与えられた数列の中で、小さい順になっている部分列を昇順部分列と言う 例) 1,5,4,7,2,5,3,9,6,8 1,2,6 など • 与えられた数列の中から、最も長い昇順列を見つける問題を最長昇順列問題という • この問題は直線構造がある
直線構造はあるか • i 番目の数 ai が最後となるような昇順列を考える • このような昇順列の後ろに何を追加しても、前半部分が昇順列でなくなることはない • つまり、もし、ai が最適解に含まれるなら、前半部分(ai より小さな添え字を持つ数)が作る昇順列は、そのようなものの中で最長になっているはず このあたりが直線構造
直線構造はあるか • i 番目の数 ai が最後となるような昇順列の中で最も長いものの長さを f(i) とする • i 番目の数 ai が最後となるような昇順列は、aj, aj<ai, j<i であるような aj で終わる昇順列の最後に ai を付け足して得られる 各aj まで終わる最長列の中で、最も長いものに ai を付け加えればいい • f(i) = max { f(j) | aj<ai, j<i } +1 • 計算時間は O(n2) • ヒープの亜種に、「添え字がここまでのものの中で最大値は何?」という質問に O(log n) 時間で答えられるやつがあるので、それを使うと、 O(nlog n)
パスの数 • 有向サイクルを含まないような、非閉路的グラフの、2点間を結ぶ有向パスの数を計算する
グラフの直線的な構造を利用 • 非閉路的なので、グラフの頂点に終点から始点方向に、終点に近いほうが必ず小さくなるような番号付けができる (自分から行ける頂点は自分よりも小さい) • この順番でスキャンすることで、表を作ることはできないが、 数が数えられる 3 6 5 7 2 9 0 4 8 1
番号順にパスの数を数える • グラフが非閉路的なので、頂点 v から終点へのパスの数は、v からいける頂点から終点までのパスの数の総和 番号順に、パスの数を計算すればよい 1+2 11+3 2+3+6 14+33+40 3 3 1+1 14 6 5 11 7 33 2 2 9 87 0 1 4 6 8 40 14+11+2+6 1 1 1+2+3 33+6+1
まとめ • 組合せ最適化問題に対する動的計画法の作り方 (漸化式を作れれば良い) • サブセットサム・その変形・ナップサック問題・最長部分昇順列・非閉路的グラフの2点間を結ぶパスの数、に対する動的計画法