制御構造
制御構造(せいぎょこうぞう)は、コンピュータ・プログラミング言語、特に手続き型プログラミング[1]や命令型プログラミング[2]において、ループや飛び越しなどといった、手続き(プロシージャ)中の実行順を順次実行から変化させたり、サブルーチン呼出しやその戻り、などといった制御を行う「文 」などの構造(言語の構成要素)である[3]。
制御構造の種類は言語によって様々だが、典型的には以下のようなものがある(用語「ブロック」については、ブロック の記事を参照)。
- 無条件に実行箇所を移動する(無条件の分岐命令、ジャンプ)
- 何らかの条件の成立・不成立に従い、ブロックの実行・不実行を選択する(条件付き分岐命令、選択)
- ブロックを繰り返し実行する(ループ)
- ジャンプの一種だが、その続きに戻れるもの(サブルーチン呼出、コルーチン)
- 継続(特にcall/cc)
- プログラムの停止(理論的には重要だが(停止性問題を参照)、実際的にはexitシステムコールなど、OSのプロセス制御機構を使うことが専らであり、言語機能として制御構造で持つ意味は無い)
割り込みとシグナルは制御フローを変化させる別の機構であり、サブルーチンに似ているが、通常は言語内からではなく外部のイベントなどの結果として非同期に発生するものである(言語内から外部のイベントを引き起こし、結果として発生させることもできる例もある。たとえば、ゼロ除算などでそのようになるシステムがある)。自己書き換えコードも副作用によって制御フローを変化させることができる。割り込み的なものを扱うことができるプログラミング言語(の機能)はいくつかある。自己書き換えをプログラマが明示的に扱えるプログラミング言語はあまりないが、初期化の時だけ特別扱いが必要といったコードの最適化に自己書き換えを利用する、処理系の実装上のテクニックといったようなものもある。
機械語において制御構造に相当するのは分岐命令で、通常は連続的にカウントが進められるプログラムカウンタを、不連続に変更する命令である。ほぼ全てのプロセッサは(条件付きおよび無条件の)分岐命令を持つ(無いと、理論的に言うと計算可能性を満たさないため、コンピュータとして成立しない)。また、サブルーチン呼出しを(大なり小なりハードウェアによる支援を含んで)サポートする命令を持つプロセッサが多い(こんにち存在する汎用プロセッサの著名なISAでこのサポートの無いのは、60年代の設計であるIBMのSystem/360ぐらいであろう)。全くハードウェアによるサポートが無いと、サブルーチン呼出しに面倒なトリックが必要なことがある。一方で前述のような、ループの処理(レジスタの値をデクリメントし、ゼロでなければジャンプ、といったような)を直接サポートするような命令を持つプロセッサは、専用命令として積極的に持つものもあるが、一般にはあまり多くなく、特にいわゆるRISCでは避けられる。そのため、コンパイラのコード生成は制御構造からジャンプ命令等を適宜組み合わせたコードを生成するように実装される。
原始的な機能
[編集]ラベル
[編集]ラベルとは、コード中の固定の位置を示す何らかのシンボルであり、gotoの飛び先や、breakで抜ける対象として参照される。GCC拡張であるが、void * ポインタの値として扱うこともできる(Labels as Values[4])。
行番号は一部の言語(FORTRANやBASIC)でラベルの一種として使われ、負でない整数がソースコードの各テキスト行の先頭に置かれる。行番号を使用する言語では、連続で実行される文には行番号が増えるように行番号を与える必要がある、とBASICしか知らない者は誤解しているが、FORTRANにはそのような制限は無い。BASICで行番号が昇順なのは、テキストの編集にフルスクリーンエディタが一般的ではなかった時代のラインエディタのみによる編集では、行番号に従ってシステムが並べ直してくれたほうが便利だったからであり、「行番号が増えるように行番号を与える必要がある」というのは本末逆転である。例えば BASIC では次のようになっている。
10 LET X = 3
20 PRINT X
CやAdaといった言語のラベルは識別子であり、文の前に書かれ、その直後にコロンが書かれる。例えば C では次のようになる。
Success: printf ("The operation was successful.\n");
Algol 60 言語はラベルとして識別子も非負整数も使用可能(どちらもその後にコロンが続く)だが、多くのAlgol系言語では非負整数をラベルとして許容していない。
goto
[編集]goto 文は最も典型的な無条件のジャンプである。キーワードとしては大文字だったり小文字だったり空白が入ってgo to
だったり、単にgoだったりするが、その構文はだいたいのものが以下のようになっている。
goto label
goto 文の実行により、その次に実行する文は、ラベルが示す箇所の直後の文となる。
サブルーチン
[編集]サブルーチンには、手続き、ルーチン、プロシージャ、関数(特に値を返す場合)、メソッド(特に何らかのクラスに属する場合)など様々な名称がある。
1950年代、コンピュータのメモリは非常に小さかったため、サブルーチンの第一の目的はプログラムのサイズを削減することにあった。サブルーチンとして書かれたコードをプログラム内のあちこちから使用することでプログラム全体のコードサイズを削減したのである。現在ではサブルーチンはプログラムを構造化するために使われる。すなわち、特定のアルゴリズムを分離したり、特定のデータにアクセスするメソッドを隠蔽したりする。多数のプログラマが共同でプログラム開発をする場合、サブルーチンはある種のモジュール性を提供し、仕事の分割点の役割も果たす。
サブルーチンに引数があればさらに便利になる。多くのプログラミング言語には平方根を求めるサブルーチンが組み込まれており、引数として平方根を求めたい数を与えることができる。
プログラミング言語によっては再帰呼び出しが可能である。つまり、サブルーチンが直接的あるいは間接的に自分自身を呼び出すことができる。クイックソートや木構造を探索するアルゴリズムなどは再帰を使った方が素直に表現できる。
サブルーチンを使用すると、引数の受け渡し、サブルーチン呼び出し、コールスタック処理、サブルーチンからの復帰などのオーバヘッドによりプログラム性能が若干低下する。実際のオーバヘッドはハードウェアおよびソフトウェアのアーキテクチャに依存する。コンパイラによってはインライン展開を効果的に使用してオーバヘッドの低減を図るものもある。
プログラミング言語によってはサブルーチンの物理的な最後尾に到達しないとサブルーチンから復帰できない方式のものもある。他の言語には return や exit 文がある。これはサブルーチンの最後尾への分岐と等価であり、制御構造を複雑化するものではない。必要に応じて複数のそれらの文をサブルーチン内に置くことができる。
必要最小限の構造化制御フロー
[編集]1966年、Böhm と Jacopini は Communications of the ACM 誌で論文を発表し[5]、goto を使って書かれたプログラムが選択 (IF THEN ELSE) とループ (WHILE condition DO xxx) のみを使って goto を使わずに書き換えられることを示した(コードの一部を複製したり、真理値フラグ変数を追加する必要がある)。後に彼らは選択もループ(と追加の真理値変数)で置き換え可能であることを示した(「構造化定理」)。
非常に良く誤解されているが、そのような書き換えが可能という事実は、単に「機械語で書けば何でも書ける」という事実と同程度の意味しかなく、それが望ましいということは全く意味しない。理論的(理論計算機科学的)にはコンピュータは一種類の命令、たとえば「subtract one number from another and branch if the result is negative」さえあれば何でもできるが(チューリング完全あるいは「万能」、en:One instruction set computer も参照)全く実用的ではなく、実際のコンピュータは多数の命令を備えているということと類似している。
Böhm と Jacopini の論文は全てのプログラムから goto 文を無くすことができることを示した。
また、他の研究により入り口と出口がそれぞれひとつになっている制御構造が他の構造よりも理解し易いということが示された。特にそのような制御構造はプログラムの任意の箇所に制御構造を乱すことなく挿入可能な点が有利とされた。
しかし実は、「理論に従ってgoto 文を無くしたプログラム」が「理解し易い」ものであるか否かは不明であり、実際のところ全くそのようにはならないのである[要出典]。
構造化された制御要素
[編集]以下ではなぜかキーワードに変にこだわっているが、そういった字句にこだわるのではなく、構文(シンタックス)として総合的に捉えれば、たいしてこだわる意味はない。この節冒頭に挙げたリンク先の各記事を参照。
- 終了キーワードがない言語
- Algol 60、C、C++、Haskell、Java、Pascal、Perl、PHP、PL/I、Python、PowerShellなど。この種の言語は文の並びをひとまとめ(ブロック)にする何らかの方法を持っている。
- Algol 60、Pascal:
begin
...end
- C、C++、Java、Perl、PHP、PowerShell: 中括弧を使用
{
...}
- PL/1:
DO
...END
- Python: インデントのレベルを使用(オフサイドルール参照)
- Haskell: インデントのレベルか中括弧を使用でき、それらを自由に混合可能
- Algol 60、Pascal:
- 終了キーワードがある言語
- Ada、Algol 68、Modula-2、Fortran 77、Visual Basic など。終了キーワードはいくつかの種類がある。
- Ada、Fortran 90: 終了キーワードは
end
+ 空白 + 開始キーワード。例えば、if
...end if
,loop
...end loop
- Algol 68: 開始キーワードを逆に綴る。例えば、
if
...fi
,case
...esac
- Fortran 77: 終了キーワードは
end
+ 開始キーワード。例えばIF
...ENDIF
,DO
...ENDDO
- Modula-2: 開始キーワードに関わらず常に
END
という終了キーワードを使う。 - Visual Basic: 制御構造毎に固有の終了キーワード。
If
...End If
;For
...Next
;Do
...Loop
;While
...Wend
- Ada、Fortran 90: 終了キーワードは
選択
[編集]if-then-(else)
[編集]条件式と条件付き実行は、条件節(たいていは「式 (プログラミング)」)の評価結果の真偽によって異なる式やブロックを選択実行する。
IF..GOTO
- 非構造化言語に見られる形式で、典型的な機械語命令をそのまま言語に持ってきたものである。条件が真なら指定されたラベル(または行番号)へジャンプ (GOTO) する。
IF..THEN..(ENDIF)
- ジャンプに限らず、単純な文や入れ子になったブロックを THEN というキーワードの後に置くことができる。構造化された形式である。
IF..THEN..ELSE..(ENDIF)
- 上と同じだが、条件が偽の場合の動作も記述できる。これが最も一般的な形式で、様々なバリエーションがある。終了キーワード
ENDIF
が必要な場合とそうでない場合がある。C言語やそこからの派生言語では終了キーワードは不要で、'then' に相当するキーワードも不要なことが多いが、その場合は条件式を括弧で囲む必要がある(といったような変な覚え方をするより、BNFを読んで構文規則を理解してしまったほうが早い)。
elseif
[編集]「宙ぶらりんelse問題(dangling else problem、en:Dangling else)」も関係するのだが、文法の設計(デザイン)によっては、
IF cond THEN ... ELSE IF cond THEN ... ELSE IF cond THEN ... FI FI FI
のように、「複数の場合に対する場合分け」の単純な多分岐であるにもかかわらず、どんどんネストが深くなるような書き方をせざるをえない場合がある(C言語の文法の設計は、こうなってはいない)。これは ELSEIF のようなキーワードの導入で解決できる。elseif
, elsif
, elif
など言語によるバリエーションが多いので、テキストエディタによるリアルタイムなシンタックスハイライトが非常に有効である。言語によっては else if
という「2語から成るキーワードのようなもの」という設計のものもある。
Pascal: | C: | シェルスクリプト: | Python: | Lisp: | Smalltalk: |
---|---|---|---|---|---|
if a > 0 then begin
writeln("yes")
end else begin
writeln("no")
end
|
if (a > 0) {
printf("yes");
} else {
printf("no");
}
|
if [ $a -gt 0 ]
then
echo "yes"
else
echo "no"
fi
|
if a > 0:
print "yes"
else:
print "no"
|
(princ
(if (plusp a)
"yes"
"no"))
|
Transcript show:
(
a > 0
ifTrue:
[
'yes'
]
ifFalse:
[
'no'
]
).
|
あまり一般的でないバリエーションとして、以下のような例がある。
- FORTRANなどの一部の言語では、3方向の分岐を扱う「算術IF文」があり、数値を正か、ゼロか、負か判定して処理を分岐させる。
- 多くの関数型言語などでは
if文
が関数や式として実装されており、そのようなifは評価した式の結果を返す。 - 一部の言語では
if文
が演算子の様に実装されており、例えばC言語の条件演算子がある。 - PerlではC言語風の
if
だけでなく、when
とunless
や、コードの後に条件式が来るif
がある。 - Smalltalkでは言語組込みの機能としてではなく、
ifTrue
とifFalse
というメッセージに手続き引数を与えることで、条件付き実行ができる。
一般論として、関数の引数を積極評価してしまう言語では、条件実行のようなものを関数にできない。遅延評価のような機構が何かあれば、条件実行を特に言語機能にしなくても、引数を遅延評価する関数によって、条件実行もできる。
パターンマッチング
[編集]ここではOCamlの例を挙げる。
match fruit with | "apple" -> cook pie | "coconut" -> cook dango_mochi | "banana" -> mix;;
switchとcase
[編集]switch文(言語によっては「case文」)は、指定された値を指定された定数群と比較し、最初に一致した定数に従ってその後の処理を決定するものである。一般にどの定数とも一致しなかった場合を想定したデフォルト動作を 'else' や 'otherwise' などとして用意しておく。ルックアップテーブルなどを使ったコンパイラ最適化が可能である。動的プログラミング言語では比較対象が定数式である必要はなく、パターンマッチに拡張することが可能である。例えば下記のシェルスクリプトの例で '*)'
は任意の文字列にマッチングする正規表現を使ってデフォルト動作を指定している。SQLの decode
のように、関数のような見た目のものもある。
Pascal: | C: | シェルスクリプト: |
---|---|---|
case someChar of
'a': actionOnA;
'x': actionOnX;
'y','z':actionOnYandZ;
else actionOnNoMatch;
end;
|
switch (someChar) {
case 'a': actionOnA; break;
case 'x': actionOnX; break;
case 'y':
case 'z': actionOnYandZ; break;
default: actionOnNoMatch;
}
|
case $someChar in
a) actionOnA ;;
x) actionOnX ;;
[yz]) actionOnYandZ ;;
*) actionOnNoMatch ;;
esac
|
ループ
[編集]ループはソースコード上で1回だけ書かれた文の並びを連続して複数回実行することである。ループの「中」のコード(本体と呼び、下記の例では xxx で表されている)は指定回数実行されるか、指定されたコレクションの各要素に対応して実行されるか、何らかの条件が成立するまで繰り返し実行される。無限に繰り返されることもある。
SchemeやHaskellのような関数型言語では、ループより再帰呼び出しや不動点コンビネータを使用してプログラミングするのが普通である。末尾再帰は再帰呼び出しの特殊ケースであり、容易にループに変換できる(正確には「末尾呼び出しは容易にジャンプに変換できる。末尾再帰は末尾呼び出しの対象が自身になっているという特殊ケースであり、ループに変換できる」)。
カウント制御ループ
[編集]指定された回数だけブロックを繰り返すループである。本来、その回数だけを指定するなどもっと抽象化されているべきであるが、「ループ変数」などを指定するなど煩雑さがともなっているものが多い。以下の例で N が 1 より小さい場合、ループ本体は全く実行されない。カウントは多くの場合増える方向だけでなく減る方向にも設定可能で、1回に増える量も 1 以外に設定できることが多い(Pascalだけが±1にしかできない)。
FOR I = 1 TO N for I := 1 to N do begin xxx xxx NEXT I end; DO I = 1,N for ( I=1; I<=N; ++I ) { xxx xxx END DO }
多くのプログラミング言語では、カウント制御ループでは整数のみが使われる。浮動小数点数はハードウェアの制限により精度に限界がある(「ハードウェアの制限により」ではない。そもそも間違ったプログラミングなのである)。従って次のようなループでは、
for X := 0.1 step 0.1 to 1.0 do
繰り返し回数が9回の場合と10回の場合がある。これは丸め誤差やハードウェアやコンパイラの違いによって変わってくる(といったように過去には信じている者が多かった。現代ではIEEE 754で標準化されているのだが、このようなプログラミングをしてはならないことに変わりはない)。さらに言えば、Xに繰り返し加算すると丸め誤差が累積していき、想定した数列である 0.1, 0.2, 0.3, ..., 1.0 からかけ離れていくことがありうる(「ありうる」ではなく、「そうなる」と表現すべき、浮動小数点表現による現象である。なお、このような例ばかりを想定して、十進浮動小数点では誤差が出ない、とナイーブに考える者が非常に多いが、間違いである)。
条件制御ループ
[編集]条件(条件式)が指定されており、その式を評価した結果が真であれば(あるいは、偽であれば)ループを繰り返す。条件のテストがループの先頭にある場合と最後にある場合がある。前者の場合、ループ本体を全く実行しないことがありうるが、後者の場合は少なくとも1回はループ本体を実行する。
DO WHILE (test) repeat xxx xxx END DO until test; while (test) { do xxx xxx } while (test);
コントロールブレイクは通常のループ内で値の変化を検出する手段として使われ、値のグループの処理のトリガーとなる。ループ内で変化する値(群)をキーで監視し、可変な値に関連したグループイベント処理へとプログラムのフローを変換する。
DO UNTIL (End-of-File) IF new-zipcode <> current-zipcode display_tally(current-zipcode, zipcount) current-zipcode = new-zipcode zipcount = 0 ENDIF zipcount++ LOOP
コレクション制御ループ
[編集]一部のプログラミング言語(例えば、Ada、D言語、Smalltalk、PHP、Perl、Object Pascal、Java、C#、Visual Basic、Ruby、Python、JavaScript、Fortran 95 およびそれ以降)では、明示的に配列や集合やコレクションの全要素に対応してループを回すことができる。
someCollection do: [:eachElement |xxx]. for Item in Collection do begin xxx end; foreach (item; myCollection) { xxx } foreach someArray { xxx } foreach (someArray as $k => $v) { xxx } Collection<String> coll; for (String s : coll) {} foreach (string s in myStringCollection) { xxx } $someCollection | ForEach-Object { $_ } forall ( index = first:last:step... )
汎用の繰り返し
[編集]C言語の for 文や Common Lisp の do のような汎用性の高い繰り返し要素を使えば、前述の各種ループもその他のループも実現できる。例えば、複数のコレクションを並列に回したりできる。もっとも、個別のループ構造がある場合、そちらを使った方がコードの目的をより明確に表現できるとも言える。
無限ループ
[編集]場合によっては無限にループする方がプログラムに適していることもあるし、何らかのエラーが発生するまでループするという場合もある。実際、イベント駆動型プログラム(サーバなど)はイベント制御ループを永遠に回り続け、プロセスが操作者によって終了させられたときだけループを停止する。
ただし一般には、無限ループはプログラミングのミスで発生する。すなわち、ループ終了条件がループ内で全く発生しないことが原因で意図しない無限ループとなる。
次の繰り返しへの継続
[編集]ループ途中でループ処理を中断してループの先頭に戻り、次の繰り返しを開始したい場合がある。言語によってはこれを実現する continue
とか skip
、next
といった文を用意している。その効果は最も内側のループ本体の実行を途中で止め、そのループの次の繰り返しを最初から行う。もしそのときの実行が最後の繰り返しであった場合、ループそのものを早期に終了させるのと同じことになる。
現在の繰り返しの再実行
[編集]Perl や Ruby といった一部の言語では redo
文によって現在の繰り返しを先頭から再実行することができる。
ループの再実行
[編集]Ruby では、retry
文でループ全体を最初から再実行することができる。
ループからの早期脱出
[編集]カウント制御型ループを使って配列上のデータを検索している際に、必要な要素を見つけたら即座にループから抜け出したいという状況がありうる。プログラミング言語によっては break
とか exit
、last
といった文を用意していて、現在のループを即座に抜けてそのループの直後の文に制御を転送する機能を持っている。サブルーチン内のループで return
を使えば、入れ子になったループからも脱出することになる。多次元配列を入れ子になったループで検索している場合、若干複雑になる(「提案された制御構造」の章参照)。
以下の例はAdaを使ったものである。Ada は「ループからの早期脱出」と「途中にテストのあるループ」の両方をサポートしている。どちらもよく似ているが、コードを比較すればその違いがわかる。いずれにしても、汎用の制御構造であるif文との組み合わせによるものか、専用の制御構造によるものか、という違いでしかない。
with Ada.Text IO;
with Ada.Integer Text IO;
procedure Print_Squares is
X : Integer;
begin
Read_Data : loop
Ada.Integer Text IO.Get(X);
exit Read_Data when X = 0;
Ada.Text IO.Put (X * X);
Ada.Text IO.New_Line;
end loop Read_Data;
end Print_Squares;
Python は break
でループを早期脱出したか否かに依存して、実行されるブロックを指定できる。以下はその例である。
for n in set_of_numbers:
if isprime(n):
print "Set contains a prime number"
break
else:
print "Set did not contain any prime numbers"
Python では for
文も while
文もこのような else
節を使うことができる。else 節は早期脱出が発生しなかったときのみ実行される。
ループ変化条件とループ不変条件
[編集]ループ変化条件とループ不変条件は、ループの正しさを表すのに使われる[6]。
現実的には、ループ変化条件とは非負の初期値を持つ整数式である。変化条件はループを回るたびに減少しなければならないが、正しいループ実行の間は負の値になってはならない。ループ変化条件はループが終了するであろうことを保証するのに使われる。
ループ不変条件は、ループを回る前と各反復において真でなければならない表明である。すなわち、ループが正しく終了するには終了条件とループ不変条件が共に真でなければならない。ループ不変条件は、ループ実行中にループの具体的属性を監視するのに使われる。
Eiffelなどのプログラミング言語でループ変化条件とループ不変条件がサポートされている。Javaではアドオンである Java Modeling Language (JML) という仕様で同様のものをサポートしている[7]。
サブ言語としてのループ
[編集]Lisp方言では、ループを記述するための多機能なサブ言語を提供することが多い。代表的な例としては、Common Lisp の LOOP が挙げられる[8]。初期の例としては Interlisp の Conversional Lisp がある。
ループ機能の比較表
[編集]プログラミング言語 | 条件制御ループ | ループ | 早期脱出 | 継続 | 繰り返しの再実行 | ループの再実行 | ループの正しさの保証 | ||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|
先頭 | 途中 | 末尾 | カウント | コレクション | 汎用 | 無限 [※ 1] | 変化条件 | 不変条件 | |||||
Ada | Yes | Yes | Yes | Yes | 配列 | No | Yes | 深い入れ子 | No | ||||
C | Yes | No | Yes | No [※ 2] | No | Yes | Yes | 深い入れ子 [※ 3] | 深い入れ子 [※ 3] | No | |||
C++ | Yes | No | Yes | No [※ 2] | Yes [※ 4] | Yes | Yes | 深い入れ子 [※ 3] | 深い入れ子 [※ 3] | No | |||
C# | Yes | No | Yes | No [※ 2] | Yes | Yes | Yes | 深い入れ子 [※ 3] | 深い入れ子 [※ 3] | ||||
Common Lisp | Yes | Yes | Yes | Yes | Yes | Yes | Yes | 深い入れ子 | Yes [※ 5] | ||||
Eiffel | Yes | No | No | Yes [※ 6] | Yes | Yes | No | 1レベル [※ 6] | No | No | No [※ 7] | 整数のみ [※ 8] | Yes |
F# | Yes | No | No | Yes | Yes | No | No | No [※ 9] | No | No | |||
FORTRAN 77 | Yes | No | No | Yes | No | No | No | 1レベル | Yes | ||||
Fortran 90 | Yes | No | No | Yes | No | No | Yes | 深い入れ子 | Yes | ||||
Fortran 95およびそれ以降 | Yes | No | No | Yes | 配列 | No | Yes | 深い入れ子 | Yes | ||||
Haskell | No | No | No | No | Yes | No | Yes | No [※ 9] | No | No | |||
Java | Yes | No | Yes | No [※ 2] | Yes | Yes | No | 深い入れ子 | 深い入れ子 | No | 拡張機能 [※ 10] | 拡張機能 [※ 10] | |
JavaScript | Yes | No | Yes | No [※ 2] | Yes | Yes | No | 深い入れ子 | 深い入れ子 | No | |||
OCaml | Yes | No | No | Yes | 配列、リスト | No | No | No [※ 9] | No | No | |||
PHP | Yes | No | Yes | No [※ 2][※ 11] | Yes [※ 12] | Yes | No | 深い入れ子 | 深い入れ子 | No | |||
Perl | Yes | No | Yes | No [※ 2][※ 11] | Yes | Yes | No | 深い入れ子 | 深い入れ子 | Yes | |||
Python | Yes | No | No | No [※ 11] | Yes | No | No | 深い入れ子 [※ 9] | 深い入れ子 [※ 9] | No | |||
REBOL | No [※ 13] | Yes | Yes | Yes | Yes | No [※ 14] | Yes | 1レベル [※ 9] | No | No | |||
Ruby | Yes | No | Yes | Yes | Yes | No | No[※ 15] | 深い入れ子 [※ 9] | 深い入れ子 [※ 9] | Yes | Yes | ||
Standard ML | Yes | No | No | No | 配列、リスト | No | No | No [※ 9] | No | No | |||
Visual Basic .NET | Yes | No | Yes | Yes | Yes | No | Yes | ループの種類毎に1レベル | ループの種類毎に1レベル | ||||
Windows PowerShell | Yes | No | Yes | No [※ 2] | Yes | Yes | No | ? | Yes |
- ^
while (true)
は構文としては無限ループ専用の構文ではないので、ここでは無限ループに含めていない。一方、for (式;;式)
は無限ループ専用とみなしている - ^ a b c d e f g h C言語の
for (init; test; increment)
は汎用であり、カウント制御専用ではないが、カウント制御として使われることが多い。 - ^ a b c d e f C、C++、C# での深い入れ子からの脱出は、ラベルとgoto文を使用する。
- ^ C++11標準で、範囲に基づくforループが導入された。STLには
std::for_each
というテンプレート関数があり、STLのコンテナに対して各要素に単項関数を適用できる[9]。同様の機能はマクロを使っても実現可能[10]。 - ^ LOOP は独自の分岐構文を内包している
- ^ a b カウント制御ループは整数 interval によるイテレーションで実現される。早期脱出は exit に条件を追加することでなされる。
- ^ Eiffelには
retry
という予約語があるが、これはループ制御用ではなく例外処理用である。 - ^ ループ変化条件は整数でなければならず、超限的変化条件はサポートしていない[1]
- ^ a b c d e f g h i 深いブレイクを実現するには、例外処理を活用する必要がある。
- ^ a b Java Modeling Language (JML) が必要
- ^ a b c カウントループは例えばPythonの
range()
を使って incrementing list や generator でシミュレートされる。 - ^ オブジェクト群のイテレーションは PHP 5 で追加された。
- ^
while
関数を使用する(関数ではないが、関数だと誤解している者が多い)。 - ^ ユーザーが汎用ループ関数を定義できる。
- ^ ただし、標準ライブラリに無限ループを実現するloopメソッドが存在する。
構造化非局所制御フロー
[編集]多くのプログラミング言語、特に動的なプログラミングスタイルを指向した言語では、「非局所制御フロー」の構造を持っている。これを使うと実行の流れは現在のコンテキストから離れ、事前に定義された場所から続行される。「条件」、「例外」、「継続」の3種類の典型的な非局所制御構造がある。
条件
[編集]PL/I は標準で22種類の条件(ZERODIVIDE、SUBSCRIPTRANGE、ENDFILE など)をサポートし、これを発生(RAISE)させ、ON condition action; で解釈することができる。プログラマは独自の条件を定義することもできる。
構造無しの IF 文のように action にはひとつの文しか書けないので、多くの場合 GOTO 文を使って制御フローを継続する必要がある。
しかし、実装によってはこれは空間と時間を無視できないくらい浪費する(特に SUBSCRIPTRANGE の場合)。多くのプログラマは条件を使わないようコードを書くことが多かった。
典型的な文法例:
ON condition GOTO label
例外
[編集]最近の言語はGOTO
文を使用せずに例外処理を行う構造化された制御構造を備えている。
try {
xxx1 // この中のどこかで以下を使用する
xxx2 // '''throw''' someValue;
xxx3
} catch (someClass& someId) { // someClass の場合をキャッチ
actionForSomeClass
} catch (someType& anotherId) { // someType の場合をキャッチ
actionForSomeType
} catch (...) { // 既にキャッチされていない任意の値をキャッチ
actionForAnythingElse
}
任意の catch 節が上記の例では使用されている。D言語、Java、C#、Python では try
構造に finally
節を追加することができる。try
部分を離れる際にはどういう理由であっても必ず finally
節が実行されることが保証されている。これは処理を終了する際に何らかの高価な資源(オープン中のファイルやデータベース接続)を解放しなければならない場合に便利である。
FileStream stream = new FileStream ("logfile.txt", FileMode.Create); // C# の例
try {
return ProcessStuff(stream); // 例外を発生する可能性がある
} finally {
stream. Close();
}
この例は非常に一般的であり、C# ではこのための特別な構文がある。
using (FileStream stream = new FileStream ("logfile.txt", FileMode.Create)) {
return ProcessStuff(stream); // 例外を発生する可能性がある
}
上記の例の using
ブロックを離れるとき、コンパイラが自動的に stm
オブジェクトを解放する。Pythonの with
文やRubyの File.open
へのブロック引数も同様の効果がある。
このような言語はいずれも標準の例外を定義し、それらがどのような状況で発生するかを定義している。ユーザーは独自の例外を発生させることもできる(実際、C++ と Python は任意の型の throw と catch が可能)。
特定の throw
にマッチする catch
がない場合、マッチする catch
が見つかるまで入れ子構造を遡り、サブルーチン呼び出しを遡る。メインプログラムまで遡っても対応する catch
がない場合、プログラムは適切なエラーメッセージを出力して停止する。
AppleScript スクリプト言語 は "try
" ブロックにいくつかの情報を提供する。
try
set myNumber to myNumber / 0
on error e number n from f to t partial result pr
if ( e = "Can't divide by zero" ) then display dialog "You must not do that"
end try
継続
[編集]非局所制御フローの比較表
[編集]プログラミング言語 | 条件 | 例外 |
---|---|---|
Ada | No | Yes |
C | No | No |
C++ | No | Yes |
C# | No | Yes |
Common Lisp | Yes | No |
D | No | Yes |
Eiffel | No | Yes |
Haskell | No | Yes |
Java | No | Yes |
Objective-C | No | Yes |
PHP | No | Yes |
PL/I | Yes | No |
Python | No | Yes |
REBOL | Yes | Yes |
Ruby | No | Yes |
Visual Basic .NET | Yes | Yes |
Windows PowerShell | No | Yes |
提案された制御構造
[編集]ドナルド・クヌースは1974年の論文 "Structured Programming with go to Statements"[11] でそれまでの制御構造でカバーされていない2種類の状況を提示し、それを実現する制御構造を例示した。他にも以下に示すような提案がある。
途中にテストのあるループ
[編集]loop loop xxx1 read(char); while test; while not atEndOfFile; xxx2 write(char); repeat; repeat;
もし xxx1 が省略されたら、テストが先頭にあるループとなる。もし xxx2 が省略されたら、テストが最後尾にあるループとなる。while が省略されれば無限ループとなる。このような一種類の制御構造で、必要な多くのタイプのループのパターンを表現できることが示されたことから、以降の言語ではこのような汎用性の高いループ構造を持つものもある(VBのDo...Loopなど)。ありうべき派生としてループ内に複数の while テストを配置することを許すことが考えられるが、その場合は後述の exitwhen の方が適切である。
一般に、任意のループ構造と、条件分岐とbreakを組み合わせて、同様のプログラムを書ける(これは要するに、どんな制御構造も、コンパイルされれば機械語では分岐命令になる、ということと同様のことを言っている)。
while (true) { xxx1 if (not test) break xxx2 }
Adaでは、上記のループ構造(loop-while-repeat)の代替として標準の無限ループ(loop-end loop)内で exit when節を使うことで同様の制御構造を実現できる(後述の exitwhen 文と混同しないよう注意されたい)。
with Ada.Text_IO;
with Ada.Integer_Text_IO;
procedure Print_Squares is
X : Integer;
begin
Read_Data : loop
Ada.Integer_Text_IO.Get(X);
exit Read_Data when X = 0;
Ada.Text IO.Put (X * X);
Ada.Text IO.New_Line;
end loop Read_Data;
end Print_Squares;
ループの命名(この例では Read_Data)は必須ではないが、ループの入れ子で外側のループまで脱出させることができる。
複数早期脱出と入れ子ループからの脱出
[編集]これは 1974年、Zahn が提案した[13]。ここではそれを若干修正したものを示す。
exitwhen EventA or EventB or EventC; xxx exits EventA: actionA EventB: actionB EventC: actionC endexit;
exitwhen は xxx 内で発生しうるイベントを指定するのに使い、イベントはイベント名を文として使用すると発生する。イベントが発生すると対応するアクションが実行され、その後 endexit 後の処理に移る。この制御構造はある状況を識別する部分と、その状況でとるべきアクションを明確に区別することができる。
exitwhen は C++ 言語の try/catch 構造と概念的によく似ているが、サブルーチン呼び出しを超えたり任意の値を渡したりしないので、より効率的と思われる。また、コンパイラは指定されたイベントが全て発生する可能性があり、それらにアクションが対応しているかどうかをチェックできる。
以下の単純な例は2次元配列から特定の要素を取り出すものである。
exitwhen found or missing; for I := 1 to N do for J := 1 to M do if table[I,J] = target then found; missing; exits found: print ("item is in table"); missing: print ("item is not in table"); endexit;
COMEFROM
[編集]Datamation誌(1973年12月)に掲載された記事で[14]、R. Lawrence Clark は COME FROM 文を提案し、面白い例をいくつか提示した。それ自体は「GO TO論争に寄与する」と称したジョークであるが、ジャーゴンファイルの記事が指摘しているように[15]、たとえばFortranのDO文は「そこで指定した行番号のある行からそこに飛ぶ」という一種のCOMEFROMであることなど、制御構造の問題に面白い視点を与えるものではある。setjmp/longjmpと関連させた指摘もある[16]。
COMEFROM文はINTERCALという難解プログラミング言語に実装された(INTERCALの実装者はジャーゴンファイルの編集者でもある)。
脚注
[編集]- ^ procedural programming
- ^ imperative programming
- ^ bit 編集部『bit 単語帳』共立出版、1990年8月15日、122頁。ISBN 4-320-02526-1。
- ^ https://gcc.gnu.org/onlinedocs/gcc/Labels-as-Values.html
- ^ Böhm, Jacopini. "Flow diagrams, turing machines and languages with only two formation rules" Comm. ACM, 9(5):366-371, May 1966.
- ^ Meyer, Bertrand (1991). Eiffel: The Language. Prentice Hall. pp. 129â131
- ^ Predicates and Specification Expressions in "JML Reference Manual"
- ^ “Common Lisp LOOP macro”. 2012年9月8日閲覧。
- ^ for_each. Sgi.com. Retrieved on 2010-11-09.
- ^ Chapter 1. Boost.Foreach. Boost-sandbox.sourceforge.net (2009-12-19). Retrieved on 2010-11-09.
- ^ Knuth, Donald E. "Structured Programming with go to Statements" =ACM Computing Surveys 6(4):261-301, December 1974.
- ^ Dahl & Dijkstra & Hoare, "Structured Programming" Academic Press, 1972.
- ^ Zahn, C. T. "A control statement for natural top-down structured programming" presented at Symposium on Programming Languages, Paris, 1974.
- ^ We don't know where to GOTO if we don't know where we've COME FROM. This (spoof) linguistic innovation lives up to all expectations. By R. Lawrence Clark* From DATAMATION, December, 1973
- ^ http://catb.org/jargon/html/C/COME-FROM.html
- ^ http://www.nurs.or.jp/~sug/soft/super/longjmp.htm#sec36
参考文献
[編集]- Hoare, C. A. R. "Partition: Algorithm 63," "Quicksort: Algorithm 64," and "Find: Algorithm 65." Comm. ACM 4, 321-322, 1961.
関連項目
[編集]- 分岐命令
- goto文
- サブルーチン
- イベントループ
- 再帰呼び出し
- スパゲティプログラム
- 構造化プログラミング
- 関数型プログラミング
- 制御抽象化
- 制御フローグラフ
- コルーチン
- 循環的複雑度
- フローチャート