java 基础回顾 - 泛型

2020-09-23  本文已影响0人  __Y_Q

1. 泛型的定义 - 定义在类上

作用域在类上, 在任何地方出现都代表是同一类型, 可以参考 ArrayList<T>
定义在类上的泛型是在创建对象时确定的

/*
* 定义具有泛型的类
* 修饰符 class 类名 <范型变量>{
*      范型变量一般用E,K,V,T;
* }
* */
public class MyClass1<E> {
    private E e;

    public MyClass1() {
    }

    public MyClass1(E e) {
        this.e = e;
    }

    public E getE() {
        return e;
    }

    public void setE(E e) {
        this.e = e;
    }

    @Override
    public String toString() {
        return "MyClass1{" +
                "e=" + e +
                '}';
    }

    public void print(){
        System.out.println(e);
    }
}

使用

MyClass1<String> stringMyClass1 = new MyClass1<>();
stringMyClass1.setE("张三");
String str = stringMyClass1.getE();

 

2. 泛型的定义 - 定义在方法上

定义在方法上的泛型是在真正调用方法的时候才确定, 一般由传入的参数确定, 包含这个方法的类创建对象时, 并不会确定方法上的类型. 定义泛型方法, 是先声明后使用.

/*
  泛型方法(方法上含有泛型)
  格式:
  修饰符 <声明泛型变量> 返回值类型 方法名称(参数列表...) {
      //...
  }
 */
private <K> K show(K k){
   return k;
}

使用

String s = show("333");
Integer integer = show(123);

 

3. 泛型的定义 - 泛型的上限和下限以及通配符

    /*
    * 泛型在使用时必须左右一致,不存在继承;
    * */
    public static void main(String[] args) {
        ArrayList<?> list1 ;
        ArrayList<String> list2 = new ArrayList<String>();
        ArrayList<Integer> list3 = new ArrayList<Integer>();
        //下面这行代码会报错,泛型在使用时必须左右一致,不存在继承
        ArrayList<Object> list4 = new ArrayList<String>();
        //下面两行代码不会报错,使用了通配符,通配符表示任意类型,这是在使用,把?当成任何一种类型,
        //?就是任何类型。
        list1 = list2;
        list1 = list3;
        //这行代码会报错  因为两者在定义时使用泛型,一旦确定了类型就不能够改变类型,
        //只能是给定的一种泛型类型。
        list2 = list3; 
    }

上面说明了 为什么泛型在使用的时候必须左右一致.

    public static void print1(ArrayList<?> list) {
        //使用了通配符里面的元素被限定了只能使用Object的方法,
        //底层使用的是Object所以不能添加元素,因为不知道里面到底是什么类型的元素
        list.add(new Object());  //会报错
        list.get(0);
        list.remove(0);
        list.set(0,new Object());  //会报错
        for(Object obj : list) {
            System.out.println(obj);
        }
    }
public class BaseActivity extends Activity {
    public void startActivity(Class clazz){
        Intent intent = new Intent(this, clazz);
        startActivity(intent);
    }
}

public class MyActivity extends BaseActivity {
    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        startActivity(MainActivity.class);
    }
}

但是在使用过程中, 我们发现在之类中调用父类的 startActivity 的时候, 无论传什么进去都可以, 什么String.class, Intent.class, Activity.class 都可以也不会报错, 这样就不爽了, 我们想接收的只是 Activity , 于是我们接着对 BaseActivitystartActivity 方法进行改造. 只接收继承自 Activityclass. 变成了下面这样.

public class BaseActivity extends Activity {
    public void startActivity(Class<? extends Activity> clazz) {
        Intent intent = new Intent(this, clazz);
        startActivity(intent);
    }
}

又或者是只接收有继承了 BaseActivityclass.

public class BaseActivity extends Activity {
    public void startActivity(Class<? extends BaseActivity> clazz) {
        Intent intent = new Intent(this, clazz);
        startActivity(intent);
    }
}

这样传入的非继承自 Activity 或者 BaseActivity 的那些类就会直接报错. 这就是泛型的上限, 无论你传入参数的中间继承了多少次, 只要最顶层是继承自 BaseActivity 就可以.


下限: 可以简单的理解为, 传入的参数最低要求都是指定的类名或者指定类名的父类. 这个我们一般不常用. 这里就不再举例,

上一篇下一篇

猜你喜欢

热点阅读