Java泛型中 extends和super

Java泛型中的extends和super理解

 


在平时看源码的时候我们经常看到泛型,且经常会看到extends和super的使用,看过其他的文章里也有讲到上界通配符和下届通配符,总感觉讲的不够明白。这里备注一下,以免忘记。

extends也称为上界通配符,就是指定上边界。即泛型中的类必须为当前类的子类或当前类。
super也称为下届通配符,就是指定下边界。即泛型中的类必须为当前类或者其父类。
这两点不难理解,extends修饰的只能取,不能放,这是为什么呢?

先看一个列子:

public class Food {}
public class Fruit extends Food {}
public class Apple extends Fruit {}
public class Banana extends Fruit{}

public class GenericTest {

public void testExtends(List<? extends Fruit> list){

//报错,extends为上界通配符,只能取值,不能放.
//因为Fruit的子类不只有Apple还有Banana,这里不能确定具体的泛型到底是Apple还是Banana,所以放入任何一种类型都会报错
//list.add(new Apple());

//可以正常获取
Fruit fruit = list.get(1);
}

public void testSuper(List<? super Fruit> list){

//super为下界通配符,可以存放元素,但是也只能存放当前类或者子类的实例,以当前的例子来讲,
//无法确定Fruit的父类是否只有Food一个(Object是超级父类)
//因此放入Food的实例编译不通过
list.add(new Apple());
// list.add(new Food());

Object object = list.get(1);
}

在testExtends方法中,因为泛型中用的是extends,在向list中存放元素的时候,我们并不能确定List中的元素的具体类型,即可能是Apple也可能是Banana。因此调用add方法时,不论传入new Apple()还是new Banana(),都会出现编译错误。

理解了extends之后,再看super就很容易理解了,即我们不能确定testSuper方法的参数中的泛型是Fruit的哪个父类,因此在调用get方法时只能返回Object类型。结合extends可见,在获取泛型元素时,使用extends获取到的是泛型中的上边界的类型(本例子中为Fruit),范围更小。

总结:
在使用泛型时,存取元素时用super,获取元素时,用extends
————————————————

package com.Iori;

import android.app.Activity;

import com.Iori.Util.CommUtil;

import java.lang.ref.WeakReference;

import com.Iori.IEntity;
import androidx.annotation.Nullable;

public abstract class CommonCallback<T extends IEntity> extends GeneralCallback<T>
{
    protected Activity __this;
    public CommonCallback()
    {
        __this = (Activity) getWeakRefActivity().get();
    }

    private boolean checkContent()
    {
        Activity activity = (Activity) getWeakRefActivity().get();
        if(activity == null || activity.isFinishing())
        {
            return false;
        }
        return true;
    }

    protected boolean isDismissLoadingDataReady()
    {
        return true;
    }

    @Override
    public void onDataReady(@Nullable T data)
    {
        if(checkContent() == false) return;
        if(isDismissLoadingDataReady())
        {
            CommUtil.dismissHUD();
        }
        if(data.Success)
        {
            doProcessData(data);
        }
        else
        {
            CommUtil.alert(
                    __this,
                    null,
                    data.Msg);
        }
    }

    public abstract void doProcessData(T data);
}

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章