• java 泛型详解


    1、定义:泛型的本质是参数化类型,就是将类型由原来的具体的类型参数化,这种参数类型可以用在类、接口、方法中,分别称为泛型类、泛型接口、泛型方法;

    2、泛型类:泛型类的声明和非泛型类的声明类似,除了在类名后面添加了类型参数声明部分,最典型的就是各种容器类,List、Set、Map;

    public class Test<T>{
        private T t;
    
        public T getT() {
            return t;
        }
    
        public void setT(T t) {
            this.t = t;
        }
    }

     a、泛型的类型参数只能是引用型类型<Integer>、<Double>(包括自定义类)等,不能是原始类型<int>、<double>等;

     b、实例化泛型类时,必须指定 T 的具体类型; 

     c、参数化类型 T 可以写成任意字符,常用的 T 、E、K、V等表示泛型;

    3、泛型接口:泛型接口与泛型类的定义及使用基本相同;

     a、实现泛型接口的类未传入泛型实参时,在声明此类的时候,需将泛型的声明也加到类中;

    class B<T> implements Test<T>{
         @Override
         public T next() {
             return null;
         }
    

    b、实现泛型接口的类传入泛型实参时,需将所有使用泛型的地方都要换成实参类型;

    public class B implements Test<String> {
    
        private String[] strs= new String[]{"a", "b", "c"};
    
        @Override
        public String next() {
            return null;
        }
    }

    4、泛型方法:调用方法的时候,指明泛型的具体类型;

    // 泛型类
    class Box<T> {
        /**
         * 在泛型类中声明了一个泛型方法,使用泛型E,这种泛型E可以为任意类型,可以类型与T相同,也可以不同。
         * 由于泛型方法在声明的时候会声明泛型<E>,因此即使在泛型类中并未声明泛型,编译器也能够正确识别泛型方法中识别的泛型。
         */
        public <E> void B_1(E t) {
            System.out.println(t.toString());
        }
    
        /**
         * 在泛型类中声明了一个泛型方法,使用泛型T,注意这个T是一种全新的类型,可以与泛型类中声明的T不是同一种类型。
         *
         */
        public <T> void B_2(T t) {
            System.out.println(t.toString());
        }
    
        // 不是泛型方法
        public void B_3(T t){
            System.out.println(t.toString());
        }
    }

    a、所有泛型方法声明都有一个类型参数声明部分(由尖括号分隔),该类型参数声明部分在方法返回类型之前(在上面例子中的<E>、<T>);

     b、只有声明了<T>、<E>的方法才是泛型方法(B_1、B_2),泛型类中使用泛型的成员方法不是泛型方法(B_3);

     c、<T>、<E>表明该方法将使用泛型类型T、E,此时才可以在方法中使用泛型类型T、E;

     d、泛型方法体的声明和其他方法一样,只能是引用型类型,不能是原始类型(int、double,char等);

    5、泛型上下边界:在使用泛型的时候,我们还可以为传入的泛型类型实参进行上下边界的限制,类型实参只准传入某种类型的父类或某种类型的子类;为泛型添加上边界,即传入的类型实参必须是指定类型的子类型;比如:

    List<? extends Animal>, 通配符的上限,不能往里存,只能往外取,因为编译器只知道容器里的是父类或者父类的子类,但不知道它具体是什么类型,所以存的时候,无法判断是否要存入的数据的类型与容器种的类型一致,所以会拒绝set操作

    <? super E>通配符的下限,往外取只能赋值给Object变量,不影响往里存因为编译器只知道它是子类或者它的父类,这样实际上是放松了类型限制,父类一直到Object类型的对象都可以往里存,但是取的时候,就只能当成Object对象使用了

    // 泛型类
    class Box<T extends Number> {
        private T t;
        public Box(T t){
            this.t = t;
        }
        public T getT(){
            return t;
        }
    }
    // Integer类型是Number类型的子类————正确
    Box<Integer> b = new Box<Integer>(1000);
    // String类型不是Number类型的子类————错误
    Box<String> s = new Box<String>("1000");

    6、类型通配符:一般是使用 ?代替具体的类型参数,?是类型实参,和Integer、Number一样都是一种实际的类型,?可以看成所有的类型的父类(List<?> 在逻辑上是List<String>,List<Integer> 等所有List<具体类型实参>的父类);

    public static void main(String[] args) {
        Box b = new Box();
        List<String> sLst = new ArrayList<String>();
        sLst.add("100");
        List<Integer> iLst = new ArrayList<Integer>();
        iLst.add(100);
        List<Number> nLst = new ArrayList<Number>();
        nLst.add(1000);
        b.BPrint(sLst);
        b.BPrint(iLst);
        b.BPrint(nLst);
    }
    
    /**
     * <? extends T>表示该通配符所代表的类型是T类型的子类
     * <? super T>表示该通配符所代表的类型是T类型的父类
     */
    class Box {
        public void BPrint(List<?> lst) {
            System.out.println(lst);
        }
    }
  • 相关阅读:
    一个Web文件上传的C#源代码
    DataSets and Serialization 数据集和序列化 (英文版)
    如何在Unity中播放影片
    靠边伸缩菜单的做法(类似QQ,碰到就会伸出来)
    Lightmapper
    Unity官方教學專案 Character Customization (紙娃娃系統)
    unity3d用鼠标拖动物体的一段代码
    [unity3d程序] 颜色渐变效果
    C# typeof()实例详解
    XNA Billboard(公告板技术)
  • 原文地址:https://www.cnblogs.com/hzzjj/p/15546618.html
Copyright © 2020-2023  润新知