美文网首页
关于泛型你要知道的二三事[4]

关于泛型你要知道的二三事[4]

作者: 坑王老薛 | 来源:发表于2018-08-14 20:07 被阅读140次

    书接上文:https://www.jianshu.com/p/578b4d3e6cf5
    本内容均属原创,转载请注明出处:https://www.jianshu.com/p/3096adcf6c8f

    泛型上下限问题:

    泛型的上限问题:

    测试用例:

    public class Test01 {
        public static void main(String[] args) throws Exception{
            List<F> lsF = new ArrayList<>();
            showAll(lsF);
            List<S1> lsS1 = new ArrayList<>();
            List<S2> lsS2 = new ArrayList<>();
            //compile-time error 
            showAll(lsS1);
            showAll(lsS2);
            
        }
        public static void showAll(List<F> ls){
        }
    }
    class F {}
    class S1 extends F{}
    class S2 extends F{}
    
    

    结论,其实这个结论上文已经提到过了,编译期间List<S1>和List<F>并不是一会事情。一个泛型实例指定了不同的泛型类型,这里不能进行互相转换。如果用?通配符,似乎能解决问题。但是不能描述清楚S1、S2和F的关系。那这里怎么办呢?我们采用泛型的上限解决该问题。? extends F。这里?代表传入的泛型对象的实际泛型类型,查看当前类型是否继承自F,如果继承就可以直接传入。

    public static void showAll(List<? extends F>){}
    

    但是注意:

    public static void showAll(List<? extends F> ls){
            ls.add(new S1());
    }
    

    这个代码不能这样写。因为传入的实际类型不确定,导致无法在集合中再添加元素,和我们之前测试的泛型通配符遇到的问题是一致的。有时候我们还会这样写:

    class Stack <T extends Number & Serializable>{
        
    }
    

    声明一个Stack类,该类中的泛型声明为必须是Number的子类,而且还需要实现Serializable接口。和我们想象的一样,这里的接口的是可以实现多个的。实现多个用&连接。

    泛型下限

    如果你还记得泛型方法,应该知道我们写了方法,将一个数组中的元素添加到集合中。我们在上文也提到可以通过下限操作。接下来我们编写一个实例。完成集合到集合拷贝,也就意味着src集合中的类型要完全兼容dest集合中的类型。

    public static void main(String[] args) {
            Collection<Integer> dest = new ArrayList<>();
            Collection<Number> src = new ArrayList<>();
            fillIntoColls(dest,src);
        }
        public static <T> void fillIntoColls(Collection<T> dest ,Collection<? super T> src){
            for(T t:dest){
                src.add(t);
            }
        }
    

    这里注意泛型Collection<? super T> src 表示最小是T类型,或者是它的父类。实际传入的T是Integer类型,而这里泛型下限的值是Number。这里面采用的泛型方法和泛型的下限解决的。

    6、拓展jdk10增强的局部变量的类型推断
    

    泛型擦除

    测试用例:

    public class Test01 {
        public static void main(String[] args) {
            Gen<Integer> g1;
            g1 = new Gen<Integer>(12);
            Gen<Integer> g2;
            Gen<String> g3 = new Gen<>("test generic");
            //Type erasure followed by conversion
            Gen gen = g1;
            Gen<String> g4 = gen;
        }
    }
    //declared a generic type Gen
    class Gen<T>{
        T t;
        public Gen(T t) {
            this.t = t;
        }
        public void getT(){
            System.out.println(t.getClass().getName());
        }
    }
    

    结论:最开始定义的g1包含了泛型类型为Integer类型,然后将g1赋给了一个gen对象,编译器在这里会丢失掉g1本身的Integer的泛型信息。这就擦除,其实我们可以理解为泛型类型由Integer变为了Object。java允许给一个对象赋给一个具体的泛型类型。这里只会包检查警告,但是如果真的通过g4做一些操作,还是会有可能出现异常的。

    jdk8增强的泛型类型推断

    测试代码:

    public class Test01 {
        public static void main(String[] args) {
            Gen<Integer> g = new Gen<>();//1
            Gen<String> g2 = Gen.test01();//2
            Gen.test02(123, new Gen<String>());// 3compile-time error
            Gen<Integer> g3 = Gen.test02(123, new Gen<>());//4
        }
    }
    //declared a generic type Gen
    class Gen<T>{
        public static <Z> Gen<Z> test01(){
            return null;
        }
        public static <Z> Gen<Z> test02(Z z,Gen<Z> g){
            return null;
        }
        public T test03(){
            return null;
        }
    }
    

    结论:
    1、第一行代码创建对象时指定泛型类型,后续通过菱形语法直接输出。这是jdk1.7就支持的
    2、第二行代码调用test01方法,没有指定泛型类型,但是通过调用方法之后可以,显示指定了泛型类型是String类型,推断出test01方法的Z泛型类型是String类型
    3、编译报错,因为调用方法是传入的时Z泛型指定的类型是Integer,而传入的Gen对象的泛型类型变为了String类型,所以编译报错。
    4、第四行代码么问题,因为调用test02方法时传入的Z类型指定是Integer类型,显式的指定返回的Gen类型也是Integer,可有推断出传入的Gen泛型类型是Integer类型,所以直接使用菱形语法。

    注意类型推断不是万能的。比如下面的代码

        //在上面的main方法加入以下代码会报错
        String str = Gen.test01().test03();
    
    

    这里并不能推断出test01方法返回的时Gen<String>,所以也无法推断出test03方法返回的时String类型。如果要推断,可以通过以下的方式:

     String str = Gen.<String>test01().test03();
    

    拓展jdk10增强的局部变量的类型推断

    注意:这里的局部变量的类型推断和泛型无关,主要作为扩展知识。建议使用
    Intellij IDEA 2018.1.1以上版本。可以知识jdk10。或者是JShell
    

    测试实例:这里是基于JShell

        jshell> /list
    
       1 : var lists = new ArrayList<String>();
       2 : lists.add("hello");
       3 : lists.forEach(System.out::println);
    

    结论:允许通过使用var类型进行类型的声明。会自动推断出当前var的类型是ArrayList:

    jshell> System.out.println(lists.getClass());
    class java.util.ArrayList
    
    

    但是注意,var是一个保留字,而不是关键词,也以为着你可以通过以下编码:

    jshell> var var = 10;
    var ==> 10
    
    

    但是这里注意,不能通过var去声明类,接口等。可以作为变量名或者是方法名。

    jshell> interface var{}
    |  错误:
    |  从发行版 10 开始,
    |    此处不允许使用 'var', 'var' 是受限制的本地变量类型, 无法用于类型声明
    |  interface var{}
    |            ^
    
    jshell> class Var{}
    |  已创建 类 Var
    jshell> /list
    
       1 : var lists = new ArrayList<String>();
       2 : lists.add("hello");
       3 : lists.forEach(System.out::println);
       4 : System.out.println(lists.getClass());
       5 : var var = 10;
       6 : class Var{}
       7 : class Test{void var(){}}
    
    

    以上算是对于10 的一些尝鲜吧,其实这只是冰山一角,我们后续继续填坑。嘿嘿,挖坑我们是认真的。

    相关文章

      网友评论

          本文标题:关于泛型你要知道的二三事[4]

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