并发编程是Java语言的重要特性之一,当然也是最难以掌握的内容。编写可靠的并发程序是一项不小的挑战。但是,作为程序员的我们,要变得更有价值,就需要啃一些硬骨头了。因此,理解并发编程的基础理论和编程实践,让自己变得更值钱吧。
问题一:什么是线程安全问题?
线程安全问题是指当多个线程同时读写一个状态变量,并且没有任何同步措施时候,导致脏数据或者其他不可预见的结果的问题。Java 中首要的同步策略是使用Synchronized 关键字,它提供了可重入的独占锁。
问题二:什么是共享变量可见性问题?
要谈可见性首先需要介绍下多线程处理共享变量时候的 Java 中内存模型。
Java 内存模型规定了所有的变量都存放在主内存中,当线程使用变量时候都是把主内存里面的变量拷贝到了自己的工作空间或者叫做工作内存。
当线程操作一个共享变量时候操作流程为:
那么假如线程 A 和 B 同时去处理一个共享变量,会出现什么情况呢?
首先他们都会去走上面的三个流程,假如线程 A 拷贝共享变量到了工作内存,并且已经对数据进行了更新但是还没有更新会主内存(结果可能目前存放在当前cpu 的寄存器或者高速缓存),这时候线程 B 拷贝共享变量到了自己的工作内存进行处理,处理后,线程 A 才把自己的处理结果更更新到主内存或者缓存,可知线程 B 处理的并不是线程 A 处理后的结果,也就是说线程 A 处理后的变量值对线程 B 不可见,这就是共享变量的不可见性问题。
构成共享变量内存不可见原因是因为三步流程不是原子性操作,下面知道使用恰当同步就可以解决这个问题。
我们知道 ArrayList 是线程不安全的,因为他的读写方法没有同步策略,会导致脏数据和不可预期的结果,下面我们就一一讲解如何解决。
这是线程不安全的
public class ArrayList<E>
{
public E get(int index) { rangeCheck(index);
return elementData(index);
}
public E set(int index, E element) { rangeCheck(index);
E oldValue = elementData(index); elementData[index] = element; return oldValue;
}
}
由于文章字数限制,这里只能展示部分文档;我这边整理了一份完整的PDF,有需要的朋友可以关注我,转发文章后;私信我“并发编程”即可免费获取到这份文档哦!
上一篇
已是最后文章