在很多java基础的面试题中,经常有问到类似StringBuilder和StringBuffer、Vector和ArrayList的区别这样的问题,很多答案都会提一下线程安全与不安全的区别,下面说一下我对java中线程不安全的理解。
在大多数时机的多线程应用中,多个线程需要共享对同一数据的存取,如果两个线程存取相同的对象,并且没一个线程都调用了一个修改该对象状态的方法,根据各线程访问程序的次序,可能会产生错误的对象,这样的一个情况被称为“竞争条件”。存在竞争的线程不安全,不存在竞争的线程就是安全的。
《Java并发编程实践》中,给出了线程安全性的解释:
A class is thread-safe when it continues to behave correctly when accessed from multiple threads
当一个类,不断被多个线程调用,仍能表现出正确的行为时,那它就是线程安全的。这里的关键在于对“正确的行为”的理解,什么意思呢?下面写几个例子来看一下就很明了 了:
public class Count implements Runnable{
private int num = 50;//共有50张车票
public void run() {
for (int i = 0; i < 50; i++) {
if (num > 0) {
try {
//模拟网络延迟
Thread.sleep(100);
System.out.println(Thread.currentThread().getName() + "抢到了第"
+ num-- + "号车票");
} catch (Exception e) {
e.printStackTrace();
}
}
}
}
}
public static void main(String[] args) {
//创建三个线程,模拟三个人抢票
Runnable count= new Count();
new Thread(count,"A").start();
new Thread(count,"B").start();
new Thread(count,"C").start();
}
我们可以看到运行结果:
为什么第38张票被BC两个人抢到了呢?
多个线程执行时,CPU对线程的调度是随机的,我们不知道当前程序被执行到哪步就切换到了下一个线程,这里是C从内存中读到38的之后,打印出来,还没来得及num–,这时切换到了B,B读到的依然是38,然后不管AB的num–哪个先执行,都是将37刷新到内存中,显然这是一个严重的问题。