(23.1) 変数名には、a, b, x, y のような一文字はもちろん、sum, total のように二文字以上からなる文字列も使える。 英語の単語でなくてももちろん可。 x11 のように数字がはいっても構わないが、 数字で始まる 1x などは不可。 (使おうとしたらどうなるか?)
(23.2) x1, x2, x3 とあったら人間は関連のある変数だろうと想像するが、 コンパイラにとっては全く無関係な、ただの三つの変数である。
(23.3) 「配列」とは、 0 からある自然数までの整数で添字づけられた有限個の変数 x[0], x[1], x[2], ..., x[N-1] を一斉に定義し、一括して扱うものである。 K&R2 では §1.6 を参照。
(23.4) 配列を使った自明なプログラムの例を示す。
#include <stdio.h> int a[10]; /* 配列の宣言。これで a[0], ..., a[9] が使える */ main() { int i, n; for (i = 0; i < 10; i++) { /* 配列の要素に代入 */ a[i] = i * i; } for (i = 0; i < 10; i++) { /* 配列の要素を印字(=出力) */ printf("%d\n", a[i]); } }
(23.5) 「配列の宣言」の行。 こう書くと a[0] から始まる十個の int 型変数が使える。 すなわち、a[0], a[1], a[2], ..., a[9] が使える。 a[10] は使えない。うっかり使ってしまいやすいので注意。
(23.6) 「配列の宣言は main() の中カッコの外で行ない、 ほかの変数の宣言は main() の中カッコの中で行なう」 という規則があるわけではないが、 とある事情により、結果として、この授業ではこうすることが多い。 だからこれが規則だと思っておけばよい。 正確なところは K&R2 §1.10, §4.3 参照。
(23.7) 上のプログラムは、配列の大きさを N として、 次のように書くほうがよい。 配列の大きさを変えることが容易になるからである。 実際に変えて実行してみよ。
#include <stdio.h> #define N 10 int a[N]; /* 配列の宣言。これで a[0], ..., a[N-1] が使える */ main() { int i, n; for (i = 0; i < N; i++) { /* 配列の要素に代入 */ a[i] = i * i; } for (i = 0; i < N; i++) { /* 配列の要素を印字(=出力) */ printf("%d\n", a[i]); } }
(23.8) 前に (20.4) で書いたプログラムを配列を使って書き直してみた。 今回は「階乗の計算」「各項の計算」「部分和の計算」 に分けて計算・出力している。 前のプログラムで見せたように配列を用いなくても書けるのだが、 このほうが書きやすいなら、いまはこれでもよい。 (階乗を double 型に入れているのは、 あとの計算が double 型であることと、 int 型よりも double 型のほうが格納できる整数の範囲が広いからである。 このような判断は、いまは自分でできなくても構わない。)
#include <stdio.h> /* 自然対数の底 e の計算、配列版 */ #define N 23 double f[N]; /* 階乗の値を入れる配列 */ double a[N]; /* 各項の値を入れる配列 */ double s[N]; /* 部分和の値を入れる配列 */ main() { int n; f[0] = 1; /* 階乗の計算 */ for (n = 1; n < N; n++) { f[n] = f[n-1] * n; } for (n = 1; n < N; n++) { printf("%2d の階乗は %.0f です.\n", n, f[n]); } for (n = 1; n < N; n++) { /* 各項の計算 */ a[n] = 1 / f[n]; } for (n = 1; n < N; n++) { printf("第 %2d 項は %.53f です.\n", n, a[n]); } s[0] = 1; /* 部分和の計算 */ for (n = 1; n < N; n++) { s[n] = s[n-1] + a[n]; printf("第 %2d 項までの和は %.53f です.\n", n, s[n]); } }
(23.9) ※ (センターの linux での)上のプログラムの出力を見ると、 第 18 項から第 22 項までの値は 0 でない。 しかし、「第 17 項までの和」以降はすべて同じ値になる。 すなわち、0 でない値を加えているにもかかわらず値が変わらない。 近似計算なので、こういうことも起こるのである。
(23.10) 次のプログラムは、 いままでのと比べると意味のあるプログラムである。
#include <stdio.h> /* エラトステネスのふるいで 10000 以下の素数を全て求める */ #define KESITENAI 0 /* 消してない */ #define KESITA 1 /* 消した */ int a[10000+1]; /* 配列 a[0], ... , a[10000] を宣言 */ main() { int n, i; for (n = 2; n <= 10000; n++) { /* 配列の初期化 */ a[n] = KESITENAI; } for (n = 2; n <= 100; n++) { /* ふるいにかける */ for (i = 2; n * i <= 10000; i++) { /* n の倍数のうち */ a[n*i] = KESITA; /* 10000 以下のものを消す */ } } for (n = 2; n <= 10000; n++) { /* 結果の出力 */ if (a[n] == KESITENAI) { printf("%d は素数.\n", n); } else { printf("%d は合成数.\n", n); } } }
(23.11) ※ C言語では、配列の要素 a[i] の値を調べたり a[i] に代入したりするとき、 添字 i が正しい範囲にあるかどうかチェックしない。 もしも上のプログラムで誤って a[10001] に代入したとしても、 コンパイル時や実行時にエラーメッセージが出るとは限らない。 しかし、 プログラムが暴走したりおかしな結果が出たりすることは起こり得るので、 絶対にそのようなプログラムを書いてはならない。
(23.12) ※ 出力が大量になり、最初の部分がよく見えない場合などは、 「./a.out」の代わりに「./a.out | more」とする。 一画面分だけ出力されると「--続ける--」が出て止まるので、 スペースバーを押せば次の一画面分に進む。 Enter なら一行分だけ進む。 次に進まないでプロンプトに戻りたい場合は「q」または Ctrl+C を押す。
(23.13) ※ 上のプログラムは、第二のループで合成数の倍数も消しているので、 まだまだ効率をあげる余地がある。 しかし、出力が始まったときには第二のループは終わっているわけで、 このくらいの計算はコンピュータにとってはあっという間である。
(23.14) 自然数 a, b の最大公約数 d = gcd(a, b) の求め方や、 ax + by = d を満たす整数 x, y の求め方は代数学で習ったと思う。 a が 19, b が 5 の場合、次のように上から表を埋めてゆく。
i a[i] b[i] r[i] q[i] x[i] y[i] 0: 19 5 4 3 1: 5 4 1 1 2: 4 1 0 4 3: 1 0 0 0 1 0ここで r は余り、q は商である。 x, y は、一番下はすぐわかるので、埋めておいた。 ここを、下から上へと埋めてゆきたい。
ai+1 = bi, bi+1 = ri = ai - biqi を ai+1xi+1 + bi+1yi+1 = d に代入すると bixi+1 + (ai - biqi)yi+1 = d となる。 これを ai , bi について整理して aiyi+1 + bi(xi+1 - qiyi+1) = d を得る。 よって xi = yi+1, yi = xi+1 - qiyi+1 でよい。 (ほかの選び方もある。)
これに従って、x[i], y[i] を下から埋めてゆく。
i a[i] b[i] r[i] q[i] x[i] y[i] 0: 19 5 4 3 -1 4 1: 5 4 1 1 1 -1 2: 4 1 0 4 0 1 3: 1 0 0 0 1 0のようになる。x = -1, y = 4 が答えである。
#include <stdio.h> #define N 50 int a[N]; int b[N]; int q[N]; int r[N]; int x[N]; int y[N]; main() { int i; scanf("%d", &a[0]); scanf("%d", &b[0]); for (i = 0; b[i] != 0; i++) { q[i] = a[i] / b[i]; r[i] = a[i] % b[i]; a[i+1] = b[i]; b[i+1] = r[i]; } a[i+1] = 0; x[i] = 1; y[i] = 0; for (i-- ; i >= 0; i--) { x[i] = y[i+1]; y[i] = x[i+1] - q[i] * y[i+1]; } printf(" i a[i] b[i] r[i] q[i] x[i] y[i]\n"); for (i = 0; a[i] != 0; i++) { printf("%2d:%8d %8d ", i, a[i], b[i]); printf("%8d %8d ", r[i], q[i]); printf("%8d %8d\n", x[i], y[i]); } }
(23.15) ※ N を 50 にしたのは、 このコンパイラが扱える int 型の範囲ではこれで(たぶん)十分であることを別に確かめたからである。
(23.16) ※ C言語の規格では、負の数を含む割り算の余りは一通りに決まってはいない。 -10 % 3 は -1 かも知れないし、2 かも知れないのである。 よって、上のプログラムを負の数も扱えるように改良することは (ここでは)やめておいた。
(24.1) 思いつくままに書いたので、難易度はまちまちである。 いままでの知識で書けるかどうかよく確かめていないものもあるが、 とりあえずは最初の Fibonacci 数列に取り組んでみよ。 なお、配列が出てくるものは、 大きさを N とし、N は #define を利用して簡単に変更できるようにするとよい。
(24.2)