メソッド (計算機科学)
メソッド (method) あるいは メンバー関数 (-かんすう, member function) とはオブジェクト指向プログラミング言語において、あるクラスまたはオブジェクトに所属するサブルーチンを指す。
概要
オブジェクト指向プログラミングにおけるメソッドという用語は元々SmalltalkによってSimulaのメンバープロシージャー (member procedure) をメッセージとメソッドに分けるために導入された。C++ ではメンバー関数と呼ばれるが、これはSimulaのメンバープロシージャーをC言語に流用したことに由来している。Javaのような言語やマイクロソフト[1]などの企業がメソッドという用語を使っているのは、元々C++よりもSmalltalkの影響を受けていたためである[要出典]。近年[いつ?]では言語設計やOS開発等で直接Smalltalkの影響を受けていない場合でも、Javaやマイクロソフト等の影響によりメンバープロシージャーやメンバー関数に当たるものをメソッドと呼ぶことが一般化している。
Smalltalkの特色を色濃く受け継いだObjective-Cでは、メッセージとメソッドを明確に使い分けている[要出典]。
メッセージとメソッドが分かれている言語では1個のメソッドに対しセレクターが異なる複数のメッセージを送ることができる。このためメンバー関数型の言語と違いメッセージとメソッドの分離を強く意識しておく必要がある。[要出典]
メソッドと通常の関数の違いは主にインスタンス内部へのアクセスの有無である。クラスに所属するフィールドおよびメソッドにはそれぞれアクセシビリティ(アクセスレベル)を設定することができ、アクセス権がないコード領域からは参照したり呼び出したりすることができない。この機能は通常カプセル化と呼ばれており、クラス定義の抽象化に貢献する。また呼び出し時に操作の対象となるインスタンス (レシーバ) を this
や self
といったキーワード(予約語)、あるいはメソッドに渡された引数によって参照することができる。
C++ではクラスに属さない関数である大域関数(グローバル関数あるいはフリー関数)と対比されることがある。
またメソッドはサブクラス化の際にオーバーライドされることがあり、実際に発生する動作(振る舞い)がレシーバに依存するという特徴を持つ。これを多態性(ポリモーフィズム)と呼ぶ。
統一モデリング言語 (UML) ではメソッドのことを操作 (operation) と呼ぶ。
インスタンスメソッドとクラスメソッド
インスタンスメソッド (instance method) とはインスタンスに属するメソッドのことであり、インスタンスに対しメッセージを送信する事で実行される。インスタンス変数の操作に使われ、インスタンスメソッドはオブジェクト指向プログラミングの中核をなし、もっともよく使われる。
一方、クラスメソッド (class method) とはクラスに属するメソッドのことであり、クラスに対しメッセージを送信する事で実行される。クラス変数の操作やオブジェクトの生成などに使われる。多くのオブジェクト指向言語はメタクラスをサポートしており、クラスオブジェクトの操作手段となる。C++やJavaといったオブジェクト指向言語では静的メンバー関数あるいは静的メソッド (static method) とよばれ、その振る舞いはクラス変数の操作が許可される点を除き非オブジェクト指向言語における関数やプロシージャと変わらない。
インスタンスメソッドとクラスメソッドの例
Smalltalkによる例
Smalltalkによるインスタンスメソッドとクラスメソッドの例を示す。
Object
subclass: #MethodSample
instanceVariableNames: 'name' "インスタンスオブジェクトに持たせるインスタンス変数"
classVariableNames: '' "クラスオブジェクトとクラスオブジェクト直属のインスタンスオブジェクトの間で共有するクラス変数"
poolDictionaries: '' "クラスオブジェクトとインスタンスオブジェクトの間で共有するプール変数"
category: 'Example'.
"インスタンスオブジェクトのインスタンス変数を操作するメソッド(インスタンスメソッド)"
MethodSample methodsFor: 'accessing'
!
givenName
^ name.
!
givenName: aString
name := aString.
!!
"インスタンスオブジェクトとクラスオブジェクトを操作するメソッド(インスタンスメソッド)"
MethodSample methodsFor: 'inter-accessing'
!
name
^ self givenName, ' ', self class familyName.
!!
MethodSample class instanceVariableNames: 'name'. "クラスオブジェクトに持たせるインスタンス変数"
"クラスオブジェクトのインスタンス変数を操作するメソッド(クラスメソッド)"
MethodSample class methodsFor: 'accessing'
!
familyName
^ name.
!
familyName: aString
name := aString.
!!
MethodSample class methodsFor: 'instance creation'
!
withGivenName: aString
^ self
new
givenName: aString;
yourself.
!!
インスタンスメソッドを実行するには、まずインスタンスオブジェクトを生成しなければならない:
| objectA objectB |
objectA := MethodSample withGivenName: 'John'.
objectB := MethodSample withGivenName: 'Joe'.
上の例では #withGivenName:
により二つのインスタンスオブジェクトを生成し変数 objectA
と objectB
に代入している。この時点で、objectA
とobjectB
のインスタンス変数 name
にはそれぞれ "John" と "Joe" が代入されている。
インスタンスメソッドを実行するには次のように記述する:
objectA givenName. "'John' を返す"
objectB givenName. "'Joe' を返す"
上の例では objectA
と objectB
それぞれのインスタンスオブジェクトに対し #givenName
メッセージを送りインスタンスメソッドを実行している。それぞれのメソッドの返り値が異なることから、同じクラスオブジェクトに属するインスタンスオブジェクトでもインスタンス変数が持つ値は、インスタンスオブジェクト毎に異なることがわかる。
一方、クラスメソッドを実行するには、クラスオブジェクトに直接属しているため、インスタンスオブジェクトの代わりにクラスオブジェクトに対してメッセージを送る。 クラスメソッドを実行するには次のように記述する:
type := MethodSample
type familyName: 'Hillton'.
type familyName. "'Hillton' を返す".
上の例では type
にクラスオブジェクトMethodSample
を代入して、#familyName
と#familyName:
メッセージを送りクラスメソッドを実行している。
クラスオブジェクトのインスタンス変数 name
は、インスタンスオブジェクトのインスタンス変数と異なり MethodSample
に属する全てのインスタンスオブジェクトで共有される。
クラスオブジェクトのインスタンス変数が共有される例を示す:
| type objectA objectB |
type := MethodSample
objectA := type withGivenName: 'John'.
objectB := type withGivenName: 'Joe'.
"#nameはクラスオブジェクトに#familyNameを送っているため、異なるインスタンスオブジェクトでも'Hillton'が共有されている。"
type familyName: 'Hillton'.
objectA name. "'John Hillton'を返す"
objectB name. "'Joe Hillton'を返す"
クラスメソッドは、変数に代入せず直接クラス名を指定して呼び出すことが多い。特にクラスがオブジェクトではない言語においては、直接クラス名を指定する書き方しかできない。
直接クラス名を指定したクラスメソッドの呼び出しは次のように記述する:
MethodSample familyName: 'Hillton'.
MethodSample familyName. "'Hillton' を返す".
なお、MethodSample
のインスタンスオブジェクト生成するときに使用した new
もクラスメソッドである。
一般的にインスタンスオブジェクトを生成する場合には new
という特別な演算子を用いる言語が多い。しかし、Smalltalkの影響が強い言語やRuby等いくつかの言語ではクラスメソッドによりインスタンスオブジェクトを生成する。クラスメソッドによりインスタンスオブジェクトを生成する言語の場合、new
を独自の実装に変更することができる。例えばnew
を実行したとき、別のクラスオブジェクトに属するインスタンスオブジェクトを返すようにすることができる。C++のようにnew
が演算子でありながら、クラスメソッド (静的メンバー関数) としてnew
を定義できる言語もある。
クラスがオブジェクトになっている言語の場合、インスタンスもクラスも同じオブジェクトとして扱われるため、インスタンスメソッドとクラスメソッドでメッセージの送り方に区別はない。インスタンスメソッドの代わりにクラスメソッドを呼び出すことも、クラスメソッドの代わりにインスタンスメソッドを呼び出すこともできる。どちらのメソッドを呼び出すかは、メッセージを送った変数にインスタンスとクラスのうち、どちらのオブジェクトを代入していたかで決まる。
多くの言語ではインスタンスメソッドとクラスメソッドは同じシグネチャ (signature: 名前と引数) を定義できる。SmalltalkやObjective-Cなどメッセージが存在する言語ではメソッドの多重定義ができないため一見無理なように見えるが、メソッドが所属するオブジェクトがインスタンスとクラスで異なるため、同一のシグネチャでインスタンスメソッドとクラスメソッドを定義することができる。
Javaによる例
Javaで記述したインスタンスメソッドと静的メソッド(クラスメソッド)の例を示す。Java では static
修飾子がついたメソッドが静的メソッドであり、ついていなければインスタンスメソッドである。
public class MethodSample {
/** インスタンスフィールド */
private String name;
/** クラスフィールド */
private static int number;
/** インスタンスを生成するためのコンストラクタ */
public MethodSample(final String name) {
this.name = name;
}
/** インスタンスメソッド、getter */
public String getName() {
return this.name;
}
/** 静的メソッド、getter */
public static int getNumber() {
return MethodSample.number;
}
/** 静的メソッド、setter */
public static void setNumber(final int number) {
MethodSample.number = number;
}
}
Smalltalk同様インスタンスメソッドを呼び出すには、まずコンストラクタを呼び出してインスタンスを生成しなければならない:
MethodSample objectA = new MethodSample("John");
MethodSample objectB = new MethodSample("Joe");
上記の例は、Smalltalkの例におけるインスタンスオブジェクトの生成と同様に動作する。C++の表記を踏襲したJavaでは、new演算子によってインスタンスを生成する。
インスタンスメソッドを呼び出すには次のように記述する:
objectA.getName(); // "John" を返す
objectB.getName(); // "Joe" を返す
上記の例は、Smalltalkの例におけるインスタンスメソッドの呼び出しと同様に動作する。
クラスメソッドを呼び出すには次のように記述する:
MethodSample.setNumber(100);
MethodSample.getNumber(); // 100 を返す
上記の例は、Rubyの例における直接クラス名を指定したクラスメソッドの呼び出しと同様に動作する。
Javaではメタクラスとしてjava.lang.Class
クラスをサポートする。java.lang.Object.getClass()
メソッドによりClass
型オブジェクトを取得できる。また、クラス名.class
という構文でClass
型オブジェクトを取得することもできる。さらにリフレクションを使うことで、Class
型オブジェクトからメソッドを呼び出したり、フィールドにアクセスしたりすることもできる。
.NET FrameworkではメタクラスとしてSystem.Type
クラスをサポートする。System.Object.GetType()
メソッドによりType
型オブジェクトを取得できる。C#ではtypeof
演算子により型シンボルからType
型オブジェクトを取得することもできる。また、リフレクションもサポートしている。
C++はクラス型オブジェクトやリフレクションをサポートせず、クラス自体を何らかの変数に代入するようなことはできない[2]。
メッセージ送信とメソッド呼び出し
オブジェクト指向を解説した書籍などでメソッド呼び出しについてオブジェクトにメッセージを送信すると表現されることがある。C++系統の言語ではオブジェクトの操作は単なるメンバー関数(メソッド)呼び出しに過ぎず比喩として捉えられる場合が多い。SmalltalkやObjective-Cにおいては、メッセージ送信は単なる比喩ではなく実体のある機構であり、メソッド呼び出しとは別物であるため注意が必要である。
仮想メソッド、抽象メソッドと具象メソッド
仮想メソッド
仮想メソッド (virtual method) とは、サブクラスでオーバーライド (再定義) し、動作を変更することのできるメソッドのことである。C++では仮想関数と呼ばれる。
C++およびC#のメソッドはデフォルトで非仮想であり、メソッドにvirtual
修飾子をつけることで仮想メソッドとすることができる。なお、C#のクラス(参照型)は仮想メソッドおよび非仮想メソッドの両方を定義することができるが、構造体(値型)は仮想メソッドを定義することができない。一方Javaのメソッド(インスタンスメソッド)は常に仮想であり、final
修飾子をつけることでオーバーライドを禁止できるが、非仮想メソッドとなるわけではない。final
メソッドを非final
メソッドに変更しても、バイナリ互換性は維持される[3]。
仮想と非仮想
C++では、一般に仮想関数はコンパイル時にどのメンバー関数を呼び出すかを確定できないため、通常の非仮想なメンバー関数呼び出しよりもパフォーマンスが悪いというデメリットがある。そのため、パフォーマンスを気にするC++プログラマには、継承する必要がないクラスのメンバー関数(特にデストラクタを含む)にvirtual
修飾子をつけることを非常に嫌う傾向がある。また、C++にはtemplateという機能が存在し、多くの場合仮想関数はtemplateで代用できてしまうため仮想関数にこだわる必要がないという事情もある[疑問点 ]。ただし、デストラクタが非仮想の場合、派生クラスのインスタンスのポリモーフィックなdeleteが不可能となる、という利便性および安全上のデメリットも発生する。
メソッドがデフォルトで非仮想というC++に準ずる設計選択をしたC#においても、仮想メソッドの呼び出しには非仮想メソッドよりもコストがかかることを念頭に置いて利用する必要がある[4][5]。
Javaのインスタンスメソッドは常に仮想であるが、クラスメソッド(静的メソッド)はオーバーライドすることのできない非仮想であるため、静的メソッドのほうが呼び出しコストが小さく、パフォーマンス上のメリットがある[6]。
Javaのfinal
修飾子は、パフォーマンス上の理由というよりはむしろ、派生クラスでの不用意なオーバーライドを禁止してバグを未然に防止することにある。「Javaではメソッドをfinal
修飾することでコンパイラの最適化によりパフォーマンスが向上する」という神話があるが、一方で、Java仮想マシンの性能によってはメソッドをfinal
と宣言したからといって優れたパフォーマンスが得られるとは限らないという指摘もある[7][8]。なお、OracleのHotSpot VMは、final
メソッドを検出して非常に効率よく実行できるように最適化されていると説明されている[3]。
抽象メソッドと具象メソッド
抽象メソッド (abstract method) とは仮想メソッドの一種で、メソッドの実装が無く、宣言だけされているもののことである。C++では純粋仮想関数 (pure virtual function) と呼ばれる。このメソッドを利用するには、このメソッドを含むクラスを継承し、そこでこのメソッドをオーバーライドして実装する必要がある。従って、抽象メソッドを含むクラスは継承しない限りインスタンス化できない。このようなクラスを抽象クラスと呼ぶ。
具象メソッド (concrete method) は抽象メソッドの逆で、実装をもつメソッドのことである。主に抽象メソッドをオーバーライドしたインスタンスメソッドのことを意味するために使われる。
JavaおよびC#ではabstract
修飾子を用いて抽象メソッドを宣言できる。抽象メソッドを持つクラス自体もまた、必ずabstract
修飾子を使って抽象クラスとして定義しなければならない。
抽象メソッドはデザインパターンの一つTemplate Method パターンで主要な役割を果たす概念であり、ソフトウェアの拡張性、再利用性、汎用性を高めるのに役立つ。
また、抽象メソッドのみを持つ抽象型として、JavaおよびC#ではインターフェイス (interface) を定義できる。JavaおよびC#においてクラスは多重継承できないが、インターフェイスを複数実装することはできる。
オーバーライド
アクセサ
アクセサ (accessor) とは、特にオブジェクトが持つフィールドに間接的にアクセスするために定義されるメソッドの総称である。フィールドに値の設定 (set) をするメソッドをsetter、フィールドから値の取得 (get) をするメソッドをgetterと呼ぶ。通例、個々のフィールドに対して、個別のアクセサが用意される(2つ以上のフィールドをまとめて設定/取得するようなものは一般的にアクセサとは呼ばれない)。つまり、setterは単一の入力を持ち、getterは単一の出力を持つ。1つのフィールドに対してsetter/getterが対称的に両方用意されることもあれば、片方だけが用意されることもある。日本語ではアクセサー、アクセッサあるいはアクセッサーとも表記する。C# 9では、オブジェクト構築時にのみ呼び出せるinitアクセサを定義することもできる[9]。
アクセサの表記
記法は言語により異なる。
Java系統では、getXXX, setXXXの様に語幹部にアクセス対象の名詞が入る。Smalltalk系統ではXXX, XXX:というようにアクセス対象の名前だけを記述し、入力と出力を引数の有無で区別する。Objective-Cでは、XXX, setXXX:と片方にだけsetをつける命名規則が用いられる。DelphiやC#などのプロパティ機能を持つ多くの言語では、プロパティ自体がアクセサにあたり、専用構文を使用する。Rubyでは、メソッド呼び出しの際の引数を囲むカッコが省略できるため、引数を持たないメソッド呼び出しが読み出し用アクセサに相当するが、書き込み用アクセサにはXXX=という構文が存在する。C++には決まった規則はなくライブラリや開発環境に左右される。
アクセサとRADツール
アクセサが他のメソッドと区別される大きな理由としてRADツールの連携を想定していることにある。多くのRADツールでは、アクセサとして定義されたメソッドを特別扱いし、グラフィカルユーザインタフェース (GUI) のデザイン時にデザイン画面からアクセサの引数を指定できるようになっている。
Javaには元々、get/setの接頭辞を記述するという規則はなかったが、RAD支援ライブラリのJavaBeans (java.beans) がset/getから始まるメソッドを値設定用のメソッドとして特別に扱うため、Beansが登場して以降、Javaのクラスライブラリ全体に渡ってset/getの接頭辞を使用する命名規則が導入された。現在でも初期の名残としてset/getを接頭辞として持たないアクセサメソッドが存在する(String.length()
など)。
アクセサと公開フィールド
多くの言語では、フィールドを公開 (public) 状態にすることで、オブジェクトのフィールドをメソッドを介さず直接読み書きできる。 しかし、不特定多数のアプリケーションから利用されるライブラリなどではカプセル化の観点から、実際にはアクセサを介してフィールドを読み書きすることがほとんどであり、フィールドを公開状態にして直接読み書きすることは避けられる。アクセサを用意することで、ライブラリの仕様に則した使い方を強制することができるというメリットがある。
フィールドとアクセサが1対1の状態であれば、公開状態のフィールドの読み書きとアクセサを介したフィールドの読み書きはほとんど変わらない。それにも関わらず、わざわざアクセサを介してフィールドを読み書きする理由は、アクセサの実装はクラス実装者の自由であり、アクセサ (setter) が受け取った値をどう処理するかはクラスの実装に委ねられるからである。例えば、アクセサ (getter) が値を返す際、フィールドを参照せず常に0を返すという実装も有りうる。
公開フィールドへの直接アクセスと比較した際のアクセサの利点としては以下のようなものがある。
- 意図的に setter メソッドを実装しないことで、フィールドの値を外部から勝手に変更させないようにすることができる。
- getter/setter でフィールドにアクセスするときにデータを加工してから取得したり設定したりすることができる。
- 例えば内部データの単位とは異なる単位の値に換算して返す、などである。
- setter でフィールドにデータを設定するときにバリデーション (validation) を実行できる。
- 例えば設定したいデータを整数のみ、自然数のみ、偶数のみ、あるいは日付形式文字列のみ、などに限定して、範囲外あるいはフォーマット外の値が渡された場合に異常系とみなして例外をスローするといった実装ができる。
また、アクセサを使う最も重要な理由のうちのひとつとして、多態性の恩恵を受けられるという点がある。スーパークラスのアクセサを仮想メソッドにして、サブクラスごとにオーバーライドすることで、上に挙げたような機能や振る舞いをオブジェクトの種別ごとにカスタマイズすることも可能となる。
純粋なオブジェクト指向環境として知られるSmalltalkでは、アクセサによる多態性がMVCの依存性辞書[要説明]の管理を始めあらゆる箇所で活用されている。
アクセサ論争
SmalltalkやObjective-Cのようなメッセージ送信メタファーの言語では、外部からのインスタンス変数(フィールド)へのアクセス時には必ずアクセサを用いるというのが常識である。一方 C++ などでは public
変数への直接アクセスが往々にして利用されることがある。
- 効率の問題。メソッドをいちいち呼び出すコストを避けたい場合。ただしほとんどはコンパイラ最適化によるインライン化で解決できる[11]。
- 記述量の問題。単純に値を設定/取得するだけのことに全てメソッドを記述するのは間違っているという考え方。
問題となるのは後者である。
まずカプセル化や多態性の観点から、アクセサを用いない変数参照は将来にわたっての変更耐久性や拡張性が著しく劣る。また記法の一貫性からアクセサを指示する向きもある。一方否定派は「強力な IDE を用いればリファクタリングは可能であり、むしろフィールドへの直接アクセスを用いる方が意味が明確となる」という主張を展開し、時にフレームに発展する場合がある。これは現代の「goto文論争」ともいうべき、半ば宗教的な対立関係である。
そのほか、C++ではC言語との相互運用時に構造体を用いることがあるが、構造体のメンバー変数は隠蔽しないことが多い。C++におけるclass
キーワードとstruct
キーワードは、デフォルトのアクセシビリティが異なるという違いしかないが、メンバー変数を隠蔽しないC言語互換の構造体として利用するPOD (Plain Old Data) 型を定義する際に、class
キーワードではなくstruct
キーワードが使われることもある。
C# はこの反省から、アクセサ定義を簡素化する要素としてプロパティを導入した[12]。また Ruby など、アクセサを簡単に定義できるメソッドや構文を備えている言語もある。
多重定義
引数の数、型、順序などが異なる同じ名前のメソッドを定義することをメソッドの多重定義といい、多くのオブジェクト指向言語ではメソッドやコンストラクタを多重定義できる機能を持つ。オーバーロード (overload) とも呼ばれるが、オーバーライドとの混同に注意が必要である。
ただし、PHPやPerlのように、プログラミング言語によっては、型の曖昧さが原因によりメソッドを多重定義できないものもある。この場合はメソッドの引数をメソッド先頭で読み取り、引数の型を判定する条件分岐で対応する。
参考文献・脚注
- ^ MFC、COM、.NET Frameworkではメソッドという用語が使われる。
- ^ C++において、
obj.staticMemberFunc()
のようにインスタンスから静的メンバー関数を呼び出す糖衣構文はサポートされるが、クラス自体をオブジェクトとして扱うことはできない。 - ^ a b Javaにおけるメソッド呼出しの仕組み | Java Magazine | Oracle
- ^ Performance Tips and Tricks in .NET Applications | Microsoft Docs
- ^ Writing Faster Managed Code: Know What Things Cost | Microsoft Docs
- ^ パフォーマンスに関するヒント | Android デベロッパー | Android Developers, Internet Archive
- ^ Javaの理論と実践: ファイナル・アンサー? finalキーワードを有効に使用するためのガイドライン | IBM, Internet Archive
- ^ Javaの理論と実践: パフォーマンスの都市伝説 | IBM, Internet Archive
- ^ プロパティ - C# プログラミング ガイド | Microsoft Docs
- ^ C++/Java/C#の場合、イミュータブルに関してはpublicなconst/final/readonlyフィールドで代用することが可能なケースも存在する。この場合、getterも不要である。
- ^ そのほか、C/C++ではコピーのコストを避けるため、関数の戻り値ではなくポインタあるいは参照による引数経由で値を返すことがあるが、コピー省略 (copy elision) およびReturn Value Optimization (RVO) をサポートするコンパイラでは、戻り値で返したとしても不要なコピー処理は除去される。コピー省略 - cppreference.com
- ^ C#の構文はC++やJavaと似ているが、プロパティのように言語機能の中にはDelphiやC++ Builderがベースとなっているものもある。