避免instanceof非预期结果

instanceof是一个简单的二元操作符,它是用来判断一个对象是否是一个类实例的,其操作类似于>=、==,非常简单,我们来看段程序,代码如下:

 
 
  1. public class Client {  
  2.      public static void main(String[] args) {  
  3.            //String对象是否是Object的实例  
  4.            boolean b1 = "Sting" instanceof Object;  
  5.            //String对象是否是String的实例  
  6.            boolean b2 = new String() instanceof String;  
  7.            //Object对象是否是String的实例  
  8.            boolean b3 = new Object() instanceof String;  
  9.            //拆箱类型是否是装箱类型的实例  
  10.            boolean b4 = 'A' instanceof Character;  
  11.            //空对象是否是String的实例  
  12.            boolean b5 = null instanceof String;  
  13.            //类型转换后的空对象是否是String的实例  
  14.            boolean b6 = (String)null instanceof String;  
  15.            //Date对象是否是String的实例  
  16.         boolean b7 = new Date() instanceof String;  
  17.         //在泛型类中判断String对象是否是Date的实例  
  18.         boolean b8 = new GenericClass<String>().isDateInstance("");  
  19.     }  
  20. }  
  21.  
  22. class GenericClass<T>{  
  23.      //判断是否是Date类型  
  24.      public boolean isDateInstance(T t){  
  25.            return t instanceof Date;  
  26.      }  

就这么一段程序,instanceof的所有应用场景都出现了,同时问题也产生了:这段程序中哪些语句会编译通不过?我们一个一个地来解说。

"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就很正常了。

就这么一个简单的instanceof,你答对几个?

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值