Home English

Kuina-chan

くいなちゃんDec 15, 2017


6さいからのプログラミング」第8話では、メインメモリのアドレスを直接やりとりしたり走査できる「ポインタ」について解説します。

変数のアドレス

第7話では、同じ処理を何度も行うときに、選択処理や反復処理を用いることでシンプルに書けることを示しました。 今回は、たくさんの変数をシンプルに扱う方法について解説します。
C言語で30個の変数を扱うとき、例えばvariables.cのように書くことができます。
int main(void)
{
  int a;
  int b;
  int c;
  int d;
  int e;
  int f;
  int g;
  int h;
  int i;
  int j;
  int k;
  int l;
  int m;
  int n;
  int o;
  int p;
  int q;
  int r;
  int s;
  int t;
  int u;
  int v;
  int w;
  int x;
  int y;
  int z;
  int aa;
  int ab;
  int ac;
  int ad;
  
  return 0;
}
variables.c
この場合、各変数に5を代入したいときには「a=5;b=5;c=5;…」と何度も書く必要があります。 しかし、仮にこれらの変数がメインメモリ上に連続して存在していたなら、変数のアドレスのように、メインメモリのアドレスを変数のサイズ分だけ進めながら書き換えることで、単純に代入できそうです。
変数のアドレス
変数のアドレス
例えばこの例では、変数aが存在しているアドレスがA0番地なので、A0番地から順に、int型のサイズである4バイトずつ移動しながら値を書き換えると、すべての変数が書き換えられることになります。 これを実現するものとして、C言語には「ポインタ」という機能が用意されています。

ポインタ

「ポインタ」とは、「変数のアドレスを入れる変数」です。 そして何の型の変数のアドレスを入れるかによってポインタには種類があり、その変数の型のうしろに「*」を付けて表現します。 例えばint型の変数のアドレスを入れるポインタの型は「int*」型です。
ポインタは変数の一種に過ぎないので、プログラム上でも変数と同じように扱います。 例えば、「int*」型のポインタを「ptr」という名前で作成するには、「int* ptr;」と書きます(pointer.c)。
#include <stdio.h>
  
int main(void)
{
  int a;
  int* ptr;
  
  a = 5;
  ptr = &a;
  *ptr = 3;
  printf("%d\n", a);
  
  return 0;
}
pointer.c
9行目には変数aの前に「&」が付いていますが、これはその変数のアドレスを返すという演算子です。 つまり「ptr=&a;」で、ポインタptrには変数aのアドレスが入ります。 ここでは仮に、そのアドレスをA0番地としておきましょう。 ptrにはA0が入ります。
10行目にはポインタの前に「*」が付いていますが、これは6行目の「int* ptr;」の「*」とは全くの別物で、ポインタに入っているアドレスの場所に存在している変数を返す演算子です。 つまり、ptrには現在A0が入っていますので、「*ptr」とはA0番地に存在している変数、すなわち変数aとなります。 よって「*ptr=3;」とは「a=3;」に等しく、変数aの値は3に書き換わります(ポインタ)。
ポインタ
ポインタ
図では変数「int a」に「5」が入っており、そのアドレス「A0」がポインタ「int* ptr」に入っています。
ややこしいですが、つまりポインタの前に付ける「*」演算子は、ポインタに入っているアドレスの先を変数に見立てて、値を読み書きできるというものです。 例えばメインメモリの03番地の値を書き換えたい場合は、「ptr=3;」として、「*ptr=1234;」とすれば書き換わります。 ただしこの場合、メインメモリの03番地が読み書きできる領域とは限らないため、おそらくエラーが発生します。 必ず読み書きできるメインメモリの領域を操作しなければなりません。

ポインタのポインタ

さて、ポインタとは変数のアドレスを入れる変数でした。 ここで、ポインタ自体も変数の一種であることを踏まえると、「ポインタのアドレスを入れるポインタ」というものを考えることができます(pointer2.c)。
#include <stdio.h>
  
int main(void)
{
  int a;
  int* ptr;
  int** ptrptr;
  
  a = 5;
  ptr = &a;
  ptrptr = &ptr;
  **ptrptr = 3;
  printf("%d\n", a);
  
  return 0;
}
pointer2.c
このプログラムでは「ptrptr」が「ポインタのポインタ」になっています。 「int*」型のポインタのアドレスを入れるポインタを作りたい場合には、int*に「*」を付けた「int**」が型となります。
この「ptrptr」に対し「*ptrptr」と書くと、ptrptrに入っているアドレス先の変数を指しますので、「ptr」のことを意味します。 更にこれに*演算子を付けて「**ptrptr」とすると、「*ptr」のこと、つまりptrに入っているアドレス先の変数を指しますので、「a」のことを意味します。 結果的に「**ptrptr=3;」とは「a=3;」と同等で、aに3が入ります(ポインタのポインタ)。
ポインタのポインタ
ポインタのポインタ
ポインタを使うと30個もの変数をシンプルに扱えたことと同様で、ポインタのポインタを使うと30個ものポインタを扱うときにシンプルに書けます。

ポインタ演算

ポインタに対する演算として、「+」と「-」の演算子があります。 これは「ポインタ+整数」もしくは「整数+ポインタ」のように書き、ポインタに入っているアドレスから指定した整数個分の変数のサイズだけ進めたり戻したりすることができます。 例えば、int*型のポインタpにアドレスA0が入っているとき、「p+1」はA0からint型1つ分だけ進めたアドレスになります。 つまりint型が4バイトの環境では「p+1」はA4です(ポインタの加減算)。
ポインタの加減算
ポインタの加減算
このように「*p」「*(p+1)」「*(p+2)」のように1ずつ増やしていくと、pに入っているアドレスA0から出発して、メインメモリ上の連続したint型の変数を順に読み書きすることができます。

void*

ポインタの一種として「void*」型があります。 これは、void型の変数のアドレスを入れるポインタ、という意味ではなく、「単にアドレスを入れるだけのポインタ」だと捉えてください。 void*は単にアドレスを入れるだけなので、例えばvoid*型のポインタpがあったとき、「p+1」「p-1」といったアドレスの演算はできません。 また、「*p」としてアドレス先にアクセスすることもできません。 void*型は主に、どんな型の変数のアドレスが入るか判らないときに使われます。

NULL

アドレスの0番地は特別で、通常どの変数のアドレスでもないことを意味します。 stdio.hなどの中では0番地のことを「NULLヌル」と名付けており、ポインタがどの変数のアドレスも入れていないことを示したい場合にNULLを入れます(null.c)。
#include <stdio.h>
  
int main(void)
{
  void* p;
  p = NULL;
  return 0;
}
null.c

配列

さてここまでで、ポインタを使うとメインメモリ上に連続した変数を読み書きできると説明しましたが、メインメモリ上に連続させて変数を作るためには、C言語では「型 変数名[要素数];」のように書きます。 例えば、「int a[30];」と書くと、メインメモリ上にint型変数30個分の連続した領域が確保されます。
このとき「a」の名前は、ポインタのように扱えます。 つまり「*a」「*(a+1)」「*(a+2)」…「*(a+29)」と書け、30個分の変数の値を読み書きできます。 ただしポインタとは異なり、「a=0;」のような、aが保持するアドレスそのものを書き換える操作はできません。 このように、連続した領域を確保しその先頭アドレスを保持するaのことを、「配列はいれつ」といいます。

糖衣構文

「int a[30];」として作成した配列aに対し「*a」「*(a+2)」のようにアクセスできますが、C言語にはさらにこれを見やすくする仕組みがあり、「*a」のことは「a[0]」、「*(a+2)」のことは「a[2]」と書けます(配列の糖衣構文)。
配列の糖衣構文
配列の糖衣構文
これは、配列を「int a[30];」と作成すると、あたかもa[0]~a[29]の30個のint型変数が作られたかのように見せるもので、人間にとって直観的な記法となっています。 この記法は配列だけでなくポインタにも同様に使え、ポインタpに対し「p[5]」などと書けます。 この「a[b]」の記法はコンパイル時に「*(a+b)」の形に自動的に置き換えられます。
それでは、30個のint型変数を作成してそれぞれに5を代入するプログラムを書いてみましょう(array.c)。
int main(void)
{
  int a[30];
  int i;
  for (i = 0; i < 30; i++)
  {
    a[i] = 5;
  }
  return 0;
}
array.c
for文と組み合わせることで、とてもシンプルに書けました。

動的な配列

C/C++以外の言語の配列では、C/C++とは仕組みが異なることがあります。 例えばJavaやC#の配列は、配列の作成時に要素数が固定されません。 実体はヒープ領域に作られ、配列は要素数と実体への参照を持っています(動的な配列)。
動的な配列
動的な配列
図では「01 23 45 …」という値が配列に入っていますが、JavaやC#ではこれらはヒープ領域に置かれています。 ヒープ領域はプログラムの実行中に自由に確保や解放ができるので、要素数を柔軟に変えることができるという仕組みです。

連想配列

PHPなどのスクリプト言語の配列は、これまでとは異なる「連想配列れんそうはいれつ」と呼ばれるものです。 連想配列は、メインメモリ上に連続した領域があるのではなく、要素番号と値との対応が保持される仕組みです。 例えば配列aに対し、a[2]に5を入れたり、a[100]に8を入れた段階で、メインメモリ上にこれらの対応関係が作られます(連想配列)。
連想配列
連想配列
このため、a[100]を使ってもメインメモリ上にたくさんの領域が確保される心配がない上に、「a["apple"]」のように要素番号に整数以外の値を用いることもできます。

文字列

最後に、文字列について解説します。 文字列とは、基本的には文字型の配列です。 JavaやC#の配列は先ほど紹介したように柔軟なため、JavaやC#における文字列も自由に長さを変えられる領域を持った扱いやすいものになっています。 一方、C言語の文字列は通常、領域の大きさが固定されています(文字列)。
文字列
文字列
C言語の配列は要素数を保持しないので、メインメモリ上のどこまでが文字列なのかが判りません。 そこで文字列の最後に'\0'という特殊な文字を付けることで文字列の終端を表すことにしています。 '\0'とは、文字コードが0の文字です。
C言語での文字列の扱いを具体的なプログラムで示すと、str.cのようになります。
#include <stdio.h>
#include <string.h>
  
int main(void)
{
  char s[6];
  strcpy(s, "Hello");
  printf("%s\n", s);
  s[1] = 'i';
  s[2] = '\0';
  printf("%s\n", s);
  return 0;
}
str.c
プログラム中に「"Hello"」のように文字列リテラルを書くと、静的領域にあらかじめ埋め込まれた文字列の先頭アドレスになります(文字列リテラル)。
文字列リテラル
文字列リテラル
図の例では、静的領域の30番地から「Hello」の値が格納されているため、ソースコード中に"Hello"と書くと30番地が返ります。
「strcpy関数」は、第2引数の文字列を第1引数にコピーする関数で、「string.h」の中に存在します。 よって7行目の「strcpy(s,"Hello");」を実行すると、s[0]~s[5]にはそれぞれ「'H'」「'e'」「'l'」「'l'」「'o'」「'\0'」の6文字が入ります。 終端文字'\0'があるので、配列sの領域は1文字分多めに確保しておく必要があります。
printf関数の「%s」は、渡されたアドレスから終端文字までの文字列を画面に表示する機能です。 よって8行目の「printf("%s\n",s);」で画面には「Hello」が表示されます。
9行目の「s[1]='i';」により、「Hello\0」は「Hillo\0」に書き換わります。 同様に、10行目の「s[2]='\0';」で「Hi\0llo\0」になります。
2回目のprintfでは、sの中身である「Hi\0llo\0」を表示しようとしますが、この文字列の途中には終端文字があるため、画面には「Hi」とだけ表示され、残りの「llo」は無視されます。
今回は、メインメモリを自在に操作するための「ポインタ」や、たくさんの変数をシンプルに扱える「配列」、そして「文字列」の仕組みについて説明しました。 次回は、さらにプログラムをシンプルに書くために、型そのものを自分で作成する方法について解説します。
1513346830ja