0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 1 year has passed since last update.

Cの基本15 ~関数の引数宣言省略と可変個の関数引数~ 備忘録

Posted at

はじめに

私は初心者である。この備忘録を参考にされようとしている他の初心者の方は、ここに書かれていることを鵜呑みになさらぬようお願いしたい。何か間違いを見つけた方はコメント欄でご報告頂きたい。


関数の引数宣言の省略

前編で関数へのポインタを入れる変数の宣言で、

int (*p)(int, int);

というのがあったが、引数の型宣言は下記のように省略できるらしい。

#include <stdio.h>

int add(int, int);

int add(int x, int y){
    return x + y;
}

int main(void){
    int z;
    int (*p)();    // ここ

    p = &add;
    z = (*p)(2,3);
    printf("%d\n", z);
    return 0;
}    // 出力は変わらず5

こうすると、勝手に実引数に対応した動作になる。
省略した方がより多くの関数に対応できて便利だ。

注意点として引数の宣言を省略すると、
コンパイラが実引数の型チェックと、仮引数の型への変換ができないらしい。

どういうことか?

仮引数を省略して書くと、仮引数の型が分からないから実引数の型から仮引数を想定するしかなくなるらしい。想定の仕方は以下だ。

  • 整数型の実引数について、整数拡張をした結果の型を仮引数の型とする。
  • float型の実引数については、仮引数はdouble型とする。

このルールには名前が付いていて既定の実引数拡張と呼ぶらしい。

#include <stdio.h>

int f(float, short int);

int f(float f, short int s){
    return f + s;
}

int main(void){
    int (*p)();
    int n;

    p = &f;
    n = (*p)(3.0F, (short int)2);
    printf("%d\n", n);
    return 0;
}    // 2と出力

出力もあるが(私の環境では)同時に警告文も出た。

incompatible function pointer types assigning to 'int (*)()' from 'int (*)(float, short)' [-Wincompatible-function-pointer-types]

型省略しているから実引数拡張が起こる。
こういう時は大人しく、(関数への)ポインタの宣言時に引数をつけるのが良さそうだ。

#include <stdio.h>

int f(float, short int);

int f(float f, short int s){
    return f + s;
}

int main(void){
    int (*p)(float, short int);
    int n;

    p = &f;
    n = (*p)(3.0F, (short int)2);
    printf("%d\n", n);
    return 0;
}    // 警告なし、出力5

諸々のエラーを起こすリスクを考えたら、引数の宣言はなるべくしときたいな。

関数へのポインタの相互変換

(新しいC言語の教科書 P469)


可変個の関数引数

引数の個数が可変である関数
printf("Hello, world\n");
printf("Your age is %d\n", age);

printf関数を見てみよう。引数1つでも呼べるし2つでも呼べる。
引数が1個以上あれば 呼べるのだ。

このように引数が可変な関数について見ていく。

以下のコードは、可変個のint型の引数をとりその総和を返す関数sumupとメイン。

#include <stdio.h>
#include <stdarg.h>

int sumup(int, ...);

int sumup(int n, ...){
    va_list ap;
    int sum = 0;

    va_start(ap, n);

    while(n-->0)
        sum += va_arg(ap, int);
    va_end(ap);
    return sum;
}

int main(void){
    int x;

    x = sumup(1, 100);
    printf("%d\n", x);
    x = sumup(3, 100, 150, 200);
    printf("%d\n", x);

    return 0;
}

まず関数宣言で可変個引数の場合は、int sumup(int, ...);
引数の可変部をピリオドで表す。関数定義の先頭でも同様。

呼ばれる側の可変個引数の関数では次のことが必要だ。

  • そのファイルでstdarg.hを取り込む
  • 引数を一つずつ取り出して処理する

引数の処理は以下のようになっている。

  1. va_list型の変数apを宣言する
  2. va_startマクロを使ってapを初期化する
  3. va_argマクロを使って引数を一つずつ取り出す
  4. va_endマクロを使ってapの後始末をする

va_list型の変数は、可変個引数を扱っている状態を保持する。

仮引数の中の最後の固定引数(ここでいうn)をva_startマクロに指定してapを初期化。

va_argはapを使って引数を一つずつ取り出し、apの状態を変えて、次の引数が取り出せるようにすると。va_argを繰り返すことで、次々と引数が取れるらしい。
va_arg(ap, int)のように、引数の型を指定しなければならない。
(新しいC言語の教科書 P476)

関数から戻る前には,va_endを使って状態の後始末をするのが決まりだと。

実引数拡張に注意せよ

関数宣言では可変部分の型の宣言はしてない。だから実引数拡張が起こる。

上記で書いたsumupの浮動小数点数verを作れ に対する私のコード

#include <stdio.h>
#include <stdarg.h>

double sumup(int, ...);

double sumup(int n, ...){
    va_list ap;
    double sum = 0.0;

    va_start(ap, n);

    while(n-->0)
        sum += va_arg(ap, double);
    va_end(ap);
    return sum;
}

int main(void){
    double x;
    float f = 100.5;
    double d = 50.5;

    x = sumup(1, f);
    printf("%f\n", x);
    x = sumup(3, f, f, d);
    printf("%f\n", x);

    return 0;
}     // 出力は100.500000と251.500000

仮引数が指定できないときは実引数拡張が起こるかもと忘れないようにしよう。

  • 整数型の実引数について、整数拡張をした結果の型を仮引数の型とする。
  • float型の実引数については、仮引数はdouble型とする。

キリがいいので〆る

今回の内容は富永和人氏の新しいC言語の教科書を参考に書き留めてます。

0
0
2

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?