ホームページ >Java >&#&チュートリアル >Java のジェネリックスの共分散

Java のジェネリックスの共分散

高洛峰
高洛峰オリジナル
2016-11-22 17:56:441496ブラウズ

仕事中に問題が発生しました。コードでは次のように説明されています:

package test;import java.util.LinkedList;
import java.util.List;public class ListTest {
    public void func(List<Base> list) {
    }
    public static void main(String args[]) {
        ListTest lt = new ListTest();
        List<Derived> list = new LinkedList<Derived>();
        lt.func(list); // 编译报错    }
}class Base {
}class Derived extends Base {
}

ここで、Base のリストをパラメータとして受け取ることができる関数 func を記述する必要があります。 Derived は Base の派生クラスであるため、Derived リストを渡しても問題ないと思っていました。そのため、Derived のリストも Base のリストの派生クラスである必要がありますが、コンパイラはエラーを報告しました。

その理由を知るために、インターネットでいくつかの情報を調べました。Java のジェネリックスは共変ではありません。

ジェネリックの共分散と反分散は両方とも用語であり、前者は、最初に指定された派生型よりも派生度が低い (具体性が低い) 型を使用できることを指し、後者は、派生型を使用できることを指します。最初に指定されたものよりも小さい、より派生した (より具体的な) 型。

たとえば、C# のジェネリックは共分散をサポートしています:

IEnumerable<Derived> d = new List<Derived>();
IEnumerable<Base> b = d;

しかし、Java のジェネリックは共分散をサポートしていません。上記と同様のコードは Java でコンパイルできません。

しかし、興味深いのは、Java の配列が共分散をサポートしていることです。例:

Integer[] intArray = new Integer[10]; 
Number[] numberArray = intArray;

要約: Java のジェネリックスは共分散をサポートしておらず、これはタイプ セーフの観点からのものです。たとえば、この設計は必ずしも必要というわけではありません。 Java の設計者は、使いやすさとタイプ セーフの間で選択をしたとしか言えません。

最後に、元の質問に戻りますが、このようなメソッド func を実装するには、

public void func(List list) {
}

に変更するか、パラメータ化された型を使用します:

public <T> void func(List<T> list) {
}

しかし、これにも問題があり、func のパラメータ型がぼやけてしまいます。より良い方法は、func を変更するのではなく、パラメーターを渡すときに Base タイプの List を渡すことです。これには、要素を List に追加するときに要素を Base タイプに変換する必要があります。

PS: パラメータのタイプを制限することにより:

public void func(List<? extends Base> list) {
}


声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。