在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方法中尝试调用:
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
有些开发者可
能会尝试修改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
为了同时支持与内部封装类型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
}
} 通过理解这些核心概念,并恰当地运用方法重载,我们可以编写出既灵活又类型安全的泛型代码,有效处理不同参数类型的调用需求。