C言語の基礎 - 配列・メモリ領域
メモリ領域を動的に確保する
C言語で、メモリ領域を動的に確保するには、stdlib.hのmalloc関数を使用する。
#include <stdlib.h>
void *malloc(size_t size);
malloc関数はsizeバイト分のメモリ領域を確保して、そのメモリ領域へのポインタを返す関数である。
なお、メモリ領域の確保に失敗した場合は、NULLを返す。
malloc関数で確保したメモリ領域は、free関数で解放する。
malloc関数を使用したメモリの動的確保は、多くの場合、以下のように実装する。
以下の例として、int型のデータが10個入るサイズのメモリ領域を確保している。
int *ptr = nullptr;
/* int型のデータが1 個入るサイズのメモリ領域を確保 */
if ((ptr = (int *)malloc(sizeof(int) * 10)) == NULL)
{
/* エラー処理 */
}
/* 何らかの処理 */
/* メモリ領域の解放 */
free(ptr);
以下の例では、malloc関数を使用して、メモリ領域を動的に確保している。
#include <stdio.h>
#include <stdlib.h>
int main(void)
{
int *ptr = nullptr, *work = nullptr;
/* int 型のデータが 10 個入るサイズのメモリ領域を確保 */
if ((ptr = (int *)malloc(sizeof(int) * 10)) == NULL)
{
fprintf(stderr, "メモリ領域を確保に失敗しました.\n");
exit(EXIT_FAILURE);
}
work = ptr;
for ( int i = 0; i < 10; i++)
{
work[i] = i;
printf("%d ", work[i]);
}
printf("\n");
/* メモリ領域の解放 */
free(ptr);
return EXIT_SUCCESS;
}
メモリ領域を確保して、その領域を0で初期化する
C言語で、メモリ領域を動的に確保して、その領域を0で初期化するには、stdlib.hのcalloc関数を使用する。
#include <stdlib.h>
void *calloc(size_t nmemb, size_t size);
calloc関数は、sizeバイトの大きさを持つオブジェクトがnmemb個分入るメモリ領域を確保して、その領域のすべてのビットを0で初期化する関数である。
メモリ領域の確保に成功した場合は、そのメモリ領域へのポインタを返し、失敗した場合はNULLを返す。
また、calloc関数で確保したメモリ領域は、free関数で解放する。
以下の例では、calloc関数を使用してメモリ領域を動的に確保している。
#include <stdio.h>
#include <stdlib.h>
int main(void)
{
double *ptr = nullptr;
int nmemb = 3;
/* double サイズ 3 個分の領域を確保 */
if ((ptr = (double *)calloc(nmemb, sizeof(double))) == NULL)
{
fprintf(stderr, "メモリ領域を確保に失敗しました.\n");
return EXIT_FAILURE;
}
/* 適当に値を入力 */
ptr[0] = 1950;
/* メモリ領域の内容を表示する */
for ( int i = 0; i < nmemb; i++ )
{
printf(" %.1f", ptr[i]);
}
printf("\n");
/* メモリ領域の解放 */
free(ptr);
return EXIT_SUCCESS;
}
確保したメモリ領域のサイズを変更する
malloc関数やcalloc関数等で確保したメモリ領域のサイズを変更するには、stdlib.hのrealloc関数を使用する。
#include <stdlib.h>
void *realloc(void *ptr, size_t size);
realloc関数は、まず、ptrが指すメモリ領域を解放した後、大きさがsizeである新しいメモリ領域を確保する関数である。
新しいメモリ領域は、可能な限り開放する前の古いメモリ領域の内容を引き継ぐが、古いメモリ領域の大きさを超えた部分の値は不定となる。
また、realloc関数は、メモリ領域の確保に成功した場合は、そのメモリ領域へのポインタを返し、失敗した場合はNULLを返す。
以下の例では、realloc関数を使用して、確保したメモリ領域のサイズを変更している。
#include <stdio.h>
#include <stdlib.h>
int main(void)
{
int *ptr = nullptr;
/* intサイズ10個分のメモリ領域を確保 */
if ((ptr = (int *)malloc(sizeof(int) * 10)) == NULL )
{
fprintf(stderr, "メモリ領域が確保できません.\n");
return EXIT_FAILURE;
}
/* 確保したメモリ領域に値を格納して表示 */
printf("realloc 関数使用前.\n");
for ( int i = 0; i < 10; i++ )
{
ptr[i] = i;
printf(" %d", ptr[i]);
}
printf("\n");
/* intサイズ10個分から15個分のメモリ領域に拡張する */
if ((ptr = (int *)realloc(ptr, sizeof(int) * 15)) == NULL)
{
fprintf(stderr, "メモリが確保できません。\n");
return EXIT_FAILURE;
}
/* 確保したメモリ領域に値を格納して表示 */
printf("\nrealloc関数使用後.\n");
for ( int i = 0; i < 15; i++ )
{
ptr[i] = i;
printf(" %d", ptr[i]);
}
printf("\n");
/* メモリ領域の解放 */
free(ptr);
return EXIT_SUCCESS;
}
メモリ領域を解放する
malloc関数やcalloc関数、realloc関数で確保したメモリ領域を解放するには、stdlib.hのfree関数を使用する。
free関数は、ptrが指すメモリ領域を解放する関数である。
#include <stdlib.h>
void free (void *ptr);
以下の例として、free関数を使用してメモリ領域を開放している。
#include <stdio.h>
#include <stdlib.h>
int main(void)
{
int *ptr = nullptr;
/* int サイズのメモリ領域を確保 */
if ((ptr = (int *)malloc(sizeof(int))) == NULL)
{
fprintf(stderr, "メモリ領域の確保に失敗しました.\n");
exit(EXIT_FAILURE);
}
/* メモリ領域の解放 */
free(ptr);
printf("メモリ領域を解放しました.\n");
return EXIT_SUCCESS;
}
配列やメモリ領域の内容を初期化する
配列を初期化する場合は、以下のように配列の宣言時に初期化子を使用する。
int a1[] = {1, 2, 3}; /* それぞれ異なる値で初期化 */
int a2[256] = {0}; /* 同一の値で初期化 */
また、malloc関数等を使用して動的に確保したメモリ領域を初期化する場合や、配列を宣言時以外に初期化する場合は、
反復処理等を使用して、個々の要素に対して個別に値を格納していく必要がある。
文字配列(文字列)を初期化する場合は、memset関数を使用する。これについては、ページの"文字列を初期化する"を参照する。
配列やメモリ領域の内容をコピーする
C言語で、配列やメモリ領域の内容をコピーするには、string.hのmemcpy関数またはmemmove関数を使用する。
memcpy関数は、s2が指すオブジェクトからs1が指すオブジェクトにn文字コピーする関数である。
なお、領域の重なり合うオブジェクト間でコピーが行われるときの動作は未定義である。
memmove関数は、memcpy関数と似ているが、領域の重なり合うオブジェクト間でコピーを行う場合でも正しく動作する。
#include <string.h>
void *memcpy(void * restrict s1, const void * restrict s2, size_t n);
void *memmove(void *s1, const void *s2, size_t n);
以下の例では、memcpy関数を使用して、int型の配列をコピーしている。
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define N 8
int main(void)
{
int x1[] = {1, 2, 3, 4, 5, 6, 7, 8};
int x2[N] = {0};
/* メモリ領域内の内容をコピー */
memcpy(x2, x1, sizeof(int) * N);
/* x1 と x2 の各要素を表示 */
for ( int i = 0; i < N; i++ )
{
printf("x1[%d] : %d, x2[%d] : %d\n", i, x1[i], i, x2[i]);
}
return EXIT_SUCCESS;
}
配列やメモリ領域の内容を比較する
C言語で、配列やメモリ領域の内容を比較するには、string.hのmemcmp関数を使用する。
memcmp関数は、s1が指すオブジェクトの始めのn文字と、s2が指すオブジェクトの始めのn文字を比較する関数である。
memcmp関数は、比較するオブジェクトが同じなら0、s1 > s2なら正の整数、s1 < s2なら負の整数を返す。
#include <string.h>
int memcmp (const void *s1, const void *s2, size_t n);
以下の例では、memcmp関数を使用して、int型の配列を比較している。
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define N 8
int main(void)
{
int x1[] = {0, 1, 2, 3, 4, 5, 6, 7};
int x2[] = {0, 1, 2, 3, 4, 5, 6, 6};
int x3[] = {0, 1, 2, 3, 4, 5, 6, 8};
int x4[] = {0, 1, 2, 3, 4, 5, 6, 7};
/* x1 と x2 を比較 */
if ( memcmp(x1, x2, sizeof(int) * N) == 0 )
{
printf("x1とx2は同じ要素を持つ配列です.\n");
}
else
{
printf("x1とx2は異なる要素を持つ配列です.\n");
}
/* x1 と x3 を比較 */
if ( memcmp(x1, x3, sizeof(int) * N) == 0 )
{
printf("x1とx3は同じ要素を持つ配列です.\n");
}
else
{
printf("x1とx3は異なる要素を持つ配列です.\n");
}
/* x1 と x4 を比較 */
if ( memcmp(x1, x4, sizeof(int) * N) == 0 )
{
printf("x1とx4は同じ要素を持つ配列です.\n");
}
else
{
printf("x1とx4は異なる要素を持つ配列です.\n");
}
return EXIT_SUCCESS;
}
配列やメモリ領域の内容を整列(ソート)する
C言語で、配列やメモリ領域の内容を整列(ソート)するには、stdlib.hのqsort関数を使用する。
qsort関数は、baseが指すオブジェクトの配列(要素数がnmemb個、各要素の大きさがsizeである配列)を、compareが指す比較関数にしたがって整列する関数である。
ちなみに、qsort関数の名前はクイックソートに由来するが、内部でクイックソートアルゴリズムを使用してる保障はない。(処理系定義)
#include <stdlib.h>
void qsort (void *base, size_t nmemb, size_t size, int (*compare\)(const void *, const void *));
qsort関数の引数は以下の通りである。
- base : 整列したい配列
- nmemb : 配列の要素数
- size : 配列の個々の要素のサイズ
- compar : 比較関数
qsort関数を使用するには、事前にプログラマが比較関数を実装する必要がある。比較関数は、以下のルールに基づいて実装する必要がある。
- 第1引数が第2引数より小さい場合 : 0より小さい値を返す
- 第1引数が第2引数と一致する場合 : 0を返す
- 第1引数が第2引数よりも大きい場合 : 0より大きい値を返す
以下の例では、qsort関数を使用して、int型の配列を昇順に整列している。
#include <stdio.h>
#include <stdlib.h>
int compare(const int *val1, const int *val2);
int main(void)
{
int ary[] = {9, 8, 7, 6, 5, 4, 3, 2, 1};
int n = sizeof(ary) / sizeof(ary[0]);
/* 配列の内容を表示 */
printf("整列前: ");
for ( int i = 0; i < n; i++ )
{
printf("%d", ary[i]);
}
/* qsort 関数を使用して昇順に並び替える */
qsort(ary, n, sizeof(int), (int (*)(const void *, const void *))compare);
/* 配列の内容を表示 */
printf("\n整列後: ");
for ( int i = 0; i < n; i++ )
{
printf("%d", ary[i]);
}
printf("\n");
return EXIT_SUCCESS;
}
/* 比較関数 */
int compare(const int *val1, const int *val2)
{
if ( *val1 < *val2 )
{
return -1;
}
else if ( *val1 == * val2 )
{
return 0;
}
else
{
return 1;
}
}
配列やメモリ領域の内容から文字を探索する
C言語で、配列やメモリ領域の内容から文字を探索するには、string.hのmemchr関数を使用する。
memchr関数は、sが指すオブジェクトの先頭からn文字分検索して、文字cが最初に現れる位置を探索する関数である。
なお、探索中はsが指すオブジェクトとcはunsigned char型として解釈される。
memchr関数は、strchr関数と似ているが、'\0'があっても探索を続ける。
memchr関数は、文字が見つかった場合は探し出した文字へのポインタを返し、文字が見つからなかった場合はNULLを返す。
memchr関数を利用すると、メモリ領域の中で指定した文字が現れる位置を探すことができる。
#include <string.h>
void *memchr (const void *s, int c, size_t n);
以下の例では、memchr関数を使用して、char型の配列を探索している。
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define N 256
int main(void)
{
char s[] = {'S', 'n', 'o', 'o', 'p', 'y', '\0', 'z'};
char *ret;
int c;
/* 入力 */
puts("文字を入力してください.");
c = getchar();
/* s が指す配列中に c があるか? */
if ((ret = (char *)memchr(s, c, sizeof(char) * N)) != NULL)
{
printf("%cは%d番目にありました.\n", c, ret - s);
}
else
{
printf("%cはありませんでした.\n", c);
}
return EXIT_SUCCESS;
}
配列やメモリ領域の内容から任意の値を探索する
C言語で、配列やメモリ領域の内容から任意の値を探索するには、stdlib.hのbsearch関数を使用する。
bsearch関数は、baseが指すオブジェクトの配列(要素数がnmemb個、各要素の大きさがsizeである配列)から、
keyが指すオブジェクトに一致する要素を探索する関数である。
なお、baseが指す配列は昇順に整列(ソート)されている必要がある。
#include <stdlib.h>
void *bsearch (const void *key, const void *base, size_t nmemb, size_t size, int (*compar)(const void *, const void *));
bsearch関数の引数は、以下の通りとなる。
- key: 探索キー
- base: 探索する配列
- nmemb: 配列の要素数
- size: 配列の個々の要素のサイズ
- compar: 比較関数
bsearch関数を使用するには、事前にプログラマが比較関数を実装する必要がある。また、比較関数は以下のルールに基づいて実装する。
比較関数は、keyへのポインタを第1引数とし、配列要素へのポインタを第2引数として呼び出される。
- 第1引数が第2引数より小さい場合 : 0より小さい値を返す
- 第1引数が第2引数と一致する場合 : 0を返す
- 第1引数が第2引数よりも大きい場合 : 0より大きい値を返す
以下の例では、bsearch関数を使用して、int型の配列を探索している。
#include <stdio.h>
#include <stdlib.h>
int compare(const int *val1, const int *val2);
int main(void)
{
int ary[] = {1, 2, 3, 4, 5, 6, 7, 8};
int key, *result;
int n = sizeof(ary) / sizeof(ary[0]);
/* keyを入力 */
scanf("%d", &key);
/* 探索 */
result = bsearch(&key, ary, n, sizeof(int), (int (*)(const void *, const void *))compare);
if ( result == NULL )
{
fprintf(stderr, "%dは見つかりませんでした.\n", key);
}
else
{
printf("%dは配列の%d番目の要素です.\n", key, (int)(result - &ary[0]));
}
return EXIT_SUCCESS;
}
/* 比較関数 */
int compare(const int *val1, const int *val2)
{
if ( *val1 < *val2 )
{
return -1;
}
else if ( *val1 == * val2 )
{
return 0;
}
else
{
return 1;
}
}