写在前面

相信很多小伙伴都知道局部变量是线程安全的,那你知道为什么局部变量是线程安全的吗?

前言

多个线程同时访问共享变量时,会导致并发问题。那么,如果将变量放在方法内部,是不是还会存在并发问题呢?如果不存在并发问题,那么为什么不会存在并发问题呢?

著名的斐波那契数列

记得上学的时候,我们都会遇到这样一种题目,打印斐波那契数列。斐波那契数列是这样的一个数列:1、1、2、3、5、8、13、21、34...,也就是说第1项和第2项是1,从第3项开始,每一项都等于前2项之和。我们可以使用下面的代码来生成斐波那契数列。

//生成斐波那契数列
public int[] fibonacci(int n){//存放结果的数组int[] result = new int[n];//数组的第1项和第2项为1result[0] = result[1] = 1;//计算第3项到第n项for(int i = 2; i < n; i++){result[i] = result[i-2] + result[i-1];}return result;
}

假设此时有很多个线程同时调用fibonacci()方法来生成斐波那契数列,对于方法中的局部变量result,会不会存在线程安全的问题呢?答案是:不会!!

接下来,我们就深入分析下为什么局部变量不会存在线程安全的问题!

方法是如何被执行的?

我们以下面的三行代码为例。

int x = 5;
int[] y = fibonacci(x);
int[] z = y;

当我们调用fibonacci(x)时,CPU要先找到fibonacci()方法的地址,然后跳转到这个地址去执行代码,执行完毕后,需要返回并找到调用方法的下一条语句的地址,也就是int[] z = y的地址,再跳到这个地址去执行。我们可以将这个过程简化成下图所示。

这里需要注意的是:CPU会通过堆栈寄存器找到调用方法的参数和返回地址。

例如,有三个方法A、B、C,调用关系为A调用B,B调用C。在运行时,会构建出相应的调用栈,我们可以用下图简单的表示这个调用栈。

每个方法在调用栈里都会有自己独立的栈帧,每个栈帧里都有对应方法需要的参数和返回地址。当调用方法时,会创建新的栈帧,并压入调用栈;当方法返回时,对应的栈帧就会被自动弹出。

我们可以这样说:栈帧是在调用方法时创建,方法返回时“消亡”。

局部变量存放在哪里?

局部变量的作用域在方法内部,当方法执行完,局部变量也就没用了。可以这么说,方法返回时,局部变量也就“消亡”了。此时,我们会联想到调用栈的栈帧。没错,局部变量就是存放在调用栈里的。此时,我们可以将方法的调用栈用下图表示。

很多人都知道,局部变量会存放在栈里。如果一个变量需要跨越方法的边界,就必须创建在堆里。

调用栈与线程

两个线程就可以同时用不同的参数调用相同的方法。那么问题来了,调用栈和线程之间是什么关系呢?答案是:每个线程都有自己独立的调用栈。我们可以使用下图来简单的表示这种关系。

此时,我们再看下文中开头的问题:Java方法内部的局部变量是否存在并发问题?答案是不存在并发问题!因为每个线程都有自己的调用栈,局部变量保存在线程各自的调用栈里,不会共享,自然也就不存在并发问题。

线程封闭

方法里的局部变量,因为不会和其他线程共享,所以不会存在并发问题。这种解决问题的技术也叫做线程封闭。官方的解释为:仅在单线程内访问数据。由于不存在共享,所以即使不设置同步,也不会出现并发问题!

特别推荐一个分享架构+算法的优质内容,还没关注的小伙伴,可以长按关注一下:

长按订阅更多精彩▼如有收获,点个在看,诚挚感谢

【高并发】面试官问我:为啥局部变量是线程安全的?相关推荐

  1. 以后面试官问你 为啥不建议使用Select *,请你大声回答他!

    前言 不建议使用 select  *  这几个字眼,做开发的都不陌生吧. 阿里的开发手册上面也是有提到: 昨晚收到一个小兄弟的反馈: (称呼打码了,这是我的隐私,不可能让你们知道的) 随后也问了下学习 ...

  2. 面试官问我为啥B+树一般都不超过3层?3层B+树能存多少数据?redo log与binlog的两阶段提交?

    我今天逛了一下CSDN,又发现了一条显眼的数据,大概是说3层B+树足以容纳2000w条数据.我当时就蒙了,3层对2000w,心想这B+树也太厉害了吧,由此勾起了我求知的欲望,我一定要搞明白他这2000 ...

  3. 【154期】面试官问:请你说说 B 树、B+ 树的原理及区别?

    点击上方"Java精选",选择"设为星标" 别问别人为什么,多问自己凭什么! 下方留言必回,有问必答! 每天 08:35 更新文章,每天进步一点点... 之前在 ...

  4. 【159期】面试官问:说说 MongoDB 批量操作与 MySQL 效率对比?

    点击上方"Java精选",选择"设为星标" 别问别人为什么,多问自己凭什么! 下方留言必回,有问必答! 每天 08:35 更新文章,每天进步一点点... 本文主 ...

  5. 【283期】面试官问:高并发场景下,如何保证全局唯一分布式 ID 生成?

    点击上方"Java精选",选择"设为星标" 别问别人为什么,多问自己凭什么! 下方有惊喜,留言必回,有问必答! 每一天进步一点点,是成功的开始... 前言 系统 ...

  6. 【高并发】关于乐观锁和悲观锁,蚂蚁金服面试官问了我这几个问题!!

    写在前面 最近,一名读者去蚂蚁金服面试,面试官问了他关于乐观锁和悲观锁的问题,幸亏他看了我的[高并发专题]文章,结果是替这名读者高兴!现就部分面试题目总结成文,供小伙伴们参考. 小伙伴们可以关注 冰河 ...

  7. hashmap扩容_面试官问:HashMap在并发情况下为什么造成死循环?一脸懵

    这个问题是在面试时常问的几个问题,一般在问这个问题之前会问Hashmap和HashTable的区别?面试者一般会回答:hashtable是线程安全的,hashmap是线程不安全的. 那么面试官就会紧接 ...

  8. 高并发面试 - 如何设计一个高并发系统?

    高并发面试 - 如何设计一个高并发系统? 面试题 如何设计一个高并发系统? 面试官心理分析 说实话,如果面试官问你这个题目,那么你必须要使出全身吃奶劲了.为啥?因为你没看到现在很多公司招聘的 JD 里 ...

  9. 【059期】面试官问:序列化是什么,为什么要序列化,如何实现?

    >>号外:关注"Java精选"公众号,回复"面试资料",免费领取资料!"Java精选面试题"小程序,3000+ 道面试题在线刷, ...

  10. 面试官问:Kafka 会不会丢消息?怎么处理的?

    点击上方蓝色"方志朋",选择"设为星标" 回复"666"获取独家整理的学习资料! Kafka存在丢消息的问题,消息丢失会发生在Broker, ...

最新文章

  1. 广东计算机好的2a学校,广东专插本2A院校排名情况
  2. abaqus 多层网格绑定_ABAQUS螺栓接触分析
  3. oracle 数据库查询多条数据的一列值
  4. C#深入浅出 关键字(一)
  5. Html Picture
  6. GDI+中发生一般性错误的解决办法(转帖)
  7. 留下考题答案造福我校后来人(考试过后再看,不要抄袭)
  8. python根据相关系数绘制热力图
  9. php钩子配置,thinkphp 行为扩展 钩子与插件的实现
  10. DataSet本地化数据的二表链接操作
  11. java完成登录页面+连接数据库
  12. MT7628学习笔记(4)——固件烧录(TFTP方式)
  13. matlab 读取odb,求教用C++方式读取abaqus的odb数据中的问题!!!
  14. 电脑网络看不到其它计算机,解决网络和共享中看不到其他计算机的问题
  15. 什么是混合云、公有云、私有云?
  16. 【教程】Excel VBA从入门到进阶(蓝色幻想) P6第6集:公式与函数
  17. 面孔“暴露”在外 人脸识别风险谁来“买单”?
  18. ITK 数据表达(图像)
  19. codeforces每日5题(均1500)-第二十一天
  20. 台大李宏毅机器学习课程

热门文章

  1. java gui构造工具_Java Web框架 静态代码块、构造代码块、构造函数、普通代码块 执行顺序 Decompiler JD-GUI 反编译工具...
  2. 深度优先搜索算法(有向图和无向图)
  3. HDU1559(二维树状数组)
  4. 枚举 ---- B. Power Sequence[Codeforces Round #666 (Div. 2)][暴力]
  5. luogu P2596 [ZJOI2006]书架(平衡树、无旋treap(按排名分裂)一些更复杂的操作)
  6. C++ 基础知识总结
  7. dpi shell命令 安卓_android 中 dumpsys 命令使用
  8. mysql报错代码10051_socket error 10061/11004/10053/10051等错误总结
  9. java update set_mybatis update set 多个字段实例
  10. 我在兰亭这三年完结篇之离开