
泛型类与类型参数的基础
在java中,泛型允许我们编写可适用于多种类型的代码,从而提高代码的重用性和类型安全性。例如,我们定义一个泛型类 mygen
class MyGen{ T ObjNum; MyGen( T obj){ ObjNum = obj; } // ... 其他方法 }
这里,T 是一个类型参数,它在 MyGen 类的实例创建时被具体化。例如,MyGen
问题分析:泛型类对象比较中的类型不匹配
考虑一个用于比较两个 Number 绝对值的方法 AbsCompare。初次尝试时,我们可能会这样定义它:
class MyGen{ 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 Objint1 = new MyGen<>(99);
MyGen 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
Objint1 = new MyGen(99); 被创建时,Objint1 的实际类型是 MyGen ,其内部的 T 被具体化为 Integer。 - 因此,方法 boolean AbsCompare(T obj) 实际上变成了 boolean AbsCompare(Integer obj),它期望一个 Integer 类型的参数。
- Objint2 的类型是 MyGen
,而不是 Integer。尽管 MyGen 内部封装了一个 Integer 对象 (ObjNum),但 MyGen 本身和 Integer 是两种完全不同的类型。它们之间没有直接的“is-a”关系(即 MyGen 不是 Integer 的子类型)。 - 同理,Objint1 自身的类型也是 MyGen
,所以将其作为参数传递给期望 Integer 的方法同样会导致类型不匹配。 - Objint3 的类型是 Integer,这与 AbsCompare(Integer obj) 方法期望的参数类型完全匹配,因此它能够正常编译和运行。
尝试解决:修改方法签名引发的新问题
为了解决 MyGen
class MyGen{ T ObjNum; MyGen( T obj){ ObjNum = obj; } // 尝试比较一个MyGen 类型的对象 boolean AbsCompare( MyGen obj){ // 方法签名改变 // 这里的 obj 是 MyGen 类型,而不是 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
public class Sample {
public static void main(String args[]){
MyGen Objint1 = new MyGen<>(99);
Integer Objint3 = 101;
boolean b3 = Objint1.AbsCompare(Objint3); // 编译错误!
}
} 现在 Objint1.AbsCompare(Objint3) 出现了编译错误,因为 AbsCompare 方法现在期望一个 MyGen
此外,在修改后的 AbsCompare(MyGen
解决方案:方法重载(Method Overloading)
要同时支持与内部封装类型 T 进行比较,以及与另一个 MyGen
class MyGen{ 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类型值 * @param myGen 待比较的MyGen 实例 * @return 如果绝对值相等则返回true,否则返回false */ boolean AbsCompare(MyGen myGen){ return Math.abs( ObjNum.doubleValue()) == Math.abs( myGen.ObjNum.doubleValue()); } }
通过这种方式,Java编译器会根据传入参数的实际类型自动选择正确的方法。
完整示例代码:
class MyGen{ 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 实例 boolean AbsCompare(MyGen myGen){ System.out.println("Comparing with MyGen 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 Objint1 = new MyGen<>(99); MyGen Objint2 = new MyGen<>(100); MyGen Objint4 = new MyGen<>(99); // 用于测试相等 // 创建一个Integer类型对象 Integer Objint3 = 99; // 用于测试相等 System.out.println("--- 测试 MyGen 与 MyGen 比较 ---"); boolean b1 = Objint1.AbsCompare(Objint2); // 调用 AbsCompare(MyGen myGen) System.out.println("Objint1 (99) vs Objint2 (100): " + b1); // false boolean b4 = Objint1.AbsCompare(Objint4); // 调用 AbsCompare(MyGen myGen) System.out.println("Objint1 (99) vs Objint4 (99): " + b4); // true System.out.println("\n--- 测试 MyGen 与 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 myGen) // System.out.println("Objint1 (99) vs Objint1 (99): " + b2); // true } }
运行上述代码,所有比较都将正常进行,并且会根据参数类型调用正确的方法。
核心概念:“Has-a” 与 “Is-a” 关系
理解这个问题的关键在于区分面向对象编程中的两种基本关系:
-
“Has-a” 关系(组合/聚合): 表示一个类包含另一个类的实例作为其成员。例如,MyGen
类“has a”一个 T 类型的 ObjNum。这意味着 MyGen 内部有一个 Integer,但它本身并不是一个 Integer。 - “Is-a” 关系(继承): 表示一个类是另一个类的子类型。例如,ArrayList “is an” List。只有当存在“is-a”关系时,子类对象才能被当作父类对象使用(向上转型)。
在我们的例子中,MyGen
虽然理论上如果 MyGen 可以继承 Integer(例如 MyGen extends Integer),那么 MyGen 实例就可以被视为 Integer 实例。但实际上,Integer 是一个 final 类,不能被继承,而且这种继承关系通常与泛型的设计初衷不符。泛型更倾向于通过组合(has-a)来增加灵活性,而不是通过继承。
总结与注意事项
-
类型匹配严格性: Java泛型在编译时会进行严格的类型检查。MyGen
实例与其内部封装的 T 类型是不同的类型,不能互相替代作为方法参数。 - 方法重载是解决方案: 当需要一个方法能够处理多种不同但逻辑相关的参数类型时,方法重载是实现这一目标的标准和推荐方式。
- 理解“Has-a”与“Is-a”: 深入理解这两种关系对于正确设计和使用泛型以及其他面向对象特性至关重要。
- 清晰的方法命名: 尽管本例使用了重载,但在某些复杂场景下,为不同参数类型的方法使用更具描述性的名称(例如 AbsCompareWithValue 和 AbsCompareWithMyGen)可以提高代码的可读性。
通过掌握这些概念,开发者可以更有效地利用Java泛型构建健壮、灵活且类型安全的代码。










