随机名言

泛型

看早期写的博客,不忍直视太简单照搬了,刚好项目遇到了就稍微重写下

@rewrite、@author Howl、@date 2021/07/21


1. 为什么使用泛型

使用泛型可让编写的代码对多种不同类型的对象重用(笔者理解为与方法的重载互补),比如你希望某个方法foo() 可接收多种参数来进行处理而不用为这多个参数各写一个方法,当然可用Object作为参数,但使用Object作为方法参数有两个缺点:

  • 获取其中的值时需要强制类型转换
  • 没有错误检查,可接收任何类型参数,导致强制转换可能报错

还有就是使用泛型了,ArrayList就是典型的例子,其中的arrayList.add(E e) 使用了泛型可处理各种类型参数





2. 什么是泛型

泛型里有个专有名词叫类型参数,就是把类型也当作一个参数,但它是一个不确定类型的特殊类型,它在使用时才明确类型,这样的好处:

  • 不用代码强转
  • 把运行问题提前到了编译时期




3. 定义泛型


3.1 泛型类

泛型写在类名后面,其泛型的作用在于指定方法的返回类型、字段等类型

public class Object<T> {}

3.2 泛型方法

泛型写在修饰符 和 返回值之间,其泛型可指定返回类型,字段等类型

public <T> void foo(T t) {}

3.3 泛型接口

public interface inter<T> {}

3.4 明确泛型

public void foo(List<Sting> t) {}

3.5 不明确泛型

public void foo(T t) {}




4. 实操

现有个需求,把User类型放入集合中,然后遍历


4.1 没使用泛型

ArrayList list = new ArrayList();

User user1 = new User();
User user2 = new User();

list.add(user1);
list.add(user2);
// 若把非User类型放入集合,这样只有运行时报错才知道

for(Object user : list){
    String username = (User) user.getUsername();
}

4.2 使用泛型

ArrayList<User> list = new ArrayList();

User user1 = new User();
User user2 = new User();

list.add(user1);
list.add(user2);

for(User user : list){
    String username = user.getUsername();
}




5. 通配符

public void foo(List<?> list){}
public void foo(List<? extends String> list) {}
public void foo(List<? super E> list) {}
// 只能限定一个实例,但可以多个接口
// 默认使用第一个,所以习惯在E的位置放入实例对象类型,而非接口
public void foo(List<? super E,F,G> list) {}




6. 非继承

不用能instanceOf判断的泛型和赋值,因为泛型之间没有继承关系,但可使用通配符来泛指子类、超类

// 报错
List<Father> list = new List<Son> ();

// 报错,完全不能比较
if (a instanceOf foo<Father>)

Class对象也是一个泛型:Class strClazz = new String("foo").getClass()

泛型是在虚拟机中擦除泛型类型的、方法的返回类型若被擦除,编译器会插入虚拟机指定强制转换



posted @ 2019-12-05 22:54  Howlet  阅读(334)  评论(0编辑  收藏  举报

Copyright © By Howl