[ 前へ | 目次 | 次へ ]

ジェネリクス

ここでは、変数型をパラメータとして指定する、ジェネリクスについて扱います。

変数型そのものを引数にする

関数や構造体の用途によっては、引数やメンバ変数の型が異なるだけで、 それ以外は全く同じ関数や構造体が必要になる場合があります。 このような場合、必要な型すべてに対して、同じような関数や構造体を用意するのは面倒です。 また、未知の型の構造体を、広く一般的に扱いたい場合もあります。

こういった場合に役立つのが、ジェネリクスです。ジェネリクスは、関数の引数と同じような感じで、 型そのものを引数に指定する機能です。

ジェネリクスを用いた関数

ジェネリクスを用いた関数は、以下のような仕様で定義します。

- ジェネリクスを用いた関数定義 -

戻り値の変数型 関数名<型引数1,型引数2,…>( 引数1, 引数2, … ) {
    処理内容 ;
    return 戻り値 ;
}

上で通常の関数と異なるのは、 <型引数1,型引数2,…> の部分です。ここに型の名前を格納する引数を宣言します。 名前は実在の型名では無く、自由に付けます。いわば仮の型です。そこに、呼び出し時に実際の型が入力されます。

実際に、加算を行う関数を、ジェネリクスを用いて記述してみましょう:


Type add<Type>(Type a, Type b) {
	Type value = a + b;
	return value;
}

// add 関数の型引数「 Type 」を int 型として呼び出す
int x = add(1, 2);
println(x);
GenericFunctionInt.vcssl

このプログラムを実行すると、VCSSL コンソールに「 3 」と表示されます。

上の例で、add 関数で用いている「 Type 」型というのは、VCSSL に標準では存在しない型です。 しかし関数内で、Type value = a + b ; のように、Type 型の変数を宣言しています。 さらに、関数の引数も戻り値もType 型となっています。これでエラーにならないのは不思議に思えるかもしれません。 これが成立する理由は、Type が型引数であって、仮の型だからです。

この関数を呼び出す時に、add<int>(1, 2) としています。この <int> という所で、 型引数 Type に int を指定しています。これにより、add 関数の中で、Type 型が int 型と見なされて実行されます。

つまり、上のadd 関数が実際に実行される際には、以下の関数と等価なものとなっています:


int add(int a, int b) {
	int value = a + b;
	return value;
}
IntFunction.vcssl

今度はfloat 型として呼び出してみましょう:


Type add(Type a, Type b) {
	Type value = a + b;
	return value;
}

// add 関数の型引数「 Type 」を float 型として呼び出す
float x = add(1.1, 2.2);
println(x);
GenericFunctionFloat.vcssl

この例ではVCSSL コンソールに「 3.3 」と表示されます。型引数に float を指定したため、 Type 型が float 型と見なされて実行されたためです。

このようにして、ジェネリクスを用いると、同じ関数を、複数の型に対して使用する事ができるようになります。

ジェネリクスを用いた構造体

ジェネリクスは、構造体でも使用する事ができます。それは以下のような仕様で定義します。

- ジェネリクスを用いた構造体定義 -

struct 構造体の名前<型引数1,型引数2,…> {
    メンバ変数a の宣言 ;
    メンバ変数b の宣言 ;
    …
}

このように、構造体名の後ろに型引数を付けます。例えば、型が未定のメンバ変数を持つ構造体は、 以下のように定義して使用します:


struct Box<Type> {
	Type width;
	Type height;
}

Box<float> b;

b.width = 1.2;
b.height = 2.4;

println("幅=" + b.width + " 高さ=" + b.height);
GenericStruct.vcssl

上のプログラムを実行すると、VCSSL コンソールに「 幅=1.2 高さ=2.4 」と表示されます。

上の例の構造体「 Box 」は、型が未知のメンバ変数「 width 」と「 height 」を持ちます。 それを変数宣言時に Box b ; というように、型パラメータに float 型を指定した段階で、 Type 型が float 型と見なされて用意されたわけです。

上の例で、b の宣言を Box b ; に変更して実行すると、メンバ変数が int 型となるため、 結果はVCSSL コンソールに「 幅=1 高さ=2 」と表示されます。


この記事の著者

松井 文宏
[ RINEARN代表, 博士(理学), 応用情報技術者 ]
VCSSLやリニアングラフ3D、その他諸々を開発しています。ガイド類や記事も書いています。


スポンサーリンク



この階層の目次
RINEARN からのお知らせ
※ VCSSL は RINEARN が開発しています。

ExevalatorのVer.2.4をリリース—MCPをサポートし、AI用の計算ツールとしても使用可能に
2025-11-15 - 式計算ライブラリExevalatorのVer.2.4をリリースしました。今回から、AIとやり取りするためのプロトコルである「MCP」をサポートし、AI用の計算ツールとしても使用可能になりました!

Exevalatorの最新版Ver.2.3をリリース、新たにPythonで使用可能に
2025-11-04 - 式計算ライブラリExevalatorのVer.2.3をリリースしました。今回から、新たにPython製のプログラムでも使用可能になりました!AI用ツール開発需要などの背景も踏まえて、詳細をお知らせします。

Exevalatorをアップデート、エラーメッセージの日本語化が手軽に
2025-10-31 - 式計算ライブラリExevalatorのVer.2.2.2をリリースしました。今回から、エラーメッセージを手軽に日本語化できるようになりました。数件のバグ修正&微調整も作んでいます。詳細をお知らせします。

関数電卓RINPn(りんぷん)オンライン版の内部構造を解説
2025-10-22 - 先日登場した、関数電卓ソフトRINPn(りんぷん)のオンライン版の内部構造を解説します。オープンソースなので、自由に改造・流用して、自分だけのWeb電卓を作る事も可能!(かもしれない)

関数電卓RINPn(りんぷん)、どこでもすぐ使えるオンライン版が登場!
2025-10-21 - フリー関数電卓ソフトRINPn(りんぷん)に、Web上でどこでもすぐ使える「オンライン版」が新登場しました!PCだけでなく、スマホでも利用可能です。詳細をお知らせします!

VCSSLのサポートAIが登場!ただし実用品質にはChatGPT有料アカウント(Plus)での利用推奨
2025-08-19 - プログラミング言語VCSSLについての質問対応や、コーディング作業を手伝ってくれるAIさんが登場しました!使用までの流れや推奨事項を解説し、実際の回答例や生成コード例などもたくさん紹介します!

各ソフトやVCSSLの英語版ドキュメント整備がほぼ完了
2025-06-30 - RINEARNでは2年前から、AIの補助による英語版ドキュメントの大幅拡充計画を進めてきました。今回、主要ドキュメント&コンテンツの英訳がほぼ完了し、一応の目標水準に達しました。詳細をお知らせします。

VCSSLの最新版をリリース:外部プログラムとの連携機能を少し強化、他
2025-05-25 - VCSSL3.4.52をリリースしました。外部プログラム(C言語製の実行ファイル等)との連携機能を少し強化し、文字化け対策やOS判別などを可能にしました。他にも細かい機能追加があります。詳細をお知らせします。

VCSSLの最新版をリリース、Java24上での非互換な挙動を対処
2025-04-22 - VCSSL3.4.50をリリースしました。Java24環境上でのネットワークドライブ関連のファイルパス解決で、従来環境とは異なる挙動が生じていたのを解消しました。詳細をお知らせします。

リニアングラフやVCSSLの最新版をリリース、目盛りの位置や内容を自由に指定可能に!
2024-11-24 - リニアングラフ3D/2Dを更新し、自由な位置に、自由な表記内容の目盛りを描けるようになりました!併せて、Java言語やVCSSLでの、プログラム制御用APIも拡張しています。詳細をお知らせします。