泛型

作者: vv_64ce | 来源:发表于2020-04-15 19:58 被阅读0次

    泛型

    jdk5.0新增特性
    注意:1.泛型的类型必须是类,不能是基本数据类型。需要用到基本数据类型的位置,用包装类替换
    2.若实例化时,没有指明泛型的类型。默认指明泛型的类型。默认类型为java.lang.Object类型
    3.jdk7后新特性:类型推断

    ArrayList<String> list = new ArrayList<String>();
            //新特性,前类型推断了后类型
            ArrayList<String> list1 = new ArrayList<>();
    

    如何自定义泛型类、泛型接口;泛型方法

    1.定义泛型类

    public class OrderT01<T> {//泛型类
        String name;
        int age;
        T orderT;
    }
    

    实例化时指明泛型类型,操作更严谨

    //未对操作的泛型指定需要的类型,默认Object类型
            OrderT01 o = new OrderT01();
            o.setOrderT("hong");
            o.setOrderT(12);
    
            //对泛型指定类型
            OrderT01<String> o1 = new OrderT01<>();
            o1.setOrderT("hong");
    //        o1.setOrderT(12);//报错
    

    继承泛型时:

    //1.继承泛型类时,子类也必须有泛型标识;SubOrder01仍是泛型类
    //public class SubOrder01<T> extends OrderT01<T> {
    //
    //}
    //2.对继承的父类泛型确定其类型时,默认子类也为此类型;SubOrder01不是泛型类
    public class SubOrder01 extends OrderT01<Integer> {
            
    }
    
    注意:
    //1.泛型类的构造器和普通类的构造方式一样
    public class OrderT01<T> {//泛型类
             T orderT;
             public OrderT01() {
              }
    }
    
    //2.不同类型的泛型不能相互赋值
            ArrayList<String> l1 = null;
            ArrayList<Integer> l2 = null;
    
            //l1=l2;//泛型不同的引用不能相互赋值
    
    //3.若泛型结构是一个接口或抽象类,则不可以创建泛型类的对象
    
    //4.静态方法中,不能使用反省类;原因是,泛型类是在创建对象的时候使用,而静态结构早于对象的创建
    public class OrderT01<T> {//泛型类
    public static void show(T t){   
        }
    }
    
    //5.异常类不能是泛型类
    
    //6.
    //编译不通过,原因是,此方式是创建对象,表明T的类型已经确定,但T只是一个参数,并没有指定确定的类型
    //T[] t = new T[10];
            //此种方式可以编译通过,将创建的Object类型数组强转成泛型数组(T[])
            T[] t = (T[]) new Object[10];
    
    //7.子父类关系泛型 子父类关系泛型1.png 子父类关系泛型2.png

    2.泛型方法:可以声明为静态的

    一个泛型类中存在如下两个方法,其泛型用<E>表示,
    public class Test<T>{//实例化Test时,确定其T类型
         <T> T[] toArray(T[] a);//泛型方法
        void add(E e); //非泛型方法
    }
    

    举例:自定义泛型方法

    public class GenericMethod {
        //泛型方法,在方法中出现了泛型的结构,泛型参数和类的泛型没有任何关系
        //即,泛型方法所属类是不是泛型都没关系
     //List<E>返回E类型的List数组;E[] arr参数为E类型的数组;<E>表示此时的E是一个泛型参数,不是一个类
        public <E> List<E> copyFromArrayList(E[] arr){
            ArrayList<E> list = new ArrayList<>();
            for(E e : arr){
                list.add(e);
            }
            return list;
        }
    }
    
            GenericMethod g = new GenericMethod();
            Integer[] arr = new Integer[]{1,2,5,2};
            //泛型方法在调用时指明泛型类型Integer
            List<Integer> integers = g.copyFromArrayList(arr);
            for(Integer i : integers){
                System.out.println(i);
            }
    //泛型方法可以声明为静态的,原因是:泛型参数是在调用方法时才确定其类型,并非在实例化时确定
     public static <E> List<E> copyFromArrayList(E[] arr){}
    

    泛型在继承方面的体现

    虽然A类是B类的父类,但是G<A>和G<B>二者不具备子父类关系,二者是并列的;A类是B类的父类(接口),A<G>是B<G>的父类

            ArrayList<Object> a1 = null;
            ArrayList<String> a2 = null;
            a1 = a2;
    
            List<String> l = null;
            ArrayList<String> a = null;
            l = a;
    

    通配符(?)的使用

    虽然A类是B类的父类,但是G<A>和G<B>二者不具备子父类关系,但二者共同的父类是G<?>

            ArrayList<Object> a1 = null;
            ArrayList<String> a2 = null;
            ArrayList<?> a = null;
            a = a1;
            a = a2;
    
    使用通配符后数据的读取和写入操作
            List<?> l = null;
            List<String> l2 = new ArrayList<>();
            l2.add("AA");
            l2.add("BB");
            l2.add("CC");
    
            l = l2;
            //添加(写入):除null之外,不能向 List<?>添加数据
    //        l.add("s");
            l.add(null);
    
            //获取(读取):允许读取数据,类型为Object
            Object o = l.get(0);
            System.out.println(o);//AA
    
    有限制条件的通配符使用

    ? extends A:( 数学中理解,(-oo,A] )
    G<? extends A>可以作为G<A>和G<B>的父类,其中B是A的子类
    ? super A:( 数学中理解,[A,+oo) )
    G<? super A>可以作为G<A>和G<B>的父类,其中B是A的父类

    相关文章

      网友评论

          本文标题:泛型

          本文链接:https://www.haomeiwen.com/subject/cxoxvhtx.html