美文网首页
方法中定义的局部变量是否线程安全?

方法中定义的局部变量是否线程安全?

作者: YAOPRINCESS | 来源:发表于2020-06-30 15:53 被阅读0次

方法中定义的局部变量是否线程安全?

了解过JVM的都知道,堆和方法区(JDK1.8后叫元空间)是线程共享的

虚拟机栈、程序计数器(PC寄存器)和本地方法栈是线程私有的

虚拟机栈里面存的是一个一个的栈帧,这里的栈帧你也可以理解为此时正在运行的方法,如果此方法调用其他方法的话,在虚拟机栈中就形成了一个个栈帧堆叠的形态,相信读到这里你已经对虚拟机栈的样子有了感觉

接着,我们讨论栈帧(方法

栈帧有五个部分组成

  • 局部变量表
  • 操作数栈
  • 动态链接【方法的符号引用,在这里我们可以讨论虚方法(在运行时确定方法,把符号引用转为直接引用)和非虚方法(在编译时确定方法,把符号引用转为直接引用)】这里有很多可以讨论的,静态链接和动态链接,虚方法表等
  • 方法返回地址(PC寄存器中的值)
  • 其他

在这里讲栈帧的组成只是为了给不熟悉的同学涨涨知识或者给大家复习一波。

接着,我们回到题目

你会发现方法区中定义的局部变量不就是存在于局部变量表中嘛,而局部变量表有存在于栈帧,栈帧存在于虚拟机栈,那它不就是线程私有的,也就是安全的嘛!

其实,这里面有些坑,看完下面这四个例子我相信你会恍然大悟!

//s1的声明方式是线程安全的
public static void method1(){
    //StringBuffer:线程不安全的
    StringBuffer s1 = new StringBuffer();
    s1.append("A");
    s1.append("B");
    ...
}

解释:
这个例子是线程安全的,s1为局部变量,返回值为void,只能被当前线程操作,是线程安全的。
注意:这里返回值为空(void)

注意:把StringBudiler作为参数传进去

//先说结论,这个是线程不安全的
//StringBuilder是线程不安全的
public static void method2(StringBuiler sBuilder){
    sBuilder.append("A");
    sBuilder.append("B");
    ...
}

//我们在main操作一下
public static void main(String[] args){
    StringBuilder s = new StringBuilder();
    
    new Thread(()->{
      s.append("a");
      s.append("b");
    }).start();
    
    method2(s);    
    
}

//它们会抢s的资源,是线程不安全的
//严格的来说sBuilder不是方法内的局部变量,它是形参的局部变量,形参也会存在局部变量表中

这次不传参数,而是返回

//结论:不是线程安全的(有可能存在问题)
public static StringBuilder method3(){
    StringBuilder s1 = new StringBuilder();
    s1.append("A");
    s1.append("B");
    ...
    return s1;
}

//引用类型和基本类型不用多说了把
//基本类型包括:byte,short,int,long,char,float,double,Boolean,
//引用类型包括:类类型,接口类型和数组。
//因为StringBuilder是类,一返回出去可能被其他位置上的多个线程所调用

返回String,String有点特殊,因为它具有不变性,看源码是String被final声明
    
//结论:线程安全的
public static String method4(){
    StringBuilder s1 = new StringBuilder();
    s1.append("A");
    s1.append("B");
    ...
    return s1.toString();
}  

2和3发生了逃逸,作用域不止在方法内部了

1和4未发生逃逸,是安全的

我们要明确一个概念:创建对象不一定在堆空间上创建,还可以在栈上创建

总结:method1和4是安全的,2和3是不安全,所以这道题方法中定义的局部变量不一定线程安全,要看具体使用。

相关文章

  • 方法中定义的局部变量是否线程安全?

    方法中定义的局部变量是否线程安全? 了解过JVM的都知道,堆和方法区(JDK1.8后叫元空间)是线程共享的 虚拟机...

  • jsp servlet

    下面是编写线程安全的Servlet的一些建议: (1)用方法的局部变量保存请求中的专有数据。对方法中定义的局部变量...

  • Java笔记

    方法中的局部变量是肯定没有安全的问题的,除非你把局部变量用引用传值给多个子线程。 方法中的局部变量是属于每个线程栈...

  • ThreadLocal

    定义:线程的局部变量;在多个线程中,分别有独立的值(副本)。特点:线程安全、无额外开销(快速)原理:java中用h...

  • spring 单例模式

    spring-bean 默认采用单例模式,如果不在类中定义局部变量不用考虑线程不安全问题

  • 概念区分

    成员变量 & 局部变量 定义位置 成员变量:定义在类中方法外 局部变量,定义在在类的方法中,或者方法的形式参数中 ...

  • HashMap和HashTable的区别

    1、线程是否安全:HashMap是非线程安全的,HashTable是线程安全的;HashTable内部的方法基本上...

  • 多线程-了解(二)

    1.线程安全与线程共享 允许被多个线程同时执行的代码称为线程安全代码。 局部变量:存储在自己的栈中,永远不会被共享...

  • multiple-thread Note

    在多线程环境下,注意确认使用的方法是否是线程安全的方法!

  • 线程安全

    线程安全 线程安全定义:线程间共享可变资源(内存)。 实现线程安全的方法:不共享资源。使用可重入函数,不对外部资源...

网友评论

      本文标题:方法中定义的局部变量是否线程安全?

      本文链接:https://www.haomeiwen.com/subject/dayxqktx.html