Maison  >  Article  >  Java  >  11.Bases de Java – Génériques

11.Bases de Java – Génériques

黄舟
黄舟original
2017-02-27 10:43:58978parcourir

Concepts de base

L'essence des génériques est l'application du Type paramétré, c'est-à-dire , le type de données sur lequel l'opération est effectuée est spécifié en tant que paramètre et le type spécifique est spécifié lors de son utilisation.

Ce type de paramètre peut être utilisé dans la création de classes, d'interfaces et de méthodes, appelées respectivement classes génériques, interfaces génériques et méthodes génériques .


1. Développement

Avant JDK 1.5, seul Object est la classe parent de tous les types et du type casting La combinaison de ceux-ci les caractéristiques peuvent réaliser une généralisation de type.

Par conséquent, lors de la compilation, le compilateur ne peut pas vérifier si le cast de cet Object est réussi, ce qui peut provoquer une ClassCastException (exception de cast).

Regardons un exemple pour comprendre le rôle des génériques :

  • Quand les génériques ne sont pas utilisés (avant la version 1.5)

ArrayList arrayList = new ArrayList();
arrayList.add(100);
arrayList.add("abc");//因为不知道取出来的值的类型,类型转换的时候容易出错  String str = (String) arrayList.get(0);
  • Lors de l'utilisation de génériques (après 1.5)

ArrayList<String> arrayList = new ArrayList<String>();
arrayList.add("abc");//因为限定了类型,所以不能添加整形,编译器会提示出错arrayList.add(100);

2. Terminologie

// 以 ArrayList<E>,ArrayList<Integer> 为例:ArrayList<E>:泛型类型

E:类型变量(或者类型参数)

ArrayList<Integer> :参数化的类型

Integer:类型参数的实例(或实际类型参数)

ArrayList :原始类型

3. Explorer

Classes génériques

class Demo<T> {    private T value;

    Demo(T value) {        this.value = value;
    }    public T getValue() {        return value;
    }    public void setValue(T value) {        this.value = value;
    }
}public class Test {
    public static void main(String[] args) {
        Demo<String> demo = new Demo("abc");
        demo.setValue("cba");
        System.out.println(demo.getValue()); // cba
    }
}

Interfaces génériques

interface Demo<K, V> {    void print(K k, V v);
}

class DemoImpl implements Demo<String, Integer> {    @Override
    public void print(String k, Integer v) {
        System.out.println(k + "-" + v);
    }
}public class Test {
    public static void main(String[] args) {
        Demo<String, Integer> demo = new DemoImpl();
        demo.print("abc", 100);
    }
}

Méthodes génériques

public class Test {
    public static void main(String[] args) {        int num = get("abc", 100);
        System.out.println(num);
    }    // 关键 --> 多了 <K, V> ,可以理解为声明此方法为泛型方法
    public static <K, V> V get(K k, V v) {        if (k != null) {            return v;
        }        return null;
    }
}

Qualification de type

La qualification de type peut être utilisée dans les classes génériques, les interfaces génériques et les méthodes génériques, mais faites attention aux points suivants :

  • Que la qualification soit une classe ou une interface, le mot-clé étend

  • peut être donné à l'aide du symbole & Qualifications multiples

  • Si la qualification comporte à la fois des interfaces et des classes, alors il ne doit y avoir qu'une seule classe et elle doit être placée en premier. Par exemple :

public static <T extends Comparable&Serializable> T get(T t1,T t2)

Analysons le rôle de la qualification de type...


1. Ne définissez pas de limites pour les paramètres de type <.>

Observez le code suivant, une erreur de compilation se produira lorsque les paramètres de type ne sont pas qualifiés de type. La raison est la suivante :

  • Parce qu'avant la compilation, le compilateur ne peut pas confirmer de quel type est le type générique (T)

  • il est donc par défaut à T Est un type primitif (Objet).

  • Vous ne pouvez donc appeler que la méthode Object, mais pas la méthode compareTo.

public static <T> T get(T t1,T t2) {    //编译错误
    if(t1.compareTo(t2)>=0);    return t1;
}

2. Définissez des limites sur les paramètres de type

Après avoir défini des limites sur le paramètre de type T, compilez l'erreur. se produit. Parce qu'à l'heure actuelle, le compilateur utilise par défaut le type d'origine de T comme comparable.

public static <T extends Comparable> T get(T t1,T t2) {    if(t1.compareTo(t2)>=0);    return t1;
}

Effacement de type

  • Les génériques en Java sont essentiellement implémentés au niveau du compilateur .

  • Les informations de type dans les génériques ne sont pas incluses dans le bytecode Java généré.

  • Les paramètres de type ajoutés lors de l'utilisation de génériques seront supprimés par le compilateur lors de la compilation. Ce processus est appelé effacement de type.


Regardez l'exemple suivant :

public class Test {
    public static void main(String[] args) {
        ArrayList<String> arrayList1 =new ArrayList<String>();
        ArrayList<Integer> arrayList2 = new ArrayList<Integer>();        // true
        System.out.println(arrayList1.getClass() == arrayList2.getClass());
    }
}
Observez le code, deux tableaux ArrayList sont définis ici :

  • L'un est le type générique ArrayList, qui ne peut stocker que des chaînes, et l'autre est le type générique ArrayList, qui ne peut stocker que des entiers.

  • En comparant leurs objets de classe, le résultat s'avère vrai.

  • Expliquez que les types génériques String et Integer sont effacés lors du processus de compilation, ne laissant que le type d'origine (c'est-à-dire Objet).


Regardons un autre exemple :

public class Test {
    public static void main(String[] args) throws Exception{
        ArrayList<String> arrayList =new ArrayList<String>();
        arrayList.add("abc");
        arrayList.getClass().getMethod("add", Object.class).invoke(arrayList, 100);         for (int i=0;i<arrayList.size();i++) {  
                System.out.println(arrayList.get(i));  
            }  
    }
}
Observez le code, ici un objet de type générique ArrayList est instancié en tant qu'Integer

  • Si vous appelez directement la méthode add, seules les données entières peuvent être stockées.

  • Utilisez la réflexion pour appeler la méthode add, mais vous pouvez stocker des chaînes.

  • Explication Les instances génériques Integer sont effacées après compilation, ne laissant que le type d'origine.


1. Type brut

  • Le type brut (raw type) efface les informations génériques, et enfin le type réel de la variable type dans le

    bytecode .

  • Tout paramètre de type générique a une variable primitive correspondante.

  • Une fois qu'une variable de type est écrasée, elle est remplacée par son type qualifié (les variables non qualifiées sont des objets).

// 此时 T 是一个无限定类型,所以原始类型就是 Objectclass Pair<T> { } 

// 类型变量有限定,原始类型就用第一个边界的类型变量来替换,即Comparableclass Pair<T extends Comparable& Serializable> { }  

// 此时原始类型为 Serializable,编译器在必要的时要向 Comparable 插入强制类型转换
// 为了提高效率,应该将标签(tagging)接口(即没有方法的接口)放在边界限定列表的末尾class Pair<T extends Serializable&Comparable>

2. Type du paramètre de type

Dans l'exemple suivant, le paramètre de type fait référence à T, le type de T C'est le type de ce qu'on appelle le [paramètre de type].

En observant le code, nous pouvons tirer les conclusions suivantes :

  • Ne spécifiez pas le type de [paramètre de type T]. le type original prend la même classe parent Le niveau minimum de

  • Lors de la spécification du type de [paramètre de type T], le type original ne peut être que le type spécifié ou une sous-classe du type

public class Test {    // 定义泛型方法
    public static <T> T add(T x, T y) {        return y;
    }    public static void main(String[] args) {        // 1.不指定泛型

        // 两个参数都是 Integer,所以 T 为 Integer 类型
        int i = Test.add(1, 2); 

        // 两个参数分别是 Integer,Float,取同一父类的最小级,T 为 Number 类型
        Number f = Test.add(1, 1.2);        // T 为 Object
        Object o = Test.add(1, "asd");        // 2.指定泛型

        // 指定了Integer,所以只能为 Integer 类型或者其子类
        int a = Test.<Integer> add(1, 2);        //编译错误,指定了 Integer,不能为Float
        int b=Test.<Integer>add(1, 2.2); 

         // 指定为Number,所以可以为 Integer,Float
        Number c = Test.<Number> add(1, 2.2);
    }

}

3. Vérification de type

La vérification de type des génériques est pour la référence, pas pour l'objet référencé lui-même.

Dans l'exemple suivant, list est un objet de référence, donc la vérification de type s'effectue par rapport à lui.

// 没有进行类型检查,等价于 ArrayList list = new ArrayLis()ArrayList list = new ArrayList<String>();
list.add(100);
list.add("hello");// 进行编译检查,等价于 ArrayList<String> list = new ArrayList<String>();ArrayList<String> list = new ArrayList();
list.add("hello");
list.add(100);  // 编译错误


4.类型擦除与多态的冲突

来看下面的例子,这里定义了一个泛型类 Parent,一个实现它的子类 Son,并在子类中重写了父类的方法。

class Parent<T> {    private T value;    public T getValue() {        return value;
    }    public void setValue(T value) {        this.value = value;
    }
}

class Son extends Parent<String>{    @Override
    public void setValue(String value) {         super.setValue(value);
    }    @Override
    public String getValue(){        return super.getValue();}
    }

在上面提到过泛型的类型参数在编译时会被类型擦除,因此编译后的 Parent 类如下:

class Parent {    private Object value;    public Object getValue() {        return value;
    }    public void setValue(Object value) {        this.value = value;
    }
}

此时对比 Parent 与 Son 的 getValue/setValue 方法,发现方法的参数类型已经改变,从 Object -> String,这也意味着不是重写(overrride) 而是重载(overload)。

然而调用 Son 的 setValue 方法, 发现添加 Object 对象时编译错误。说明也不是重载。

public class Test {
    public static void main(String[] args) {
        Son son = new Son();
        son.setValue("hello");        // 关键 -->编译错误
        son.setValue(new Object());
    }
}

那么问题来了,通过上面的分析?Son 中定义的方法到底是重写还是重载?答案是:重写。这里 JVM 采用了桥方法(Brige)来解决类型擦除和多态引起的冲突。

我们对 Son 进行反编译(”Javap -c 类名.class”),得到如下内容:

Compiled from "Test.java"class Son extends Parent<java.lang.String> {
  Son();
    Code:       0: aload_0       
       1: invokespecial #8                  // Method Parent."<init>":()V
       4: return        

  public void setValue(java.lang.String);
    Code:       0: aload_0       
       1: aload_1       
       2: invokespecial #16                 // Method Parent.setValue:(Ljava/lang/Object;)V
       5: return        

  public java.lang.String getValue();
    Code:       0: aload_0       
       1: invokespecial #23                 // Method Parent.getValue:()Ljava/lang/Object;
       4: checkcast     #26                 // class java/lang/String
       7: areturn       

  public java.lang.Object getValue();
    Code:       0: aload_0       
       1: invokevirtual #28                 // Method getValue:()Ljava/lang/String;
       4: areturn       

  public void setValue(java.lang.Object);
    Code:       0: aload_0       
       1: aload_1       
       2: checkcast     #26                 // class java/lang/String
       5: invokevirtual #30                 // Method setValue:(Ljava/lang/String;)V
       8: return        }

发现这里共有 4 个 setValue/getValue 方法,除了 Son 表面上重写的 String 类型,编译器又自己生成了 Object 类型的方法,也称为桥方法。结果就是,编译器通过桥方法真正实现了重写,只是在访问时又去调用表面的定义的方法。


注意事项

  • 不能用基本类型实例化类型参数,可以用对应的包装类来实例化类型参数

// 编译错误ArrayList<int> list = new ArrayList<int>();// 正确写法ArrayList<Integer> list = new ArrayList<Integer>();
  • 参数化类型的数组不合法

Demo<T >{
}public class Test {
    public static void main(String[] args) {        // 编译错误 --> 类型擦除导致数组变成 Object [],因此没有意义
        Demo<String>[ ]  demo =new Demo[10];
    }
}
  • 不能实例化类型变量

// 编译错误,需要类型参数需要确定类型Demo<T> demo = new Demo<T>
  • 泛型类的静态上下文中不能使用类型变量

public class Demo<T> {
    public static T name;    public static T getName() {
        ...
    }
}
  • 不能抛出也不能捕获泛型类的对象

//异常都是在运行时捕获和抛出的,而在编译的时候,泛型信息全都会被擦除掉。会导致这里捕获的类型一致try{  
}catch(Problem<Integer> e1){  
    //do Something... }catch(Problem<Number> e2){  
    // do Something ...}

 以上就是11.Java 基础 - 泛型的内容,更多相关内容请关注PHP中文网(www.php.cn)!


Déclaration:
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn
Article précédent:10.Bases de Java - ProxyArticle suivant:10.Bases de Java - Proxy