
理解Java泛型类中的类型参数
在java中,泛型类允许我们定义一个类,它能够操作多种数据类型,从而提高代码的重用性和类型安全性。例如,我们定义一个泛型类mygen
class MyGen{ T ObjNum; // 存储一个T类型的对象 MyGen( T obj){ ObjNum = obj; } // 假设我们有一个方法用于比较 // boolean AbsCompare(T Obj) { ... } }
这里,MyGen是一个泛型包装类,它“拥有”(has-a)一个T类型的对象ObjNum。当实例化MyGen
方法参数类型不匹配的困惑
考虑以下场景,我们希望MyGen类能够比较其内部封装的T类型值,或者与另一个MyGen
首先,定义一个AbsCompare方法,期望参数类型为T:
// 在MyGen类中
boolean AbsCompare( T obj){
// 比较当前对象的ObjNum与传入参数obj的绝对值
if( Math.abs( ObjNum.doubleValue()) == Math.abs( obj.doubleValue()))
return true;
else
return false;
}然后在main方法中尝试调用:
立即学习“Java免费学习笔记(深入)”;
class Sample{
public static void main(String args[]){
MyGen Objint1= new MyGen<>( 99);
MyGen Objint2= new MyGen<>( 100 ); // 另一个MyGen实例
Integer Objint3=101; // 一个普通的Integer对象
// 尝试使用AbsCompare方法进行比较
boolean b1= Objint1.AbsCompare( Objint2); // 编译错误!
boolean b2= Objint1.AbsCompare( Objint1); // 编译错误!
boolean b3= Objint1.AbsCompare( Objint3) ; // 编译通过!
}
} 为什么Objint1.AbsCompare(Objint2)和Objint1.AbsCompare(Objint1)会报错,而Objint1.AbsCompare(Objint3)却能正常编译?
答案在于类型匹配。当Objint1被声明为MyGen
- Objint3是一个Integer对象,完美匹配T(即Integer),所以b3的调用是正确的。
- Objint1和Objint2都是MyGen
类型的对象,它们不是Integer类型。MyGen 与Integer之间没有直接的“is-a”关系(继承关系),它们是两种完全不同的类型。因此,将一个MyGen 对象传递给期望Integer参数的方法会导致编译错误。
尝试修改方法签名引发的新问题
有些开发者可能会尝试修改AbsCompare方法的签名,使其直接接受一个MyGen
// 在MyGen类中 boolean AbsCompare( MyGenobj) { // 方法签名改为接受MyGen // 比较当前对象的ObjNum与传入参数obj的内部ObjNum的绝对值 if(Math.abs(ObjNum.doubleValue()) == Math.abs(obj.doubleValue())) // 编译错误!obj.doubleValue() return true; else return false; }
现在,main方法中的调用行为发生了变化:
class Sample{
public static void main(String args[]){
MyGen Objint1= new MyGen<>( 99);
MyGen Objint2= new MyGen<>( 100 );
Integer Objint3=101;
boolean b1= Objint1.AbsCompare( Objint2); // 编译通过!
boolean b2= Objint1.AbsCompare( Objint1); // 编译通过!
boolean b3= Objint1.AbsCompare( Objint3) ; // 编译错误!
}
} 这次,Objint1.AbsCompare(Objint2)和Objint1.AbsCompare(Objint1)能够编译通过,因为它们都传入了MyGen
然而,Objint1.AbsCompare(Objint3)现在报错了,因为Objint3是一个Integer,而不是MyGen
更重要的是,在新的AbsCompare(MyGen
解决方案:方法重载(Overloading)
为了同时支持与内部封装类型T的比较,以及与另一个泛型包装类MyGen
class MyGen{ T ObjNum; MyGen( T obj){ ObjNum = obj; } /** * 方法1: 比较当前对象的ObjNum与传入的T类型参数的绝对值 * @param obj 待比较的T类型对象 * @return 绝对值是否相等 */ public boolean AbsCompare( T obj){ return Math.abs( ObjNum.doubleValue()) == Math.abs( obj.doubleValue()); } /** * 方法2: 比较当前对象的ObjNum与传入的MyGen 对象的内部ObjNum的绝对值 * @param myGen 待比较的MyGen 实例 * @return 绝对值是否相等 */ public boolean AbsCompare(MyGen myGen){ // 注意:这里需要访问传入MyGen对象的内部ObjNum return Math.abs(ObjNum.doubleValue()) == Math.abs(myGen.ObjNum.doubleValue()); } }
现在,main方法中的所有调用都将正常工作:
class Sample{
public static void main(String args[]){
MyGen Objint1= new MyGen<>( 99);
MyGen Objint2= new MyGen<>( 100 );
Integer Objint3=101;
// 调用AbsCompare(MyGen myGen)
boolean b1= Objint1.AbsCompare( Objint2); // 编译通过,调用重载方法1
boolean b2= Objint1.AbsCompare( Objint1); // 编译通过,调用重载方法1
// 调用AbsCompare(T obj)
boolean b3= Objint1.AbsCompare( Objint3) ; // 编译通过,调用重载方法2
System.out.println("b1: " + b1); // 输出 false
System.out.println("b2: " + b2); // 输出 false (99 vs 99)
System.out.println("b3: " + b3); // 输出 false
}
} 核心概念总结
-
“Has-a” vs. “Is-a” 关系:
- MyGen
与T之间是“has-a”(拥有)关系,即MyGen对象内部含有一个T类型的对象。MyGen 不是一个Integer。 - “is-a”关系通常指继承关系。例如,Integer“is-a”Number,所以可以将Integer传递给期望Number参数的方法。
- MyGen
-
方法重载(Overloading):
- 允许在同一个类中定义多个同名但参数列表(参数类型、参数数量或参数顺序)不同的方法。
- 编译器会根据调用时提供的实际参数类型和数量,自动选择最匹配的重载方法。这是解决上述类型匹配问题的标准做法。
-
泛型类型安全:
- Java泛型在编译时提供类型检查,确保只有符合类型约束的对象才能被传递和操作。这种严格的类型检查机制帮助开发者在早期发现潜在的类型错误。
通过理解这些核心概念,并恰当地运用方法重载,我们可以编写出既灵活又类型安全的泛型代码,有效处理不同参数类型的调用需求。










