开启掘金成长之旅!这是我参与「掘金日新计划 · 12 月更文挑战」的第7天,点击查看活动详情
一、泛型引入
1.1、问题引入
如果需要产生多个对象,每个对象逻辑一样,只是对象内的成员变量类型不同,如何做?
- 创建多个类,给每一个类成员设置指定数据类型 缺点:会导致类的膨胀,重用性太差
- 创建一个类,给这个类成员变量设置Object数据类型 缺点:编译正常,运行时可能会异常
针对以上2个的缺点,就引用了泛型
class Cls3<T>{ //<T>,T可以是任何字母,不固定
T a;
public Cls3(T a){
this.a = a;
}
public T getData(){
return a;
}
}
public class Test {
public static void main(String[] args) {
Cls3<Integer> c1 = new Cls3<Integer>(10); //泛型使用的时候在<>里加入数据类型
System.out.println(c1.getData());
Cls3<String> c2 = new Cls3<String>("xiao");
System.out.println(c2.getData());
}
}
1.2、泛型简介(JDK1.5后引入)
- 泛型可以在编译的时候检查类型安全,并且所有的强制转换都是自动和隐式的
- 泛型的原理就是“类型的参数化",即把类型看做参数。也就是说把所要操作的数据类型看做参数,就像方法的形式参数是运行时传递的值的占位符一样
- 简单的说,类型变量扮演的角色就如同一个参数,它提供给编译器用来类型检查的信息
- 泛型可以提高代码的扩展性和重用性
二、泛型特点
- 泛型的类型参数可以是泛型类
- 泛型类可以同时设置多个类型参数
- 泛型类可以继承泛型类
- 泛型类可以实现泛型接口
三、限制泛型可用类型
- 在定义泛型类别时,默认在实例化泛型类的时候可以使用任何类型,但是如果想要限制使用泛型类型时,只能用某个特定类型或者是其子类型才能实例化该类型时,可以在定义类型时,使用extends关键字指定这个类型必须是继承某个类,或者实现某个接口
- 当没有指定泛型继承的类型或接口时,默认使用extends Objext,所以默认情况下任何类型都可以做为参数传入
四、类型通配声明
4.1、引入
- 同一泛型类,如果实例化时给定的实际类型不同,则这些实例的类型是不兼容的,不能相互赋值
- 泛型类实例之间的不兼容性会带来使用的不便,我们可以使用**泛型通配符(?)**声明泛型类的变量。
Cls1<Integer> c1 = new Cls1<Integer>(10);
Cls1<Double> c2 = new Cls1<Double>(10.12);
Cls1<Object> c3 = new Cls1<Object>(1);
c1 = c2; //不能使用
c3 = c1; //不能使用
4.2、通配符使用
应用场景:不知道返回来是一个什么类型,需要一个变量来接收返回
- “?”代表任意一个类型
- 和限制泛型上下限相似,同样可以使用extends限定通配类型
Cls1<Integer> c1 = new Cls1<Integer>(10);
Cls1<Double> c2 = new Cls1<Double>(10.12);
Cls1<Dog> c3 = new Cls1<Dog>(new Dog());
Cls1<? extends Animal> c4; //通配符
c4 = c3; //只有这条可以
//c4 = c2;
//c4 = c1;
- 还可以使用super将通配符匹配类型限定为某个类型的下限
Cls1<Integer> c1 = new Cls1<Integer>(10);
Cls1<Double> c2 = new Cls1<Double>(10.12);
Cls1<Dog> c3 = new Cls1<Dog>(new Dog());
//Cls1<? extends Animal> c4; //通配符
Cls1<? super Dog> c4;
c4 = c3; //只有这条可以
//c4 = c2;
//c4 = c1;
五、泛型方法
5.1、引入
- 不仅类可以声明泛型,类中的方法也可以声明仅用于自身的泛型,这种方法叫做泛型方法,其定义格式为
访问修饰符<泛型列表>返回类型 方法名(参数列表){
代码
}
- 在泛型列表中声明的泛型,可用于该方法的返回类型声明、参数类型声明和方法代码中的局部变量的类型声明
- 类中其他方法不能使用当前方法声明的泛型
5.2、什么时候使用泛型方法
不希望方法中的数据类型被类所限制
-
添加类型约束只作用于一个方法的多个参数之间,而不涉及到类中的其他方法时
-
施加类型约束的方法为静态方法,只能将其定义为泛型方法,因为静态方法不能使用其所在类的类型参数