建议18: 避免instanceof非预期结果
instanceof是用来判断一个对象是否是一个类的实例。
public class Client {
public static void main(String[] args) {
//String对象是否是Object的实例
boolean b1 = "Sting" instanceof Object;
//String对象是否是String的实例
boolean b2 = new String() instanceof String;
//Object对象是否是String的实例
boolean b3 = new Object() instanceof String;
//拆箱类型是否是装箱类型的实例
boolean b4 = 'A' instanceof Character;
//空对象是否是String的实例
boolean b5 = null instanceof String;
//类型转换后的空对象是否是String的实例
boolean b6 = (String)null instanceof String;
//Date对象是否是String的实例
boolean b7 = new Date() instanceof String;
//在泛型类中判断String对象是否是Date的实例
boolean b8 = new GenericClass<String>().isDateInstance("");
}
}
class GenericClass<T>{
//判断是否是Date类型
public boolean isDateInstance(T t){
return t instanceof Date;
}
}
答案:
- "Sting"instanceof Object
返回值是true,这很正常,“String"是一个字符串,字符串又继承Object,那当然是返回true了。 - new String() instanceof String
返回值是true,没有任何问题,一个类的对象当然是它的实例了。 - new Object() instanceof String
返回值是false,Object是父类,其对象当然不是String类的实例了。要注意的是,这句话其实完全可以编译通过,只要instanceof关键字的左右两个操作数有继承或实现关系,就可以编译通过。 - 'A' instanceof Character
编译不通过。因为'A'是一个char类型,也就是一个基本类型,不是一个对象,instanceof只能用于对象的判断,不能用于基本类型的判断。 - null instanceof String
返回值是false,这是instanceof特有的规则:若左操作数是null,结果就直接返回false,不再运算右操作数是什么类。这对我们的程序非常有利,在使用instanceof操作符时,不用关心被判断的类(也就是左操作数)是否为null,这与我们经常用到的equals、toString方法不同。 - (String)null instanceof String
返回值是false,不要看这里有个强制类型转换就认为结果是true,不是的,null是一个万用类型,也可以说它没类型,即使做类型转换还是null。 - new Date() instanceof String
编译通不过,因为Date类和String没有继承或实现关系,所以在编译时直接就报错了,instanceof操作符的左右操作数必须有继承或实现关系,否则编译会失败。 - new GenericClass<String>().isDateInstance("")
编译通过了,返回值是false,T是个String类型,与Date之间没有继承或实现关系,为什么''t instanceof Date''会编译通过呢?那是因为Java的泛型是为编码服务的,在编译成字节码时,T已经是Object类型了,传递的实参是String类型,也就是说T的表面类型是Object,实际类型是String,那''t instanceof Date''这句话就等价于''Object instance of Date''了,所以返回false就很正常了。