zoukankan      html  css  js  c++  java
  • Java范型随笔

     最近在帝都好无聊啊, 排遣寂寞就只有让自己不要停下来,不断的思考了 QWQ;

     最近做ndk, java有点忘了,突然看到了一些java范型方面的问题, 踌躇了一会, 想着想着,决定还是写个随笔记录下来。

     范型语法这个网上找度娘可以要到一大把, 我就不记了, 主要是范型上下限问题。

     案例:

       

    public class Test {
    
     
    
      public static class Base{
    
     
    
      }
    
     
    
      public static class A extends Base{
    
     
    
      }
    
     
    
      public static class Fanxin<T>{
    
      T t;
    
      public T get(){
    
      return t;
    
      };
    
     
    
      public void set(T t){
    
      this.t = t;
    
      }
    
    }
    
     
    
    public static class B extends A{
    
     
    
    }
    
     
    
      public static void main(String[] arvg){
    
       
    
        Fanxin<? extends Base> fanxin = new Fanxin<A>();
    
        Base a = fanxin.get();
    
        //fanxin.set(new A()); 1编译报错
    
        Fanxin<? super A> fanxin1 = new Fanxin<Base>();
    
        A obj = (A)fanxin1.get();
    
        //A obj = fanxin1.get(); 2编译报错
    
        //fanxin1.set(new Base());3编译报错
    
        fanxin1.set(new A());
    
      }
    
    }

     首先java范型我理解的是为了安全, 这样可以在编译期检测类型,而不会到运行的时候出现类型转化异常。 第一个 fanxin.set(new A());报错,是因为,这里类型是定义了上限,所以这里通配符代表的类型是base或者base的子类, 虽然A是base的子类, 但是这是不安全的,考虑这种情况:

            Fanxin<B> fanxin = new Fainxin<B>();

         ......各种操作......

            Fanxin<? extends Base> fanxin1 = fanxinB;

            fanxin1.set(new A());

            B b = fanxin.get();//这里有问题

         这时候不安全的因素发生了,我们对fanxin1设置了一个对象A, 但是fanxin1实质是fanxin<B>对象, 这时候如果我们fanxin.get()获得的实际上是A对象,运行的时候会有类型转化异常。

     这样是不安全的, 所以编译的时候会报错,如果一定要set, 那么可以强制类型转换来做,但是这样也失去了意义:

             Fanxin<A> fanxinA = (Fanxin<A>)fanxin1;

             fanxinA.set(new A());

     

        第二个问题,通配符是A或者A的父类, 所以我们可以赋值, 面向对象的原则是子类可以替代父类, 所以是安全的。 但是如果我们读区值,然后赋值给其他对象,那么就会有问题, 比如;

                Fanxin<Base> fanxinBase = new Fanxin<Base>();

               Fanxin<? super A> fanxin1 = fanxinBase;   //Fanxin<? super A> fanxin = new Fanxin<? super A>();编译报错

               A a =  fanxin1.get();

             这里fanxin1.get()实际获得的是Base类, 赋值给A类,编译会报错, 运行的时候会有类型转化异常。

     

              当然如果我们知道这个类的类型,那么我们也可以用强制类型转换避免编译报错。

           例:

              Base b = (Base) fanxin1.get();

          

     

          总之 ?是范型里面的通配符,extends, super只是提供上下限的信息给编译起, 通配符有集合的概念在里面,Fanxin< ? extends Base>, ?类型可能是Base的任何子类,所以我们不能有赋值操作,因为子类和子类是不同的类型,只是子类可以替换父类而已,所以可以按父类来读取值, Fanxin<? super A>, ?类型是A的超类, 所以我们可以将A的任何子类赋值给他,这样是安全的,但是读取的时候,是无法知道这是什么类型的,但是java类都是Object类的扩展, 所以可以获取值赋给obj对象, Object obj = fanxin.get(); 如果要赋值给其他对象那么只有强制类型转换了。

         

     

     

  • 相关阅读:
    通向全栈之路——(4)nginx反向代理配置
    通向全栈之路——(3)node环境搭建
    通向全栈之路——(2)服务器安全设置
    通向全栈之路——(1)服务器新增用户及授权
    window.location.href跳转至空白页
    webpack 1.x 学习总结
    win7下使用apache ab 比较测试node与 tomcat
    Enum,Int,String的互相转换
    读写appSettings配置节方法
    SC命令
  • 原文地址:https://www.cnblogs.com/zhangyan-2015/p/5840807.html
Copyright © 2011-2022 走看看