泛型用法
泛型类/接口:在类/接口后增加关键字"<泛型标识>",申明后该类/接口的具体方法可以使用此标识作为,入参/出参的类型。
泛型方法:在方法访问修饰符后增加关键字"<泛型标识>"。
泛型标识:可以是任意字符串,通常使用单个大写字母表示. 可以有多个泛型标识代表不同参数
泛型作用:
编译前编译工具检查
运行过程中自动转换类型
下面主要说明泛型在继承场景下的使用。
父类与接口
interface GenericityInterface<T> { String getString(T t);}abstract class GenericityClass<A extends Collection, B> { abstract String getString(A a, B b);}
子类/实现类可使用@Override让编译工具检查是否重写了父类中方法,否则可能是重载了父类方法, 导致在使用父类/接口封装业务时出现偏差。
沿用所有泛型
class SubGenericity1<T, B, A extends Collection> extends GenericityClass<A, B> implements GenericityInterface<T> { @Override public String getString(T o) { return o.getClass().toString(); } @Override String getString(A a, B o) { return o.getClass().toString() + "|" + o.getClass(); }}
GenericityInterface/GenericityClass 中泛型标识符可以使用任意字符,但个数不能大于实际个数。
SubGenericity1<T, B, A extends Collection>,表示当前类使用三个泛型标识符,T,B,A,标识需要和继承的类/接口保持一致 当父类泛型存在限制时子类的泛型需要小于或等于父类中泛型范围,例如父类范围是Collection,子类可以是List,但反过来不行。
合并泛型
class SubGenericity2<T extends Collection, B> extends GenericityClass<T, B> implements GenericityInterface<T> { @Override public String getString(T o) { return o.getClass().toString(); } @Override String getString(T a, B o) { return o.getClass().toString() + "|" + o.getClass(); }}class SubGenericity21<T extends Collection> extends GenericityClass<T, T> implements GenericityInterface<T> { @Override public String getString(T o) { return o.getClass().toString(); } @Override String getString(T a, T o) { return o.getClass().toString() + "|" + o.getClass(); }}
GenericityClass<T, T>:从上述代码可知泛型申明类似与占位符,只要与父类个数相同即可,标识符一致表示是同一个泛型约束
不使用泛型
class SubGenericity3 extends GenericityClass implements GenericityInterface { @Override public String getString(Object o) { return o.getClass().toString(); } @Override String getString(Collection collection, Object o) { return o.getClass().toString() + "|" + o.getClass(); }}
继承时不写泛型则表示不继承父类/接口中泛型约束,覆盖方法时使用Object代替了泛型约束,类型擦除的一种体现
部分使用泛型
class SubGenericity4<T> extends GenericityClass implements GenericityInterface<T> { @Override public String getString(T o) { return o.getClass().toString(); } @Override String getString(Collection collection, Object o) { return o.getClass().toString() + "|" + o.getClass(); }}class SubGenericity5<T extends Collection,B> extends GenericityClass<T,B> implements GenericityInterface { @Override public String getString(Object o) { return o.getClass().toString(); } @Override String getString(T collection, B o) { return o.getClass().toString() + "|" + o.getClass(); }}
继承时指定类型
class SubGenericity6 extends GenericityClass<List,String> implements GenericityInterface<Map> { @Override public String getString(Map o) { return o.getClass().toString(); } @Override String getString(List collection,String o) { return o.getClass().toString() + "|" + o.getClass(); }}
通常使用此模式使用泛型
其它-获取运行时类型
class Test<T>{ public Class getActualClass(){ ParameterizedType type = (ParameterizedType) this.getClass().getGenericSuperclass(); return (Class) type.getActualTypeArguments()[0]; }}
在子类需要沿用泛型,且需要创建泛型对应对象时会使用此方法, 其中type.getActualTypeArguments()返回的是该类申明的所有泛型,可能存在多个。
参考资料
https://docs.oracle.com/javase/tutorial/java/generics/types.html
原文:https://juejin.cn/post/7101569979446722568