教妹学 Java 第 24 讲:instanceof 操作符

沉默王二

共 3042字,需浏览 7分钟

 ·

2021-04-23 16:50

嗨,《教妹学 Java》专栏的同学们,大家好呀,久违了!

连更 23 讲后,感觉实在是太累了!几乎每天都在写文章,再加上工作和生活,完全是一副丧尸的模样,扛不住。于是就不得不停下来脚步,结果这一停,直接停到黄花菜都凉了,很多喜欢这个专栏的同学在后台不停地催啊催!

催催催,于是第 24 讲——instanceof 操作符就来了!这说明催催还是有效果的哈。

“哥,我终于该上场了呀!”三妹调皮的说,“这段时间,我可急坏了!”

“我的亲妹妹呀,作为补偿,五一我带你去张家界玩,怎么样?到时候再给哥拍点帅照作为封面,还有你的,主要是作为同学们的福利~”我瞥了一眼旁边的三妹,一听到我要带她出去玩,脸上立马是一副期待的样子。

“那哥,你赶紧开始吧,我要学起来了!”


instanceof 操作符的用法其实很简单:

(object) instanceof (type)

用意也非常简单,判断对象是否符合指定的类型,结果要么是 true,要么是 false。在反序列化的时候,instanceof 操作符还是蛮常用的,因为这时候我们不太确定对象属不属于指定的类型,如果不进行判断的话,就容易抛出 ClassCastException 异常。

我们来建这样一个简单的类 Round:

class Round {
}

然后新增一个扩展类 Ring:

class Ring extends Round {
}

这时候,我们就可以通过 instanceof 来检查 Ring 对象是否属于 Round 类型。

Ring ring = new Ring();
System.out.println(ring instanceof Round);

结果会输出 true,因为 Ring 继承了 Round,也就意味着 Ring 和 Round 符合 is-a 的关系,而 instanceof 操作符正是基于类与类之间的继承关系,以及类与接口之间的实现关系的。

我们再来新建一个接口 Shape:

interface Shape {
}

然后新建 Circle 类实现 Shape 接口并继承 Round 类:

class Circle extends Round implements Shape {
}

如果对象是由该类创建的,那么 instanceof 的结果肯定为 true。

Circle circle = new Circle();
System.out.println(circle instanceof Circle);

这个肯定没毛病,instanceof 就是干这个活的,大家也很好理解。那如果类型是父类呢?

System.out.println(circle instanceof Round);

结果肯定还是 true,因为依然符合 is-a 的关系。那如果类型为接口呢?

System.out.println(circle instanceof Shape);

结果仍然为 true, 因为也符合 is-a 的关系。如果要比较的对象和要比较的类型之间没有关系,当然是不能使用 instanceof 进行比较的。

为了验证这一点,我们来创建一个实现了 Shape 但与 Circle 无关的 Triangle 类:

class Triangle implements Shape {
}

这时候,再使用 instanceof 进行比较的话,编译器就报错了。

 System.out.println(circle instanceof Triangle);

错误信息如下所示:

Inconvertible types; cannot cast 'com.itwanger.twentyfour.instanceof1.Circle' to 'com.itwanger.twentyfour.instanceof1.Triangle'

意思就是类型不匹配,不能转换,我们使用 instanceof 比较的目的,也就是希望如果结果为 true 的时候能进行类型转换。但显然 Circle 不能转为 Triangle。

编译器已经提前帮我们预知了,很聪明。

Java 是一门面向对象的编程语言,也就意味着除了基本数据类型,所有的类都会隐式继承 Object 类。所以下面的结果肯定也会输出 true。

Thread thread = new Thread();
System.out.println(thread instanceof Object);

“那如果对象为 null 呢?”三妹这时候插话了。

“这个还真的是一个好问题啊。”我忍不住对三妹竖了一个大拇指。

System.out.println(null instanceof Object);

只有对象才会有 null 值,所以编译器是不会报错的,只不过,对于 null 来说,instanceof 的结果为 false。因为所有的对象都可以为 null,所以也不好确定 null 到底属于哪一个类。

通常,我们是这样使用 instanceof 操作符的。

// 先判断类型
if (obj instanceof String) {
    // 然后强制转换
    String s = (String) obj;
    // 然后才能使用
}

先用 instanceof 进行类型判断,然后再把 obj 强制转换成我们期望的类型再进行使用。

JDK 16 的时候,instanceof 模式匹配转了正,意味着使用 instanceof 的时候更便捷了。

if (obj instanceof String s) {
    // 如果类型匹配 直接使用 s
}

可以直接在 if 条件判断类型的时候添加一个变量,就不需要再强转和声明新的变量了。

“哇,这样就简洁了呀!”三妹不仅惊叹到!

好了,关于 instanceof 操作符我们就先讲到这吧,难是一点都不难,希望各位同学也能够很好的掌握。

至于下一篇什么时候会写,大家多催催就是了。这周打算把对应的 GitHub 和码云,以及 docsify 都重新更新一波~敬请期待。

浏览 5
点赞
评论
收藏
分享

手机扫一扫分享

分享
举报
评论
图片
表情
推荐
点赞
评论
收藏
分享

手机扫一扫分享

分享
举报