多重定義

出典: フリー百科事典『ウィキペディア(Wikipedia)』

多重定義 (たじゅうていぎ) あるいは オーバーロード (overload)とは、プログラミング言語において関数演算子メソッドの同一名や同一の演算子記号について複数定義し、利用時にプログラムの文脈に応じて選択することで複数の動作を行わせる仕組みである。 例えば整数型実数型複素数型の値について同じ"+"演算子を使って加算を行う、クラスごとに個々の意味で名前やIDを返すメソッドを定義するなどが挙げられる。多重定義する対象に応じてそれぞれ関数オーバーロード (function overloading)、演算子オーバーロード (operator overloading)、メソッドのオーバーロード (method overloding) と呼ばれる。

目次

[編集] 概要

動作を選択する際に用いられる代表的な文脈情報としては、型付けられたプログラミング言語においては関数や演算子に実引数(演算子ならばオペランド)として与えられた式や変数に関連付けられたの情報が用いられる(稀ではあるが戻り値を利用できるプログラミング言語も存在する)。関数の名称とそれらの型情報の組を合わせたものをシグネチャと呼ぶが、プログラム内でシグネチャが唯一に決まれば、関数名やメソッド名、演算子の記号が重複していても呼び出すべき対象を唯一に決定することが出来る。 このような型づけによる多重定義は、暗黙の型変換implicit type conversion あるいは 型強制 (type coercion))、継承inheritance あるいは包含(inclusion))、総称型generic type、あるいはパラメタ付型(parametric type))と並んでプログラミング言語において多相性 (polymorphism) を実現するための一つの手段である。

理論的には関数の名前や演算子記号は単なる記号であり、意味的必然があるわけではないので、これを反映して多重定義を許すプログラミング言語では多重定義された関数や演算子、メソッドの意味や動作の定義はかなり自由に行うことが出来る(演算子については構文解析の都合上、優先順位などが制限される場合も有る)。とはいえ関数名やメソッド、特に演算子の用法には各分野及びプログラミング言語毎に慣習が育っている場合があり、著名な関数(例えば数学関数のsinなど)やメソッド、演算子に対して慣習とあまりにかけ離れた意味、即ち動作の定義を与えるとプログラムの可読性の著しい低下をもたらす可能性があるので注意が必要である。

[編集] 演算子オーバーロード

例えば、C++において何らかの数値型例えば有理数型のためのクラスを定義するとして、その際+演算子を加算以外の意味で定義すれば間違いなく混乱するであろう。 一方意味が大きく変わる場合でも比較的安全な事例もある。例えば、C++において整数型に対する<<は左ビットシフト演算を意味するが、C++の標準入出力ライブラリで提供されるストリーム型(入出力列を抽象化した型)については右オペランドの値を左オペランドのストリームに書き込むことを意味する。これは一見混乱するようにも見えるが、整数シフトという意味づけはCから引き継いだ以外には特に数学的背景などの慣習もなく、プログラミングにおいて整数型とストリーム型を混同することも考えにくいため比較的無難な多重定義と言える。

boostの提供するライブラリに至っては、より前衛的な多重定義が行われている。各種演算子がことごとく異なる意味で用いられたり、本来配列の添え字を表現するための演算子であった[]を、数式上の括弧の一種であるかのように扱っていたりするケースがある。こうして、本来のプログラム言語の構造から離れた独自の文法をもたせて高度なプログラミングを行うことができるのも多重定義の特徴であるが、このようなスタイルには否定的な意見もある。

[編集] メソッドのオーバーロード

[編集] Javaでの例

Javaのオーバーロードは、引数の数や型による違いだけでなく、異なる型を持つ引数の並べ替えによってもメソッドをオーバーロードすることができる。 たとえば、method()というシグネチャの引数無しメソッドがあるとする。

int method() {}

これは以下のようにオーバーロードできる。

int method(int x) {}

このとき、さらに異なる型のメソッドもオーバーロードできる。

int method(Sting name) {}

引数を増やしたオーバーロードも可能。

int method(int x, String name) {}

さらに、引数の数が同じであっても、順番が異なるためこの状況で以下のようなオーバーロードも可能。

int method(String name, int x) {}

このように同じ型の引数を複数持つメソッドとしてさらにオーバーロードすることも可能。

int method(int x, int y) {}

このように、一度に同じ名前のメソッドをいくつも定義できる。

[編集] オーバーロードの濫用

しかし、便利そうだからといって不必要に無闇にメソッドをオーバーロードすることは、ソフトウェア工学エクストリームプログラミングの観点から推奨されていない。後で使うかも知れないからと無闇にオーバーロードして結局使わなくなることや、リファクタリングするときに、オーバーロードしたメソッドを手動で修正しないときもある。オーバーロードの濫用がもたらすものは、それだけでなく、メソッド名がすべて同じであるため、一見、どのような振る舞いを実装しているのかわかりづらくなるという問題がある。そのときは、意図的に異なるメソッド名で、できるかぎり名前を見ただけでどのような機能を持つメソッドであるのかをわかるようにしておくとよい。そのため不必要なオーバーロードは避けるべきである。 とくに、引数の数が同じで型が異なるオーバーロードされたメソッドの定義は混乱の元である。

また、以下のようにオーバーロードされたメソッドは、誤った使い方の例である。

static void method(String id) { System.out.print("String "); }
static void method(Object id) { System.out.print("Object "); }

このようなメソッドのオーバーロードは、問題なくできる。以下のオブジェクトを作成したとき、

Object[] o = new Object[]{ new String("name"), new Object("obj") };

このオブジェクトを上記のメソッド引数に入れる。

for(int i = 0; i < o.length; i++){
  method(o[i]);
}

このとき、設計者は、出力結果に"String Object"と返されることを期待するが、実際には"Object Object "を返す。

[編集] 曖昧な型を持つ言語

PerlPHPのような曖昧な型を持つ言語では、メソッドのオーバーロードができない、あるいは制限されていることがある。そのときはメソッドの先頭で引数の型を判定する条件分岐で対応する。

[編集] 関連項目

他の言語