看早期写的博客,不忍直视太简单照搬了,刚好项目遇到了就稍微重写下 @rewrite、@author Howl、@date 2021/07/21
使用泛型可让编写的代码对多种不同类型的对象重用(笔者理解为与方法的重载互补),比如你希望某个方法foo() 可接收多种参数来进行处理而不用为这多个参数各写一个方法,当然可用Object作为参数,但使用Object作为方法参数有两个缺点:
还有就是使用泛型了,ArrayList就是典型的例子,其中的arrayList.add(E e) 使用了泛型可处理各种类型参数
泛型里有个专有名词叫类型参数,就是把类型也当作一个参数,但它是一个不确定类型的特殊类型,它在使用时才明确类型,这样的好处:
泛型写在类名后面,其泛型的作用在于指定方法的返回类型、字段等类型
public class Object<T> {}
泛型写在修饰符 和 返回值之间,其泛型可指定返回类型,字段等类型
public <T> void foo(T t) {}
public interface inter<T> {}
public void foo(List<Sting> t) {}
public void foo(T t) {}
现有个需求,把User类型放入集合中,然后遍历
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();
}
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();
}
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) {}
不用能instanceOf判断的泛型和赋值,因为泛型之间没有继承关系,但可使用通配符来泛指子类、超类
// 报错
List<Father> list = new List<Son> ();
// 报错,完全不能比较
if (a instanceOf foo<Father>)
Class对象也是一个泛型:Class strClazz = new String("foo").getClass()
泛型是在虚拟机中擦除泛型类型的、方法的返回类型若被擦除,编译器会插入虚拟机指定强制转换