VBA技術解説
フィボナッチ数列(再帰呼び出し)

ExcelマクロVBAの問題点と解決策、VBAの技術的解説
公開日:2020-02-08 最終更新日:2020-07-06

フィボナッチ数列(再帰呼び出し)


フィボナッチ数列は、最初の二項は 0, 1 で、以後は直前の2つの項の和となっている数列です。
VBAでフィボナッチ数列を出力してみましょう。
もちろん、フィボナッチ数を勉強しようとか、何かに使おうという事ではありません。
再帰呼び出しの練習、再帰プロシージャーをしっか理解しようという趣旨になります。


フィボナッチ数の詳細については、フィボナッチ数(ウィキペディア) こちらをご覧ください。

フィボナッチ数列を出力する再帰VBA

Sub FibonacciMain()
  Call FibonacciNumber(100)
End Sub

Sub FibonacciNumber(nEnd, Optional n1, Optional n2)
  If IsMissing(n1) Then
    n1 = 0: Debug.Print n1
    n2 = 1: Debug.Print n2
  End If
  If n1 + n2 > nEnd Then Exit Sub
  Debug.Print n1 + n2
  Call FibonacciNumber(nEnd, n2, n1 + n2)
End Sub
※ここではデータ型の指定を省略しています。

上のFibonacciMainを実行すると、イミディエイトに以下のように出力されます。
0
1
1
2
3
5
8
13
21
34
55
89
ステップイン(F8)で、1行ずつ確認しながら、n1n2の変化を確認して下さい。

Call FibonacciNumber(nEnd, n2, n1 + n2)
これが実行されると、再び、
Sub FibonacciNumber(nEnd, Optional n1, Optional n2)
ここに入ってくることが確認できます。

そして、
If n1 + n2 > nEnd Then Exit Sub
この後は、
End Sub
が繰り返されることが分かる筈です。
ぜひ、ステップイン(F8)で確認してみてください。

再帰呼び出しを使わない場合のVBA

Sub sample()
  Dim n1, n2, nTmp
  n1 = 0: Debug.Print n1
  n2 = 1: Debug.Print n2
  Do While n1 + n2 < 1000
    nTmp = n1 + n2
    Debug.Print nTmp
    n1 = n2
    n2 = nTmp
  Loop
End Sub

これは理解しやすいのではないでしょうか。
このVBAと再帰VBAを見比べてください。
それぞれをステップインで比べると、より理解しやすいはずです。

再帰プロシージャで考慮すべき事項

制限条件の確認

再帰プロシージャでは、再帰を終了する条件を最低 1 つテストする必要があります。
また、妥当な回数の再帰呼び出しを行ってもこの条件が満たされない場合の処理も必要です。
必ず満たされる条件を最低 1 つ用意しないと、プロシージャが無限ループに陥る可能性が高くなります。
再帰プロシージャを作成した場合、最低 1 つの制限条件を満たしていることを必ずテストする必要があります。
また、再帰呼び出しが多すぎるためにメモリを使い果たすことがないことを確認する必要があります。

メモリ使用状況

アプリケーションがローカル変数に使用できる領域は限られています。
プロシージャが自分自身を呼び出す際、ローカル変数のコピーが毎回作成され、領域を消費します。
このプロセスがいつまでも続くと、最終的には StackOverflowException エラーが発生します。

Functionプロシージャーの戻り値を使って再帰呼び出しする場合

Functionプロシージャーの戻り値を使って再帰呼び出しする場合については以下で解説しています。
再帰呼出しについて(再帰プロシージャー)
・指定数値の階乗を求める再帰VBA ・再帰プロシージャで考慮すべき事項 ・再帰呼び出しの実践例

実際の再帰処理では、Functionの戻り値とByRef引数を組み合わせて使用することも多々あります。

再帰呼び出しの実践例

ここでは、再帰呼出しを使い、FileSystemObjectで全サブフォルダのファイル一覧を取得しています。
サブフォルダの階層は際限がない為、一般的なループでは処理が難しいのですが、
再帰呼出しを使う事で、短いVBAコードで実現する事が出来ます。
再帰呼出しとFileSystemObjectが出てくるのは、№6からですが、Dir関数と合わせて、一度は目を通しておくと良いでしょう。

各種ゲーム

以下のようなゲーム作成においては、再帰呼び出しは必須になります。

数独(ナンプレ)を解くVBAに挑戦
数独は、一般に「ナンバープレース(ナンプレ)」と呼ばれ、外国では「sudoku」と呼ばれているようです、この数独をExcelマクロVBAで解いてみようと言う事です。解き方は、とにかく片っ端から数字を当てはめていくという、なんとも芸の無い方法です。
数独(ナンプレ)を解くアルゴリズムの要点とパフォーマンスの検証
数独(ナンプレ)を解くアルゴリズムを例に、アルゴリズムの要点と、それによるパフォーマンスを検証します、数独(ナンプレ)を解くVBAに挑戦 ここでは、とにかく全ての数字を当てはめていくという、いわば全数チェックでの解法を使いました。考察するまでもなく、かなりの無駄がある事は明白です。

ナンバーリンク(パズル)を解くVBAに挑戦
ナンバーリンクというパズルがあります、これをエクセルVBAで解いてみようと思います、数独(ナンプレ)に続くパズルVBA解法の第二弾です。ナンバーリンクをご存じない方は、、ウィキペディア ナンバーリンクのおためし問題 このあたりをお読みください。
ナンバーリンクを解くVBAのパフォーマンス改善
「ナンバーリンク(パズル)を解くVBAに挑戦」で作成したVBAでナンバーリンクを解く事には成功しました、しかし、10×10なら数分で解けるものの、10×18でやったところ、4時間半もかかってしまいました。12×12では、待ちきれずに途中で止めてしまいました。

オセロを作りながらマクロVBAを学ぼう
ExcelマクロVBAでオセロ(リバーシ)を作っていきながら、マクロVBAを学んで行きましょう。目的は、マクロVBAの学習であり、思考を整理しVBAでプログラミングする学習です。従って、強いソフトを作ることが目的ではありませんので、最近流行のAIなんちゃら…なんていうのは考えるつもりはありません。



同じテーマ「マクロVBA技術解説」の記事

ExecuteExcel4Macroについて
「Excel 4.0 マクロ」の使い方
再帰呼出しについて(再帰プロシージャー)
フィボナッチ数列(再帰呼び出し)
文字列でのセル参照と文字列の計算式について(Evaluate,INDIRECT)
リボンを非表示、2003以前ならメニューを非表示
印刷ページ設定の余白をセンチで指定する(CentimetersToPoints)
文字列としてのプロシージャー名を起動する方法(Run,OnTime)
ドキュメントの作成者を取得(GetObject,BuiltinDocumentProperties)
画像サイズ(横x縦)の取得について
文字種(ひらがな、全半角カタカナ、半角英大文字等々)の判定


新着記事NEW ・・・新着記事一覧を見る

ExcelマクロVBA入門目次|エクセルの神髄(2024-03-20)
VBA10大躓きポイント(初心者が躓きやすいポイント)|VBA技術解説(2024-03-05)
テンキーのスクリーンキーボード作成|ユーザーフォーム入門(2024-02-26)
無効な前方参照か、コンパイルされていない種類への参照です。|エクセル雑感(2024-02-17)
初級脱出10問パック|VBA練習問題(2024-01-24)
累計を求める数式あれこれ|エクセル関数応用(2024-01-22)
複数の文字列を検索して置換するSUBSTITUTE|エクセル入門(2024-01-03)
いくつかの数式の計算中にリソース不足になりました。|エクセル雑感(2023-12-28)
VBAでクリップボードへ文字列を送信・取得する3つの方法|VBA技術解説(2023-12-07)
難しい数式とは何か?|エクセル雑感(2023-12-07)


アクセスランキング ・・・ ランキング一覧を見る

1.最終行の取得(End,Rows.Count)|VBA入門
2.RangeとCellsの使い方|VBA入門
3.セルのコピー&値の貼り付け(PasteSpecial)|VBA入門
4.繰り返し処理(For Next)|VBA入門
5.変数宣言のDimとデータ型|VBA入門
6.ブックを閉じる・保存(Close,Save,SaveAs)|VBA入門
7.並べ替え(Sort)|VBA入門
8.条件分岐(IF)|VBA入門
9.セルのクリア(Clear,ClearContents)|VBA入門
10.マクロとは?VBAとは?VBAでできること|VBA入門




このサイトがお役に立ちましたら「シェア」「Bookmark」をお願いいたします。


記述には細心の注意をしたつもりですが、
間違いやご指摘がありましたら、「お問い合わせ」からお知らせいただけると幸いです。
掲載のVBAコードは動作を保証するものではなく、あくまでVBA学習のサンプルとして掲載しています。
掲載のVBAコードは自己責任でご使用ください。万一データ破損等の損害が発生しても責任は負いません。



このサイトがお役に立ちましたら「シェア」「Bookmark」をお願いいたします。
本文下部へ