関数ポインタ
[Wikipedia|▼Menu]

関数へのポインタ (: pointer to function) あるいは関数ポインタ (: function pointer) は、C言語, C++, D言語やその他多くのプログラミング言語におけるポインタの一種である。関数へのポインタをデリファレンス (dereference) すれば、そのポインタが指し示す関数(サブルーチン)を呼び出せる。応用例としては、switch文を置き換えるテーブルジャンプを実装する、コールバック関数によるカスタマイズポイントを提供する、などといったものがある。

関数オブジェクト (: function object) は、関数へのポインタに似ているが、コード領域中のエントリポイントを指す単なるポインタである関数へのポインタと違い、データ領域上に実体を持つオブジェクトであるという点が異なっている(実装の詳細は言語や処理系により異なるが)。そのため、関数オブジェクトはデータを保持でき、クロージャを再現することもできる。ゆえに、関数オブジェクトは、「関数へのポインタ」ではなく「関数」という型と値を持つようなものと言え、より強力である。

C#Visual Basic .NETなどといった.NET Framework用の言語には、メソッドを参照する型として、デリゲートがある。P/Invokeなどの.NET相互運用において、デリゲートは関数へのポインタにマーシャリングされる[1]

Javaはバージョン8でメソッド参照を導入し、関数ポインタやデリゲート類似の機能を利用できるようになったが、バージョン7まではメソッド参照を持たず、代替としてメソッドを1つだけ持つインタフェースを利用して同等機能を実現する必要がある。

第一級オブジェクトとして関数を使用できる(第一級関数がある)言語では、関数も引数で渡したり、戻り値で返したり、他の関数から動的に作成したりできるなどデータ同様に扱えるため、関数へのポインタは必要とされない。
目次

1 Cでの例

2 C++での例

3 脚注

4 関連項目

5 外部リンク

Cでの例

以下の例では、関数へのポインタとしてfunc_ptrが宣言され、そこへ関数my_functionのアドレスを割り当てている。そしてfunc_ptrを通じて関数を呼び出している。#include <stdio.h>static int my_function(int a) { printf("my_function: %d\n", a); return 2 * a + 3;}int main(void) { int (*func_ptr)(int a) = my_function; /* あるいは以下でも可能 */ /* int (*func_ptr)(int) = &my_function; */ int x; x = (*func_ptr)(10); /* あるいは以下でも可能 */ /* x = func_ptr(10); */ printf("main: %d\n", x); return 0;}

注1: 関数シンボルf自体は関数型 (function type) であり、関数型は関数ポインタとは異なる。単項のアドレス演算子 (address operator) &を付与した式&fは関数ポインタを返す。しかし、関数型の式fは関数へのポインタに暗黙変換されるため、アドレス演算子を適用せずとも関数へのポインタに代入可能である。

注2: 関数へのポインタfpに単項の間接演算子 (indirection operator) *を付与した式*fpは関数指示子[2] (function designator) となる。(*fp)(arg)という構文は、関数へのポインタfpを通じて関数を呼び出す構文である。しかしCではfp(arg)という構文も認められている[3]

注3: 関数ポインタの宣言における仮引数リストは、型が一致してさえいればいいので、仮引数の名前は省略できる。

次の例では、関数へのポインタを引数として他の関数に渡している。ここでは、関数my_functionが、先の例のように関数へのポインタを通じて呼び出される。関数callerは、引数として関数へのポインタと整数値を1つ取る。引数の整数値は、その関数へのポインタを通じて関数を呼び出すときに渡す引数として用いられる。そこで宣言されている関数へのポインタのプロトタイプに適合しさえすれば、callerの第1引数には、どんな関数でも渡すことが可能である。#include <stdio.h>static void my_function(int a) { printf("my_function: %d\n", a);}static void caller(void (*func_ptr)(int a), int p) { (*func_ptr)(p);}int main(void) { caller(my_function, 10); return 0;}

3番目の例でも関数へのポインタを引数として他の関数に渡して用いている。関数fは、指定された区間で積分 ∫ a b f ( x ) d x {\displaystyle \textstyle \int _{a}^{b}f(x)\;dx} の近似を計算する関数integへ渡されている。 x {\displaystyle x} における f ( x ) {\displaystyle f(x)} の値を求めるために、f(x)がintegから呼び出されている。integでは、「double型の引数を1つ取り、double型の値を返す関数」でありさえすれば、どんな関数でも計算させることが可能である。

なお、関数へのポインタを定義する際には、事前にtypedefを用いて関数型もしくは関数ポインタ型のエイリアスを定義しておくと便利である。#include <stdio.h>#include <math.h>#define PI 3.14159265358979323846typedef double (*fx_ptr_t)(double x);/* あるいは以下でも可能 *//*typedef double fx_t(double x);typedef fx_t* fx_ptr_t;*/double integ(double a, double b, fx_ptr_t fp) { double sum = 0.0; double x; int n; /* 積分 {a,b} f(x) dx の計算 */ for (n = 0; n <= 100; n++) { x = (n / 100.0) * (b - a) + a; sum += (*fp)(x) * (b - a) / (100.0 + 1.0); } return sum;}int main(void) { int i; struct pair { fx_ptr_t fp; const char* name; } pairs[3] = { { cos, "cos" }, { sin, "sin" }, { tan, "tan" }, }; printf("From 0 to pi/4:\n"); for (i = 0; i < 3; ++i) { printf("\t" "Integral of %s = %g\n", pairs[i].name, integ(0, PI/4, pairs[i].fp)); } return 0;}

関数ポインタを引数として渡すことで、処理を外部から組み込む(カスタマイズする)ことが可能である。関数ポインタを受け取って関数を呼び出す側は、その関数内で何が実行されるかを関知する必要がない。引数として渡される関数は、コールバック関数と呼ばれることもある。主にイベント処理や、判断を外部に任せて処理を行なうときなどに用いられる。
C++での例

C++において、以下の関数へのポインタは、Cの関数へのポインタとの互換性があり、呼び出し規約が同じであれば直接の相互運用が可能である。

クラスに属さない名前空間レベルのグローバル関数(フリー関数)

クラスの静的メンバー関数

一方で、クラスの非静的メンバー関数へのポインタは互換性がない。また、非静的メンバー関数へのポインタを経由した関数の呼び出しには、そのクラスのインスタンスが必要となる。

C++では関数への参照を定義することもできるが、非静的メンバー関数への参照を定義することはできない[4]


次ページ
記事の検索
おまかせリスト
▼オプションを表示
ブックマーク登録
mixiチェック!
Twitterに投稿
オプション/リンク一覧
話題のニュース
列車運行情報
暇つぶしWikipedia

Size:13 KB
出典: フリー百科事典『ウィキペディア(Wikipedia)
担当:undef