プログラミング論 I 2009年06月18日 関数の再帰呼び出し http://www.ns.kogakuin.ac.jp/~ct13140/Prog.2009/ F-1 概要 • 関数の再帰的に使用する – 結構難しいです. • 前期で最も重要な事項 –関数 • 後期で最も重要な事項 – ポインタ F-2 再帰呼び出し 関数の作成 と 呼び出し がC言語の真骨頂 F-3 関数の多段呼び出し 0/3 1 2 3 4 10 11 12 void func0(){ printf("func0 start!¥n"); printf("func0 end!¥n"); } void main(){ main() func0(); } func0() 実行結果 func0() func0 start! func0 end! F-4 関数の多段呼び出し 1/3 1 2 3 4 5 6 7 8 9 10 11 12 void func0(){ printf("func0 printf("func0 } void func1(){ printf("func1 func0(); printf("func1 } void main(){ func1(); } start!¥n"); end!¥n"); start!¥n"); end!¥n"); 実行結果 func1 func0 func0 func1 start! start! end! end! F-5 関数の多段呼び出し 2/3 func0() func1() func0() func1() main() F-6 関数の再帰呼び出し • ある関数から,自分自身を呼び出すことを 再帰呼び出しという. – 例えば,関数funct() の中で,funct() を 呼び出すなど. • 使い方を間違えると,永遠に終了しないプ ログラムになりやすいので注意. F-7 関数の再帰呼び出し 1 2 3 4 5 6 7 8 void pr_hl(){ printf("Hello!start\n"); pr_hl(); printf("Hello!end\n"); } void main(){ pr_hl(); } 動作の解説は次ページ 注意!これは失敗例です. F-8 pr_hl()が 呼ばれたので, pr_hl()に移動. main() pr_hl() 呼び出しを永遠に続け, このプログラムは 終わらない. pr_hl() main()から, pr_hl()が 呼ばれたので, pr_hl()に移動. pr_hl() pr_hl() F-9 再帰:nの階乗 0/5 • nの階乗を,n!と記述する. • n! は,n×(n-1)×(n-2)×…×2×1 • n! は,n × (n-1)! • 5! は,5×4×3×2×1 • 5! は,5×4! F-10 再帰:nの階乗 1/5 • nの階乗は もし n == 1 なら, 答えは,1 もし n > 1 なら, 答えは,n×(n-1)! 上(n==1)が無いと,永遠に終わらない F-11 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 int kaijoh(int x){ int r; if( x == 1 ){ return 1; } else { r = x * kaijoh(x-1); return r; } } void main(){ int n, k; n = 5; k = kaijoh( n ); printf("%d! = %d\n", n, k); } 再帰:nの階乗 2/5 F-12 main() 再帰:nの階乗 3/5 kaijoh(3) kaijoh(3)は,3*kaijoh(2) kaijoh(2) kaijoh(2)は,2*kaijoh(1) kaijoh(1) kaijoh(1)は,1 kaijoh(1)=1 kaijoh(1)=1 kaijoh(2)=2 kaijoh(2)は,2*1 kaijoh(2)=2 kaijoh(3)は,3*2 kaijoh(3)=6 n==1の時は, これ以上再帰呼 び出ししない. これがないと 無限に続く F-13 再帰:nの階乗 4/5 kaijoh( 4 =4* kaijoh( 3 =4* (3* kaijoh( 2 =4* (3* (2* kaijoh(1) =4* (3* (2* 1 =4* (3* ( 2 =4* ( 6 =24 ) ) ) ) ) ) ) ) ) ) ) F-14 失敗例 1 2 3 4 5 6 7 8 9 10 11 int kaijoh(int x){ int r; r = x * kaijoh(x-1); return r; } void main(){ int n, k; n = 5; k = kaijoh( n ); printf("%d! = %d\n", n, k); } kaijoh(5)を呼び出し ↓ kaijoh(4)を呼び出し ↓ kaijoh(3)を呼び出し ↓ kaijoh(2)を呼び出し ↓ kaijoh(1)を呼び出し ↓ kaijoh(0)を呼び出し ↓ kaijoh(-1)を呼び出し ↓ kaijoh(-2)を呼び出し ↓ kaijoh(-3)を呼び出し ↓ 無限に続く F-15 再帰:組み合わせ nCr • 組み合わせ(Conbination) nCr は, r>1なら, nCr=n-1Cr-1 * n / r r==1なら, nCr=n F-16 再帰:ハノイの塔 • ハノイの塔というパズル 引用 奥野かるた店より http://www.okunokaruta.com/goods/goods12e10.jpg F-17 再帰:ハノイの塔 • ルール – 円盤置き場が3カ所ある. • 通常,円盤の中心に穴があり, 円盤置き場には杭が立っている. – 円盤がn枚あり,全て大きさが異なる. – 小さい円盤の上に大きな円盤は置けない. – 円盤置き場の1番上の1枚を別の円盤置き場 に移動することができる. F-18 再帰:ハノイの塔 • 目的 – n枚の円盤を,ある円盤置き場から別の円盤 置き場に移動する. – できるだけ少ない円盤移動が好ましい. • 解法 – 最適解は, 円盤がn枚の場合,2n-1回の円盤移動. – プログラミングでは,再帰的手法で解ける. F-19 ハノイの塔:2枚の例 1 2 ↓開始!(円盤2枚を左から中央に) 1 2 2 円盤 円盤置き場 1回目の移動. 円盤1を左から右に. 1 F-20 ハノイの塔:2枚の例 2 1 2 1 2 1 1 2 2回目の移動. 円盤2を左から中央に. 3回目の移動. 円盤1を右から中央に. 22-1=3回で終了. 完了! F-21 ハノイの塔:3枚の例 1 2 3 1回目の移動. 円盤1を左から中央に. 2 3 1 3 1 3 2回目の移動. 円盤2を左から右に. 円盤3の上が空に. 2 1 2 3回目の移動. 円盤1を中央から右に. 中央の円盤置き場が 空いた.円盤3を 中央に移動可能. F-22 ハノイの塔:3枚の例 1 2 3 3 1 2 1 3 2 1 2 3 1 2 3 4回目の移動. 円盤3を左から中央に. ついに円盤3が移動! 5回目の移動. 円盤1を左から右に. 6回目の移動. 円盤2を右から中央に. 7回目の移動. 円盤1を左から中央に. F-23 ハノイの塔:4枚の例 1 2 3 4 • 24-1=15回の移動. • 円盤3枚の移動方法(7回)は分かってい る... ↓ • 円盤3枚の移動方法を使えば, 円盤4枚の移動方法を導き出せる. F-24 ハノイの塔:4枚の例 1 2 3 4 先ほどの方法で, 円盤3枚(1~3)を 左から右に移動. 7回の円盤移動. 1 2 3 4 4 1 2 3 4 1 2 3 円盤4を左から中央に. 1回の円盤移動. 先ほどの方法で, 円盤3枚(1~3)を 右から中央に移動. 7回の円盤移動. 合計 7+1+7=15回 F-25 ハノイの塔:n枚 • n枚の移動方法が分かれば, それを用いて n+1枚の移動方法が分かる. • 1枚の移動方法は分かる. ↓ • 任意の枚数の移動方法が分かる F-26 ハノイの塔:n枚 1 : n-1 n n-1枚を左から右に 移動する. 移動回数2n-1-1回. 1 : n-1 n n 1 : n-1 n 1 : n-1 円盤nを左から中央に. 移動回数1回. n-1枚を右から中央に 移動する. 移動回数2n-1-1回. 合計 (2n-1-1)+1+(2n-1-1)=2n-1回 F-27 ハノイの塔を解く関数 • ハノイの塔の移動(移動枚数,移動元の円盤置 場,移動先の円盤置場,待避円盤置場) もし,移動枚数が1なら, 1枚を「移動元」から「移動先」に移動する. もし,移動枚数が1より大きければ, (n-1)枚を「移動元」から「待避」に移動. n枚目を「移動元」から「移動先」に移動. (n-1)枚を「待避」から「移動先」に移動. F-28 ハノイの塔を解く関数 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 #include <stdio.h> void hanoi(int n, int plFrom, int plTo, int plOther){ if( n == 1 ){ printf("move disk.%d ", n); printf("[%d -> %d]\n", plFrom, plTo); } else { hanoi(n-1, plFrom, plOther, plTo); printf("move disk.%d ", n); printf("[%d -> %d]\n", plFrom, plTo); hanoi(n-1, plOther, plTo, plFrom); } } void main(){ hanoi(5, 0, 1, 2); } F-29 フィボナッチ数列 • 以下の数列がフィボナッチ数列 1,1,2,3,5,8,13,21,34,… an+an+1=an+2 となる. ただし,a0=1,a1=1 • fibo(n)=fibo(n-1)+fibo(n-2) という再帰的な発想で記述可能. – ただし,for文で加算した方が短時間で計算可能 F-30 フィボナッチ数列 fibo(5) fibo(4) fibo(3) fibo(2) fibo(1) =1 fibo(2) fibo(0) fibo(1) =1 =1 fibo(3) fibo(1) =1 fibo(2) fibo(0) fibo(1) =1 =1 f(2)計算などの, 同じ処理を何度も fibo(0) fibo(1) 行っている. =1 =1 大変に非効率的 F-31 計算量/計算時間 • n枚のハノイの塔 – 2n-1回 ≒ 2n に比例する計算量/時間 • n個の数字をバブルソートで並び替える. – n2 に比例する計算量/時間 • n個の数字をクイックソートで並び替える – n×log(n) に比例する計算量/時間 • n個の数字をバケツソートで並び替える – n に比例する計算量/時間 F-32 練習 (い) • fibo(n)=fibo(n-1)+fibo(n-2) という 再帰的な考え方で, フィボナッチ数列の第n項を求める関数を 記述せよ. F-33 解答 (い) #include <stdio.h> int fibo(int n){ if( n <= 1 ){ return 1; } else { return fibo(n-1)+fibo(n-2); } } void main(){ printf("%d\n", fibo(7)); } これは例 F-34
© Copyright 2025 ExpyDoc