
在java中,泛型允许我们编写可适用于多种类型的代码,从而提高代码的重用性和类型安全性。例如,我们定义一个泛型类 mygen<t extends number>,它封装了一个 number 类型的对象 objnum:
class MyGen <T extends Number> {
T ObjNum;
MyGen( T obj){
ObjNum = obj;
}
// ... 其他方法
}这里,T 是一个类型参数,它在 MyGen 类的实例创建时被具体化。例如,MyGen<Integer> 表示 ObjNum 将是一个 Integer 类型。
考虑一个用于比较两个 Number 绝对值的方法 AbsCompare。初次尝试时,我们可能会这样定义它:
class MyGen <T extends Number> {
T ObjNum;
MyGen( T obj){
ObjNum = obj;
}
// 尝试比较一个T类型的对象
boolean AbsCompare( T obj){
if( Math.abs( ObjNum.doubleValue()) == Math.abs( obj.doubleValue()))
return true;
else
return false;
}
}现在,我们来看一个使用该方法的示例:
public class Sample {
public static void main(String args[]){
MyGen <Integer> Objint1 = new MyGen<>(99);
MyGen <Integer> Objint2 = new MyGen<>(100);
// 创建一个Integer类型对象
Integer Objint3 = 101;
// 使用方法比较对象
boolean b1 = Objint1.AbsCompare(Objint2); // 编译错误!
boolean b2 = Objint1.AbsCompare(Objint1); // 编译错误!
boolean b3 = Objint1.AbsCompare(Objint3); // 正常编译运行
}
}令人困惑的是,为什么 Objint1.AbsCompare(Objint2) 和 Objint1.AbsCompare(Objint1) 会出现编译错误,而 Objint1.AbsCompare(Objint3) 却正常?
立即学习“Java免费学习笔记(深入)”;
原因在于类型不匹配:
为了解决 MyGen<Integer> 对象之间的比较问题,我们可能会尝试修改 AbsCompare 方法的签名,使其接受一个 MyGen<T> 类型的参数:
class MyGen <T extends Number> {
T ObjNum;
MyGen( T obj){
ObjNum = obj;
}
// 尝试比较一个MyGen<T>类型的对象
boolean AbsCompare( MyGen<T> obj){ // 方法签名改变
// 这里的 obj 是 MyGen<T> 类型,而不是 T
// 所以 obj.doubleValue() 会报错,应该访问其内部的 ObjNum
if( Math.abs( ObjNum.doubleValue()) == Math.abs( obj.ObjNum.doubleValue()))
return true;
else
return false;
}
}现在,Objint1.AbsCompare(Objint2) 和 Objint1.AbsCompare(Objint1) 可以正常编译了,因为参数类型 MyGen<T> 与 MyGen<Integer> 匹配。然而,新的问题出现了:
public class Sample {
public static void main(String args[]){
MyGen <Integer> Objint1 = new MyGen<>(99);
Integer Objint3 = 101;
boolean b3 = Objint1.AbsCompare(Objint3); // 编译错误!
}
}现在 Objint1.AbsCompare(Objint3) 出现了编译错误,因为 AbsCompare 方法现在期望一个 MyGen<Integer> 类型的参数,而 Objint3 是 Integer 类型,再次造成了类型不匹配。
此外,在修改后的 AbsCompare(MyGen<T> obj) 方法内部,如果仍然尝试使用 obj.doubleValue(),也会导致编译错误。因为 obj 现在是 MyGen<T> 类型,它没有 doubleValue() 方法。我们必须通过 obj.ObjNum.doubleValue() 来访问其内部封装的 T 对象的 doubleValue() 方法。
要同时支持与内部封装类型 T 进行比较,以及与另一个 MyGen<T> 实例进行比较,最简洁和类型安全的方法是使用方法重载。我们可以定义两个具有相同名称但参数类型不同的 AbsCompare 方法:
class MyGen <T extends Number> {
T ObjNum;
MyGen( T obj){
ObjNum = obj;
}
/**
* 比较当前对象内部封装的T类型值与另一个T类型值
* @param obj 待比较的T类型对象
* @return 如果绝对值相等则返回true,否则返回false
*/
boolean AbsCompare( T obj){
return Math.abs( ObjNum.doubleValue()) == Math.abs( obj.doubleValue());
}
/**
* 比较当前对象内部封装的T类型值与另一个MyGen<T>实例内部封装的T类型值
* @param myGen 待比较的MyGen<T>实例
* @return 如果绝对值相等则返回true,否则返回false
*/
boolean AbsCompare(MyGen<T> myGen){
return Math.abs( ObjNum.doubleValue()) == Math.abs( myGen.ObjNum.doubleValue());
}
}通过这种方式,Java编译器会根据传入参数的实际类型自动选择正确的方法。
完整示例代码:
class MyGen <T extends Number> {
T ObjNum;
MyGen( T obj){
ObjNum = obj;
}
// 方法1: 比较与内部封装类型T相同的对象
boolean AbsCompare( T obj){
System.out.println("Comparing with T type: " + obj);
return Math.abs( ObjNum.doubleValue()) == Math.abs( obj.doubleValue());
}
// 方法2: 比较与另一个MyGen<T>实例
boolean AbsCompare(MyGen<T> myGen){
System.out.println("Comparing with MyGen<T> type: " + myGen.ObjNum);
return Math.abs( ObjNum.doubleValue()) == Math.abs( myGen.ObjNum.doubleValue());
}
// 为了更好的输出,可以重写toString
@Override
public String toString() {
return "MyGen{" + "ObjNum=" + ObjNum + '}';
}
}
public class Sample {
public static void main(String args[]){
MyGen <Integer> Objint1 = new MyGen<>(99);
MyGen <Integer> Objint2 = new MyGen<>(100);
MyGen <Integer> Objint4 = new MyGen<>(99); // 用于测试相等
// 创建一个Integer类型对象
Integer Objint3 = 99; // 用于测试相等
System.out.println("--- 测试 MyGen<T> 与 MyGen<T> 比较 ---");
boolean b1 = Objint1.AbsCompare(Objint2); // 调用 AbsCompare(MyGen<T> myGen)
System.out.println("Objint1 (99) vs Objint2 (100): " + b1); // false
boolean b4 = Objint1.AbsCompare(Objint4); // 调用 AbsCompare(MyGen<T> myGen)
System.out.println("Objint1 (99) vs Objint4 (99): " + b4); // true
System.out.println("\n--- 测试 MyGen<T> 与 T 比较 ---");
boolean b3 = Objint1.AbsCompare(Objint3); // 调用 AbsCompare(T obj)
System.out.println("Objint1 (99) vs Objint3 (99): " + b3); // true
// 也可以自己与自己比较,但通常意义不大
// boolean b2 = Objint1.AbsCompare(Objint1); // 这会调用 AbsCompare(MyGen<T> myGen)
// System.out.println("Objint1 (99) vs Objint1 (99): " + b2); // true
}
}运行上述代码,所有比较都将正常进行,并且会根据参数类型调用正确的方法。
理解这个问题的关键在于区分面向对象编程中的两种基本关系:
在我们的例子中,MyGen<Integer> 和 Integer 之间是“has-a”关系,而不是“is-a”关系。因此,Java的类型系统不允许我们将 MyGen<Integer> 实例作为 Integer 类型的参数传递给方法,反之亦然,除非通过显式的方法重载来处理这些不同的类型。
虽然理论上如果 MyGen 可以继承 Integer(例如 MyGen extends Integer),那么 MyGen 实例就可以被视为 Integer 实例。但实际上,Integer 是一个 final 类,不能被继承,而且这种继承关系通常与泛型的设计初衷不符。泛型更倾向于通过组合(has-a)来增加灵活性,而不是通过继承。
通过掌握这些概念,开发者可以更有效地利用Java泛型构建健壮、灵活且类型安全的代码。
以上就是理解Java泛型类型参数与方法重载:解决泛型类对象比较中的类型不匹配问题的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号