• ベストアンサー

interfaceとabstract

タイトル通りなのですが、interfaceとabstractはどのように使い分ければいいのでしょうか?また、HashTable, HashMap, HashTreeもどのように使い分ければ良いのかが分かりません。よろしくお願いします。

  • Java
  • 回答数6
  • ありがとう数2

質問者が選んだベストアンサー

  • ベストアンサー
  • Snowman
  • ベストアンサー率66% (6/9)
回答No.6

>>本質的には interface と abstract >は同列に対比されるような概念ではありません > >この事について、もう少し説明して頂けますでしょうか  ごめんなさい。もう一度考え直しまして、この発言は撤回することにしました。  interface はまったく別なクラスに使うもの、abstract は兄弟関係のある クラスに使うもの、ということで本質的に違うやんと思ったのですが、 実装を継承せずにメソッドや変数を規定するという働きは共通なので、そっちに 注目すると、同類ですね。  参考になるか分かりませんが、もうひとつ。  interface は基本的にすべて public ですが、 abstract はそうでない場合も あります。interface は継承関係のない複数のクラス間のやりとりを規定するもの なので本質的に public でなければならないのに対して、 abstract は(たとえば Template Method パターンで使われるような場合)、public にする必要がないためです  したがって、「継承関係のない他のクラスに使わせるようなメソッドの定義」 をする場合はたいてい interface にするべきです。そのほうが、外から使えるのは これだというのがわかりやすくなりますし。  この点、ANo.#5 に私も同意です。  ちなみに、interface と abstract の変更は随時やってます。 >感覚的に、interfaceを使ってまずは航空管制英語の定 >義をし、abstractを使って(地球人クラスの中で)航空 >管制英語の実装を出来るだけし、後は(日本人クラスな >りドイツ人クラスなりで)残りの実装を試みたほうが、 >柔軟性があると考えてよろしいでしょうか?  そういうことです。

その他の回答 (5)

  • mojimojio
  • ベストアンサー率51% (14/27)
回答No.5

abstract classにするかinterfaceにするか迷うような場合には、interfaceを使用すべきと考えます。あとで「多重継承ができない!」と困ることがありません。abstract classは何かを実装するクラスのベースとするものであって、それ自体にinterface的な役割を持たせない方が設計として柔軟になります。何でしたら、あるinterfaceをimplementsしたabstract classがあってもまったく問題ありません。 HashMapやHashSetの違いについては、java.utilのCollection, Set, Map, ListあたりのAPIドキュメントを読むとすんなり理解されるのではないかと思います。これらは性能の違いではなくて、明確に機能の違いがあります。 Hashtable(同期化されるHashMap)やVector(同期化されるArrayList)は、昔のAPIの名残なので一切使う必要がなかったりします。同期化したい場合には、Collectionsクラスのsynchronized...メソッドで同期化するラッパオブジェクトを作成することができます。

  • Snowman
  • ベストアンサー率66% (6/9)
回答No.4

 さっきの回答に補足します。  もし、航空管制英語を abstract として実装しようとすると、 abstract class 地球人{  abstract 航空管制英語の定義 } class 日本人 extends 地球人{ 航空管制英語の実装(1); } class ドイツ人 extends 地球人{ 航空管制英語の実装(2); }  というようなことになるでしょう。ここで、航空管制英語の実装(1)と(2)の間にどの 程度の違いがあるのかが問題です。  (1)と(2)の大部分が同じで一部が違うだけならば、それは 地球人 クラスに持っていって、 (1)(2)では必要なところだけをオーバーライドするべきです。このとき、地球人クラスでは abstract になるのはごく一部のメソッドだけのはずです。  逆に、(1)と(2)の実装の大部分が違う、あるいはまったく違う(そのため、地球人クラスでは ほとんどを abstract にせざるを得ない)のであれば、それは日本人とドイツ人が同じ親(地球人)を 継承していること自体が間違っている可能性があります。その場合は、「航空管制英語」を interface にして切り出した上で、日本人とドイツ人を別なクラスとして設計し、それぞれで implement すること を考えるべきですね。

northcurlcurl
質問者

補足

実際に例を用いて説明して下さって、ありがとうございます。 航空管制英語の定義をabstractにするか、それとも interfaceにするかですが、実際にプログラムを書く上 で、そのような事はいつ決定するのでしょうか?もちろ ん上流工程で、航空管制英語を実装するクラスを決定し てからだとは思うのですが、プログラムを書き始めてか ら、そのような変更はしてもよろしいのでしょうか? 感覚的に、interfaceを使ってまずは航空管制英語の定 義をし、abstractを使って(地球人クラスの中で)航空 管制英語の実装を出来るだけし、後は(日本人クラスな りドイツ人クラスなりで)残りの実装を試みたほうが、 柔軟性があると考えてよろしいでしょうか? >本質的には interface と abstract は同列に対比されるような概念ではありません この事について、もう少し説明して頂けますでしょう か。よろしくお願いします。

  • Snowman
  • ベストアンサー率66% (6/9)
回答No.3

 あなたは航空管制官です。航空管制の共通語は英語です。  いま、東から日本人がパイロットのJ111便がやってきました。英語で航空 管制をして、無事着陸させました。  次に、西からドイツ人がパイロットのG13便がやってきました。英語で管制 をして、無事着陸させました。  次に、宇宙から水星人がパイロットのM777便がやってきました。英語で管制 をして、無事着陸させました。  ・・・・このようなケースを想定すると、次のようなクラス/インタフェースの 構造になります。 interface 航空管制英語 {  航空管制英語のインターフェイスの定義 } abstract class 地球人 implements 航空管制英語 {  地球人に共通の記述;  航空管制英語の実装(A); } class 日本人 extends 地球人 { 日本人に固有の記述; } class ドイツ人 extends 地球人 { ドイツ人に固有の記述; } class 水星人 implements 航空管制英語{  水星人の記述  航空管制英語の実装(B); } class 航空管制官{  航空管制英語を話せる相手に対して航空管制を行うプログラム; }  地球人と水星人にはまったくなんの関係もありませんが、どちらも航空管制官と お話ができます。なぜなら、どちらも航空管制英語を身につけている(implements) からです。  それに対して、日本人は生まれながらにして(extends) 地球人です。これらの 違いがイメージつきますか?  まあ要するに、全然別なクラス(地球人と水星人)を、共通のやり方(航空 管制英語)でコントロールしたい場合は、その「やり方」を interface にして、 それぞれのクラスに実装させるといいわけです。  それに対して、同じ種類のクラス(日本人とドイツ人)を、共通のやり方(航空 管制英語)でコントロールしたい場合は、その「やり方」を、上位のクラスに実装 しておくと、一箇所に書けばすむわけです。その「上位のクラス」がたまたま abstract であることはあるでしょうけれど、本質的には interface と abstract は同列に対比されるような概念ではありません。  たとえば、上の例で言うと、「地球人」は別に abstract でなくても何の問題 もありませんから。

  • pasta500g
  • ベストアンサー率46% (30/65)
回答No.2

HashtableとHashMapは、Hashtableが同期型なのに対してHashMapは非同期です。機能的には大差ありません。(両クラスのAPIリファレンスにも書かれていたと思うので読んでみて下さい。) 私は素直にそれに従って、マルチスレッドでインスタンス共有されることがなければ性能を考えてHashMapを使う様にしています。(私の場合ほとんどこれに該当するのでHashtableはあまり使いません) HashTreeって、、、スミマセン知らないです。^^;(J2SEじゃないですよね) interfaceとabstractの使い分けですが、個人的な感覚では、サーバコンテナの様な大きなサービスプログラムがユーザプログラムに断片的/制限的に機能を公開するのにはinterfaceが向いていると思います。JDBC-APIやServlet-APIでセッションを司る機能が殆どinterfaceな意味を考えて、最近そう悟りました。 interfaceを幾つも持つクラスは、それぞれのinterfaceを通して違った顔をユーザプログラムにサービスしながらも、内部的にはひとつのインスタンス上でリソースを統合管理する、なんてのが美しいのでは。そう考えるとinterfaceという名称がとても納得出来るものに感じられました。 対してabstractは、クラスのテンプレートだと解釈しています。

northcurlcurl
質問者

補足

誠に申し訳ありませんが、同期型と非同期方の違いとは 何でしょうか? >HashTreeって、、、スミマセン知らないです。^^;(J2SEじゃないですよね すいませんでした。HashSetの間違いです。それぞれ使 用しているinterfaceが違うというだけで、性能の違い というのがピンときません…。 >abstractは、クラスのテンプレートだと解釈しています interfaceもクラスのテンプレートのように見えるので すが、私だけでしょうか…?

回答No.1

abstractはクラスですから、多重継承ができません。 これに対してinterfaceの場合複数実装することができます。 × class A extends B, C {...} ○ class A implements B, C {...} ○ class A extends B implements C, D {...} したがって、すでに他のクラスから継承されている可能性があるクラスに対して、ある性質を追加しよう、というときにはinterfaceの方が向いていることになります。 一方、interfaceに定義されているメソッドはすべてabstractメソッドであるのに対して、abstractクラスに定義されているメソッドはabstractでないメソッド(すなわち実装されたメソッド)を含めることができます。 ○ abstract class X{  public void baa() {...}; } × interface X{  public void baa() {...}; } ○ interface X{  public void baa(); // interfaceでは暗黙のうちにabstractメソッドとなる } ということで、すでに実装されたメソッドを用意しておきたい場合はabstractクラスを使うと便利です。

northcurlcurl
質問者

補足

合っているかどうか分かりませんが…、 例えば”人間”というクラスで視覚クラス、聴覚クラ ス、等々を実装したい場合、Javaでは多重継承が出来な いため、視覚クラス、聴覚クラス、等々をinterfaceで 宣言することによって、これらの定義を人間クラスで使 用することが出来る、という考えでよろしいのでしょう か。言ってみれば、多重継承をするためには、 interfaceを使うというような…?

関連するQ&A

  • アブストラクトとインターフェースの簡単な一例文

    JAVAについて質問よろしくお願いします。 abstract抽象クラスとインターフェースについて下記理解なのですが、超かんたんな一例など欲しいのですが何かないでしょうか。たとえば package round1.chapter1; public class Helloworld { public static void main(String[] args) { System.out.println("こんにちは!"); } } というような一文です。 ~私の理解~ アブストラクトとは、違ったクラス内で同一メソッドがある場合は親クラスに置き、違ったメソッドがある場合は抽象クラスとして親クラスに置いて、実装はサブクラスに任せるという形式で使われる言葉 インターフェースとは、機能だけを仕様書として書いておき、定義されたとおりのメソッドを実装しないといけないという呼び出し方の統一を図る際に使われるメソッド置き場 ~~~~~ このふたつについてものすごく簡単な使用例を頂きたいです。 よろしくお願いいたします。

    • ベストアンサー
    • Java
  • interfaceを実装するとは?

    interfaceを実装するとはどういうことですか? abstractはクラスの継承で 学校 -abstract(拡張)― 教室 ―abstract(拡張)― 生徒 と機能を細分化するものとわかりました。 インターフェースってなんですか・・

  • abstractなClassについて

    (1)abstractなClassはabstractなmethodを持っていてもnewすることができる。 (2)(1)のabstractなClassを継承したClassは、(1)のabstractなmethodを実装しないとnewすることができない。 この認識は正しいでしょうか?

    • 締切済み
    • PHP
  • コンストラクタ,interface,abstractの呼び出し順について

    現在、java初心者入門などの本で勉強しております 下記についてご教授お願い致します コンストラクタ、interface、abstractの呼び出し順について質問なのですが、 下記プログラムを実行したところ(同一のパッケージ内に明記) // インターフェイス public interface interFaceClassSS {    public void show(); } public interface interFaceClassS {    public void show(); } // 抽象クラス public abstract class ClassSSSab {    // 共通のメソッドを実装    //個々のメソッド    abstract void show(); } // スーパークラス public class ClassSSS extends ClassSSSab{    public ClassSSS(){       System.out.println("ClassSSS");       show();    }    public void show(){       System.out.println("ClassSSSabの抽象メソッドを実装");    } } public class ClassSS extends ClassSSS implements interFaceClassSS {    public ClassSS() {       System.out.println("ClassSS");       show();    }    public void show() {       System.out.println("interFaceClassSSを実装");    } } // メイン処理 public class ClassS extends ClassSS implements interFaceClassS {    public ClassS() {       System.out.println("ClassS");       show();    }    public static void main(String[] args) {       new ClassSSS();       System.out.println("");       new ClassSS();       System.out.println("");       new ClassS();    }    public void show() {    System.out.println("interFaceClassSを実装");    } } // 結果 ClassSSS ClassSSSabの抽象メソッドを実装 ClassSSS interFaceClassSSを実装 ClassSS interFaceClassSSを実装 ClassSSS interFaceClassSを実装 ClassSS interFaceClassSを実装 ClassS interFaceClassSを実装 の結果となりました。 当方が望む結果は、 ClassSSS ClassSSSabの抽象メソッドを実装 ClassSSS ClassSSSabの抽象メソッドを実装 ClassSS interFaceClassSSを実装 ClassSSS ClassSSSabの抽象メソッドを実装 ClassSS interFaceClassSSを実装 ClassS interFaceClassSを実装 上記となります。 new ClassSSSの処理は当方が望む結果なのですが、 new ClassSSでは、 newClassSSSの結果に ClassSS interFaceClassSS が追加されると理解していたのですが、結果は、 違っておりました。 どこに誤りがあるのが、数日検討したのですが、分からない状態です。 ご教授の程お願い致します。

    • ベストアンサー
    • Java
  • javaのabstractの意味

    javaのabstractの意味 abstract class Factory { abstract ClassA createClassA(); } createClassA()メソッドは定義していません。 abstract ClassA createClassA();の意味はなんですか?

  • Javaのabstractについて

    Javaのabstractメソッドは、確か中身がないメソッドということで、オーバーライドしないと呼び出すことができないんではなかったでしょうか・・。 それなのに、javax.xml.DocumentBuilderFactoryクラスのnewDocumentBuilderメソッドって、abstract宣言でも呼び出したらちゃんとDocumentBuilderが返るみたいなんですけど、これ如何に。

    • ベストアンサー
    • XML
  • インターフェイスとインタフェース

    タイトルにもありますインターフェイスとインタフェースですが、正式にはどちらが正しいのでしょうか?なにか正式な書物はあるでしょうか?

  • abstract と static を一緒に付けることはある?

    抽象クラスの抽象メソッドにstaticを付ける(abstract と static をメソッドに一緒に付ける)ことはありますか? 例えば、 abstract class Super {  static abstract void meth(); } class Sub extends Super {  static void meth() {   System.out.print("hello");  } } として、mainの中に  Sub.meth(); を書いてみたのですが、普通にコンパイルは通りhelloが出力されます。 staticの意味を考えると、こういうことをするのは意味がないと思うのですが、独学なので身近に聞ける人もいません。 御教授よろしくお願いします。

  • アブストラクトとイントロダクションについて

    こんにちは、 論文を書く際、アブストラクトとイントロダクションにはどのようなことを書けばよいのでしょうか?両者の違いは何でしょうか?

  • HashMapの容量圧縮について

    初期サイズがわからない状態でHashMap(Hashtableでも可)を作成し、HashMapの容量を圧縮する方法などありますか? ArrayListのTrimToSizeメソッドのようなものがあったらいいのですが、ご存知でしょうか?

専門家に質問してみよう