小心踩坑!s1 = s1 + 1 和 s1 += 1 的区别你真的懂吗?

简介: 小米,一位29岁的技术爱好者,通过一个Java面试题解析了隐式类型转换与赋值运算符特性的知识点。题目涉及`short s1 = 1; s1 = s1 + 1;`与`short s1 = 1; s1 += 1;`的区别,前者因类型不匹配报错,后者则因`+=`运算符的隐式类型转换功能而成功编译。小米通过故事化讲解,深入浅出地解释了Java中数据类型转换的规则及其背后的逻辑,帮助读者更好地理解和记忆这一知识点。



Hello,大家好!我是你们的老朋友小米,一个每天都在技术世界里快乐遨游的“29岁技术爱好者”。最近啊,我收到一位粉丝私信,提了一个Java面试题,问题乍一看特别简单,但别小看它,背后暗藏玄机!

问题是这样的:

面试官问:“这段代码能正常编译运行吗?如果有错,是为什么?如果没错,为什么?”

哈哈,看到这题,许多人可能第一反应是:“有啥区别?不都一样加1嘛!” 其实,这里暗藏了Java中隐式类型转换赋值运算符特性的知识点。接下来,我就通过讲故事的方式,手把手带你理解清楚这道题背后的玄机!

故事时间:数据类型的大小争执

从前有个小村子,叫Java类型村。村里住着八个小伙伴,分别是:byte、short、int、long、float、double、char、boolean。他们每天生活得很开心,但是有一天,他们吵了起来。

吵什么呢?他们在争论“谁的容量最大”。

  • byte最小,只有1个字节。
  • short有2个字节,觉得自己比byte强。
  • int有4个字节,自认为是默认哥(默认整数类型就是int)。
  • long更厉害,8个字节,觉得自己称霸村子。
  • float和double不甘示弱,他们虽然是小数类型,但也爱参与这种“争夺战”……

最后,村长Java编译器出面调解,说:“从小到大是这样的——byte < short < int < long < float < double,你们谁也别吵了!”

第一幕:short s1 = 1; s1 = s1 + 1; 出现了问题

好,回到我们的代码,先看这行:

让我们一行一行拆解分析:

1. short s1 = 1;

这句代码很简单,定义了一个short类型的变量s1,并赋值为1。

short的范围是-32,768到32,767,1当然在这个范围内,所以没问题。

2. s1 = s1 + 1;

好戏来了!这句代码里,s1 + 1 是什么类型?

这里涉及到一个Java中的运算规则

  • 在Java中,所有参与**算术运算(+、-、*、/)**的变量,最小都会被提升为int类型
  • 即使是byte和short这样的“小家伙”,一旦参与运算,也会临时变成int类型!

所以,在这里:

  • s1(short类型)先被“偷偷”转成了int。
  • 然后1也是int。
  • 所以,s1 + 1 的结果是一个 int!

最后,你试图把这个 int 类型的结果,赋值给 short s1。

这就问题大了!

Java是强类型语言,int类型的值不能直接赋值给short,因为可能会有数据丢失,编译器不允许这种危险操作。

因此,这段代码会报错,错误信息类似于:

解决方法呢?可以通过显式强制类型转换,把int转成short:

但需要注意的是,这种强制类型转换,可能会导致数据溢出问题

第二幕:short s1 = 1; s1 += 1; 为什么没问题?

再看第二段代码:

乍一看,这不就是“换了个写法”吗?凭什么这一段就能正常运行?

答案就在+=运算符的“特权”上!

1. += 是什么?

+= 是 复合赋值运算符,它本质上包含了两个操作:

  • 加法运算:s1 + 1。
  • 隐式强制类型转换:把结果自动转换成左侧变量的类型。

换句话说,s1 += 1 这个表达式的完整执行过程是:

看到没?+= 运算符自带“隐式强制类型转换”的功能,不需要我们手动加(short),编译器会帮我们自动处理。

因此,第二段代码可以顺利通过编译,也不会报错。

第三幕:面试官为什么爱问这题?

这道题虽然简单,但它考察了多个Java的基础点:

  • 数据类型的大小关系:byte < short < int < long,默认整型是int。
  • 算术运算的隐式类型提升:byte和short参与运算会被提升为int。
  • 复合赋值运算符的特性:自动进行类型转换。

如果不理解这些基础知识,面试时一慌,很容易说错。

总结:如何回答这道题?

如果你在面试中遇到这道题,可以这样回答:

  • 第一段代码 short s1 = 1; s1 = s1 + 1; 会报错,因为 s1 + 1 的结果是 int 类型,不能直接赋值给 short。
  • 第二段代码 short s1 = 1; s1 += 1; 没问题,因为 += 运算符自带隐式类型转换,会自动把结果转换成 short 类型。

加分项可以补充说明:

  • Java是强类型语言,不允许隐式类型转换可能导致数据丢失。
  • += 的底层实现相当于强制类型转换,因此编译能通过。

END

这道题看似简单,其实很有深度。理解它不仅能加深你对Java基础的掌握,还能帮助你在实际编码中避免类似错误。

如果你觉得我的解答对你有帮助,记得 “三连”,也欢迎分享给你的小伙伴!我是小米,下一次,我们继续聊聊Java的那些“迷惑行为”。

我是小米,一个喜欢分享技术的29岁程序员。如果你喜欢我的文章,欢迎关注我的微信公众号软件求生,获取更多技术干货!

相关文章
|
Java 数据库连接 数据库
源码分析系列教程(完) - 终章总结
源码分析系列教程(完) - 终章总结
73 0
源码分析系列教程(完) - 终章总结
|
消息中间件 缓存 安全
讲理论,重实战!阿里独家SpringBoot王者晋级之路小册,太强了!
大家平时学习SpringBoot的方式也一般是看大量博客或者是找一些业界评价好点的书籍,虽然SpringBoot相关资料很多,但是大多不成体系,很少有真正有能从0到1,详解Spring Boot一切从代码案例出发的案头笔记。 今天给小伙伴分享的就是来自阿里的SpringBoot王者晋级之路小册,这份小册从SpringBoot的开发环境部署开始,把Spring Boot搭建Web项目、操作数据库、使用缓存、日志、整合安全框架、结合消息队列和搜索框架,以及在实际应用中的部署全部讲得清清楚楚。
|
10月前
|
SpringCloudAlibaba Java Nacos
前阿里P8甩我一份内部SpringCloud笔记手册,真香
SpringCloud Alibaba 因配置灵活、迭代高速、兼容性强,近几年在受到国内不少开发者的广泛关注。其中,Sentinel 作为服务治理开源组件相当出彩,可以帮助解决很多难题,比如:秒杀限流,消息削峰填谷,集群流量控制,实时熔断等。而作为国内的Spring爱好者,最最苦恼的莫过于操作性强的进阶干货太少了!
|
10月前
|
前端开发 JavaScript 开发者
30分钟熟练使用最常用的ES6,还不学是等着被卷死?
30分钟熟练使用最常用的ES6,还不学是等着被卷死?
|
10月前
|
XML Java 数据格式
🚀今天,我们来详细的聊一聊SpringBoot自动配置原理,学了这么久,你学废了吗?
🚀今天,我们来详细的聊一聊SpringBoot自动配置原理,学了这么久,你学废了吗?
134 0
|
SQL 安全 前端开发
Web安全性测试包括哪些要点?梳理下,总算搞明白了
Web安全性测试包括哪些要点?梳理下,总算搞明白了
488 0
Web安全性测试包括哪些要点?梳理下,总算搞明白了
|
Windows
谈一谈|下载软件的门道你懂吗?
谈一谈|下载软件的门道你懂吗?
171 0
|
算法 NoSQL API
到底该不该看源码(懂这三点儿就够了)
1、不要为了看源码而看源码 2、代码积累到一定程度,遇到问题自然就去查源码了,然后你就看懂了 3、两年内不要刻意去看源码,可以点开简单了解一下就行,前两年疯狂做项目就行了,后期项目做的多了,你自己就会有疑问,每次写代码就会问自己为什么要这样写?底层的原理是什么?很自觉的带着问题就去看源码了,如果你没有这样的疑问,那说明你也不适合去看源码了,写写业务代码,了了一生
222 0
并发程序设计,你真的懂吗?
并发程序设计,你真的懂吗?
109 0
并发程序设计,你真的懂吗?
|
Java 程序员 Linux
面经手册 · 第24篇《为了搞清楚类加载,竟然手撸JVM!》
写个代码加载下 1. 案例工程 2. 代码讲解 五、解析字节码文件 1. 提取部分字节码 2. 解析魔数并校验 3. 解析版本号信息 4. 解析全部内容对照
172 0
面经手册 · 第24篇《为了搞清楚类加载,竟然手撸JVM!》