Kuina-chan

くいなちゃん2018年12月11日


6さいからのプログラミング」第10話は、これまでに紹介しきれなかった知っておくべき細かな機能を補足します。

1識別子の命名

多くのプログラミング言語では大文字「ABC…」と小文字「abc…」とを区別し、例えば変数「X」と変数「x」は別の変数になります。
また多くの言語では、アルファベット「ABC…Zabc…z」と数字「012…9」とアンダースコア「_」を変数名や関数名などに使用できますが、先頭に数字が付けられないなどの決まりがあります。
全角文字が変数名などに使用できる言語もありますが、入力が大変だったりソースコードが読みにくくなることもあるため、通常は半角文字が使われます。 JavaScriptではさらに「$」の記号が命名に使えます。

2改行とスペース

多くの言語では、図2-1のようにソースコード上で比較的自由に改行したりスペースを挟んだりできます。
  1. #include      <stdio.h>
  2.   int
  3. main(
  4.   void){
  5.   printf
  6. ("Hello, world!\n"
  7. );return
  8.  
  9. 0
  10.   ;}
図2-1: line_feed_and_space.c
ただし無秩序に改行やスペースを入れるとソースコードが読みにくくなるため、一定のルールに従って書かれます。 また改行はスペース1つ分として解釈されることが多く、「if」の「i」と「f」の間で改行することはできません。
改行やスペースのルールや、大文字小文字のルールなどを合わせたものを「コーディング規約きやく」と呼びます。 コーディング規約は世界に1つだけ存在しているわけではなく、開発チームごとに用意されます。 一人で作るプログラムでは、自分が読み書きしやすいコーディング規約を定めておくと良いでしょう。

3コメント

ソースコード中に、日本語や英語などで好きなメッセージを書き込むことができます。 これは「コメント」と呼ばれ、プログラムの動作には影響しません。 C言語では「/*」と「*/」で囲んだ部分がコメントとして、コンパイル時に除外されます(図3-1)。
  1. #include <stdio.h>
  2.  
  3. /* メイン関数 */
  4. int main(void)
  5. {
  6.   /*
  7.   printf("Hello, world!\n");
  8.   */
  9.   return 0;
  10. }
図3-1: comment.c
7行目のように、プログラムの一部の動作を無効化したいときにもコメントが役立ちます。
C++やC#では、この「/*」「*/」形式のコメントの他に、「//」形式のコメントも書けます。 これは「//」を書いた箇所からその行の終わりまでをコメント扱いします。 図3-2はC#での例です。
  1. class HelloWorld
  2. {
  3.   // メイン関数
  4.   static void Main(string[] args)
  5.   {
  6.     // System.Console.WriteLine("Hello, world!");
  7.   }
  8. }
図3-2: comment.cs

416進数と8進数

ほとんどの言語では16進数の数値がソースコード中に書けます。 C言語やJavaなどで16進数を書くには、頭に「0x」を付けます。 例えばA0は「0xA0」です。 またあまり使いませんが、C言語では8進数の場合は頭に「0」を付けて、「037」のように書けます。 注意が必要なのは、10進数のつもりで「0123」と書くと8進数になってしまう点です。

5指数表記

多くの言語では「a×(10のb乗)」の形の小数が書けます。 これを「指数表記しすうひょうき」といい、非常に大きな数や小さな数を書くときに使われます。 例えばC言語やJavaなどでは「6.02×(10の23乗)」は「6.02e23」と書け、「1×(10の-9乗)」は「1e-9」のように書けます。 この「e」は指数部を意味する「exponent」に由来します。

6初期化

C言語やJavaなどで「int a; int b; int c;」のように同じ型の変数を複数作成するときは、「int a, b, c;」のようにまとめて作成できます。
変数は、作成と同時に値を代入することもできます。 例えばC言語で「int n;」を作成して「n=2;」と代入する処理は、「int n=2;」のようにまとめることができます。 これらはJava、C#など、多くの言語で同様です。
またC言語の配列は、配列作成時にまとめて全要素に代入することができます(図6-1)。
  1. int main(void)
  2. {
  3.   char a[4] = { 'A', 'B', 'C', '\0' };
  4.   char b[] = "ABC";
  5.   return 0;
  6. }
図6-1: initialization.c
このとき、配列a、bは両方とも「'A'」「'B'」「'C'」「'\0'」の4要素となります。 このように「{}」とカンマ区切りで各要素の値が指定でき、char型の配列に限っては4行目のように{}の代わりに文字列を指定することもできます。 4行目のbのように要素数を省略すると、指定した値が丁度収まる要素数(この場合は4)に自動的になります。
C言語では構造体も同様に、構造体作成時にまとめて代入できます(図6-2)。
  1. struct person
  2. {
  3.   char name[32];
  4.   int age;
  5.   int array[3];
  6. };
  7.  
  8. int main(void)
  9. {
  10.   struct person x = { "Kuina-chan", 6, { 1, 2, 3 } };
  11.   return 0;
  12. }
図6-2: initialization2.c
ちなみに作成しただけで一度も代入していない変数には、0ではなく、以前その領域が使われていた際のメインメモリ上のデータの残骸が入っていることがありますので、変数を作成した後には代入を忘れないようにしましょう。

7再帰呼び出し

関数がその関数自身を呼び出すことを「再帰さいき」といいます。
再帰は数学の漸化式と相性が良く、例えば「」という数学の漸化式はC言語で図7-1のように書けます。
  1. #include <stdio.h>
  2.  
  3. int a(int n)
  4. {
  5.   if (n == 0)
  6.   {
  7.     return 3;
  8.   }
  9.   else
  10.   {
  11.     return a(n - 1) + 2;
  12.   }
  13. }
  14.  
  15. int main(void)
  16. {
  17.   printf("%d\n", a(10));
  18.   return 0;
  19. }
図7-1: recursion.c
3~13行目で、「」がそのまま書けていることが何となく分かると思います。
このプログラムの関数aは、a(0)が呼ばれた場合は、7行目により3を返します。 a(1)が呼ばれた場合は、11行目により「a(1-1)+2」を返そうとしますが、ここで自分自身の関数a(0)を呼び出します。 a(0)は3を返すため、「a(1-1)+2」は「3+2」、つまり「5」です。
同様にa(10)が呼ばれた場合を追ってみると、11行目によりa(9)、a(8)、a(7)…と順に呼ばれ、a(0)にたどり着いて「3」を返したのち、呼び出された順序を逆にさかのぼって値を返していきます。
再帰は慣れるまでは扱いにくいですが、データ探索の処理などで多用します。
またHaskellなどの関数型言語では、反復処理が基本的に再帰で書かれるなど、再帰はよく使われます。 先ほどの「」という漸化式をHaskellで書くと、図7-2のようになります。
  1. a 0 = 3
  2. a n = a (n - 1) + 2
  3. main = print $ a 10
図7-2: recursion.hs
1~2行目に漸化式がほぼそのまま書かれており、3行目でa(10)の値を画面に表示する旨を書いています。 C言語に比べてシンプルなソースコードになっています。

8ヒープ領域

ヒープ領域とは、プログラムの実行中に自由に確保や解放ができるメインメモリの領域でした。 C言語でこの領域を扱うには、確保に「mallocマロック」関数、解放に「freeフリー」関数を使います。 これらの関数は、stdlib.h内に存在します(図8-1)。
  1. #include <stdlib.h>
  2.  
  3. int main(void)
  4. {
  5.   int* p;
  6.   p = malloc(sizeof(int) * 3);
  7.   p[0] = 100;
  8.   p[1] = 200;
  9.   p[2] = 300;
  10.   free(p);
  11.   return 0;
  12. }
図8-1: heap.c
malloc関数は、引数に渡したサイズ分をヒープ領域から確保し、そのアドレスを返します。 mallocは「Memory ALLOCation」の略です。
sizeofサイズオブ」演算子は、型のサイズをバイト単位で返す演算子です。 「sizeof(int)」と書くと、int型変数1つ分のサイズが返ります。 6行目では「sizeof(int)*3」としているので、丁度「int p[3];」の配列に相当する領域が確保されます。 よって7~9行目のように読み書きできます。
最後に10行目の「free(p);」で、ヒープから確保した領域を解放しています。 これを書かないと、いつまでも解放されない領域がメインメモリに残り続ける問題「メモリリーク」が発生します。

9関数ポインタ

C言語には、関数のアドレスを入れられるポインタが存在します。 これを「関数かんすうポインタ」といいます。 関数の名前だけを書くと、その関数が静的領域に配置されたときの、アドレスとなります(図9-1)。
  1. #include <stdio.h>
  2. #include <math.h>
  3.  
  4. int main(void)
  5. {
  6.   double(*p)(double);
  7.   p = sin;
  8.   printf("%f\n", p(0.0));
  9.   return 0;
  10. }
図9-1: function_pointer.c
7行目に「sin」とだけ書いていますが、こうするとsin関数のアドレスが返り、ポインタpに入ります。 sin関数は「double sin(double x){~}」として、math.h内で作られていると以前説明しましたが、このsin関数のアドレスを入れる関数ポインタの型は「double(*)(double)」型となります。 そしてこの型のポインタをpという名前で作成するには、6行目のように「double(*p)(double);」と書きます。
関数ポインタに関数のアドレスを入れたものは、その関数と同様に扱えます。 8行目に「p(0.0)」と書かれていますが、これは元のsin関数を「sin(0.0)」と呼び出すことと等しいです。
関数ポインタの使い道としては、例えば関数ポインタを関数の引数にすることで、「渡された関数を10回呼び出す関数」のような関数が作れます(図9-2)。
  1. #include <stdio.h>
  2. #include <math.h>
  3.  
  4. void call_ten_times(double(*p)(double))
  5. {
  6.   int i;
  7.   for (i = 0; i < 10; i++)
  8.   {
  9.     printf("%f\n", p((double)i));
  10.   }
  11. }
  12.  
  13. int main(void)
  14. {
  15.   call_ten_times(sin);
  16.   return 0;
  17. }
図9-2: call_ten_times.c
この例では、call_ten_times関数の引数pに渡された関数が10回呼び出されます。 7行目のfor文でiを0~9まで数えながら10回繰り返し、9行目でiの値を関数ポインタpに渡して結果を画面に表示しています。 15行目でこのcall_ten_times関数にsin関数を渡していますので、このプログラムはsin(0.0)、sin(1.0)、sin(2.0)、…、sin(9.0)の結果を表示します。
今回は、今までに紹介しきれなかった細かな機能を補足しました。 次回は、目的を達成するためにはどのようにプログラムで書けば良いかについて解説します。
1544538783jaf