谈谈线程安全

谈谈线程安全

线程安全问题是由于多个线程同时执行写操作引起的,例如多个线程同时对一个变量进行+1操作,伪代码如下。

int count  = 1;
Function A(){
    count = count + 1;
}
//thread1开始执行A()
//thread2开始执行A()
//thread3开始执行A()
......
//threadn开始执行A()

我们再来看看count=count+1的汇编代码,如下:

mov eax,[count的内存地址]  //将count值放入寄存器eax中
add eax,1                //对eax加1
mov [count的内存地址],eax  //将eax寄存器中的值放入count的地址中
  • 从上述代码中可以看出,我们对count 执行加一的操作需要三步,①取出count值。②将count加一。③将count写回内存。当有多个线程同时对count进行操作时,可能会发生什么情况?
  • 有可能,线程1刚把count取出来,恰巧线程2也把count取出来了,两个线程都执行加一操作,两个线程的count值就都变成了2,最后,线程1和线程2写回的count值都是2,但是我们想要的结果是3。这就造成了线程不安全的问题。

如何解决线程不安全的问题呢?

有两种解决方案:

  • 第一种,在线程1读取count值、count+1、写回count值的过程中,线程2必须等待,直到线程1结束。
  • 第二种,线程1、2可以同时读取count值,同时给count+1,但是写回的时候,线程1、2要竞争系统锁,如果线程1竞争到了,线程1要判断现在内存中的count值(1)是否与之前读到的count值(1)相同,如果一致则写回,然后释放锁;线程2发现内存中的count值变成了2,与之前读到的1不一致,需要重复加载count,计算、加锁比较,直到成功。

本文中的例子,参考《深入理解Java高并发编程》P59

posted @ 2022-06-27 17:09  classic123  阅读(58)  评论(0编辑  收藏  举报