JAVA多线程中访问变量问题

时间:2021-10-13 18:06:37
  1. 类变量(类里面static修饰的变量)保存在“方法区”
  2. 实例变量(类里面的普通变量)保存在“堆”
  3. 局部变量(方法里声明的变量)“虚拟机栈”


“方法区”和“堆”都属于线程共享数据区,“虚拟机栈”属于线程私有数据区。

因此,局部变量是不能多个线程共享的,而类变量和实例变量是可以多个线程共享的。事实上,在java中,多线程间进行通信的唯一途径就是通过类变量和实例变量。也就是说,如果一段多线程程序中如果没有类变量和实例变量,那么这段多线程程序就一定是线程安全的。

以Web开发的Servlet为例,一般我们开发的时候,自己的类继承HttpServlet之后,重写doPost()、doGet()处理请求,不管我们在这两个方法里写什么代码,只要没有操作类变量或实例变量,最后写出来的代码就是线程安全的。如果在Servlet类里面加了实例变量,就很可能出现线程安全性问题,解决方法就是把实例变量改为ThreadLocal变量,而ThreadLocal实现的含义就是让实例变量变成了“线程私有”的,即给每一个线程分配一个自己的值。


如何安全的共享变量

现在唯一的问题就是要让多个线程安全的共享变量(下文中的变量一般特指类变量和实例变量),上文提到了一种ThreadLocal的方式,其实这种方式并不是真正的共享,而是为每个线程分配一个自己的值。

比如现在有一个特别简单的需求,有一个类变量a=0,现在启动5个线程,每个线程执行a++;如果用ThreadLocal的方式,最后的结果就是5个线程都拥有一份自己的a值,最终结果都是1,这显然不符合我们的预期。

那么如果不使用ThreadLocal呢?直接声明一个类变量a=0,然后让5个线程分别去执行a++;这样结果依旧不对,而且结果是不确定的,可能是1,2,3,4,5中的任一个。这种情况叫做竞态条件(Race Condition),要理解竞态条件先要理解Java内存模型:

要理解java的内存模型,可以类比计算机硬件访问内存的模型。由于计算机的cpu运算速度和内存io速度有几个数量级的差距,因此现代计算机都不得不加入一层尽可能接近处理器运算速度的高速缓存来做缓冲:将内存中运算需要使用的数据先复制到缓存中,当运算结束后再同步回内存。

每个java线程都有一份自己的工作内存,线程访问变量的时候,不能直接访问主内存中的变量,而是先把主内存的变量复制到自己的工作内存,然后操作自己工作内存里的变量,最后再同步给主内存。

现在就可以解释为什么5个线程执行a++最后结果不一定是5了,因为a++可以分解为3步操作:

  1. 把主内存里的a复制到线程的工作内存
  2. 线程对工作内存里的a执行a=a+1
  3. 把线程工作内存里的a同步回主内存
package test;

public class Test16 extends Thread{
static int i=0;
public void run(){//线程并发访问,出现这种结果线程能同时访问一个方法吗?
i++;
System.out.println(i);
}


public static void main(String[] args) {
new Test16().start();
new Test16().start();
new Test16().start();
new Test16().start();
new Test16().start();

}

}
运行结果如下:
JAVA多线程中访问变量问题

 

这里我理解为两种,一种就是根据上面的描述,第三个输出5的情况恰巧另外三个线程把主内存里的a复制到线程的工作内存,线程对工作内存里的a执行a=a+1,最后同时把线程工作内存里的a同时同步回主内存导致输出5,另外一种是没有上面这种机制是因为,i++和print中间可能出现其他线程的插入导致print没有输出,i++执行了五次后才开始打印i.