美文网首页
【泛型】泛型上下边界

【泛型】泛型上下边界

作者: 秀叶寒冬 | 来源:发表于2020-03-02 15:52 被阅读0次

    上一篇 【泛型】通配符与嵌套
    通配符可以是任意类类型,在实际业务中使用通配符时,可能会遇到很多安全问题,如传入的泛型类没有特定的方法或属性,类型转换错误等。为了防止这些问题的发生,就有了上下边界,用于指定通配符的范围。

    1 泛型上限extends

    上限extends指定的类型必须是继承某个类,或者某个接口,即<=,如

    ? extends Fruit
    T extends List
    
    网络图片
    // 容器类(装食物用)
    public class Container<T> {
        private T obj;
    
        public Container(){}
    
        public Container(T obj){
            this.obj = obj;
        }
    
        public T getObj() {
            return obj;
        }
    
        public void setObj(T obj) {
            this.obj = obj;
        }
    }
    

    示例类:

    public class Food<T> {
        private T obj;
        public Food(T obj){
            this.obj = obj;
        }
        public T getObj() {
            return obj;
        }
    
        public void setObj(T obj) {
            this.obj = obj;
        }
    }
    public class Fruit<T> extends Food{
        public Fruit(T obj) {
            super(obj);
        }
    }
    public class Apple<T> extends Fruit{
        public Apple(T obj) {
            super(obj);
        }
    }
    public class Banana<T> extends Fruit{
        public Banana(T obj) {
            super(obj);
        }
    }
    
    public class Meat<T> extends Food{ 
        public Meat(T obj) {
            super(obj);
        }
    }
    public class Pork<T> extends Meat{
        public Pork(T obj) {
            super(obj);
        }
    }
    
    

    调用

    // 主函数
    public static void main(String[] strs) {
              // 水果盘
            Container<Fruit> fruits = new Container<Fruit>();
            // 香蕉篮
            Container<Banana> bananas = new Container<Banana>();
            // 菜盘
            Container<Pork> porks = new Container<Pork>();
            // 一个水果
            Fruit fruit = new Fruit("水果");
            // 一根香蕉
            Banana banana = new Banana("香蕉");
            // 一个苹果
            Apple apple = new Apple("苹果");
            // 一块猪肉
            Pork pork = new Pork("土猪肉");
            Meat meat = new Meat("肉");
    
            // 把洗好的水果装盘
            fruits.setObj(fruit);
            fruits.setObj(apple);
            bananas.setObj(banana);
    
            Container<? super Meat> container = new Container<>();
            container.setObj(pork);
    
            Container<? extends Meat> container1= new Container<>();
            //container1.setObj(meat);会报错
    
            // 把炒好的土猪肉装盘
            porks.setObj(pork);
    }
    
    

    2 泛型下限super

    ? super 指定类型
    

    指定类型不能小于操作的类,即指定类型或指定类型的父类...父类的父类最终至Object,且不能为任意父类的其他子类。


    图片来自网络
    // 加菜
    public static void addDish(Container<? super Meat> container) {
        // 装土猪肉
        container.setObj(new Pork("土猪肉"));
        // 装牛肉
        container.setObj(new Pork("烤肥牛"));
    }
    
    // 主函数
    public static void main(String[] strs) {
        // 菜盘 
        Container<Food> foods = new Container<Food>();
        // 专用装肉盘 
        Container<Meat> meats = new Container<Meat>();
        // 水果篮
        Container<Fruit> fruits = new Container<Fruit>();
            
        // 我们吃饭的时候菜吃完,所以我们加菜
        // 厨师准备用盘子装菜
            
        // 用菜盘装菜
        addDish(foods);
        // 用专用装肉盘装菜
        addDish(meats);
        // 但不能用水果篮装菜,一使用编译器就会提示我们异常
    //    addDish(fruits);
    }
    

    <meta charset="utf-8">

    <article class="_2rhmJa">

    总结

    这两种方式基本上解决了之前所说的问题,但是同时,也有一定的限制。

    1.上限<? extends T>不能往里存,只能往外取 (即:只能get)

    因为编译器只知道容器里的是Fruit或者Fruit的子类,但不知道它具体是什么类型,所以存的时候,无法判断是否要存入的数据的类型与容器种的类型一致,所以会拒绝set操作。

    2.下限<? super T>往外取只能赋值给Object变量,不影响往里存

    因为编译器只知道它是Fruit或者它的父类,这样实际上是放松了类型限制,Fruit的父类一直到Object类型的对象都可以往里存,但是取的时候,就只能当成Object对象使用了。

    所以如果需要经常往外读,则使用<? extends T>,如果需要经常往外取,则使用<? super T>。

    参考:

    1. https://www.jianshu.com/p/0c318bb54502
    2. https://zhuanlan.zhihu.com/p/97517915

    相关文章

      网友评论

          本文标题:【泛型】泛型上下边界

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