1.如何理解线程是否安全
多个线程并发执行时,仍旧能够保证数据的正确性,这种现象称之为线程安全。
多个线程并发执行时,不能能够保证数据的正确性,这种现象称之为线程不安全。
譬如模拟多线程售票时多个线程操作成员变量ticket
- 导致线程不安全的因素有哪些?
1). 多个线程并发执行。
2). 多个线程并发执行时存在共享数据集(临界资源)。
3). 多个线程在共享数据集上的操作不是原子操作(不可拆分的一个操作)
我们在这里设计一个程序安全的计数器:
/**
* 测试多线程技术 看是否安全
*/
public class TestThread03 {
static class Counter implements Runnable{
private volatile int count; //volatile 禁止指令重排序 不保证原子性
public synchronized void doCount(){
count++;//此操作 非原子操作 JVM在执行过程中 可能会对指令重排序 导致数据不安全
// synchronized 保证操作原子性
}
@Override
public void run() {
for (int i=0;i<10;i++) {
doCount();
}
}
public int getCount() {
return count;
}
}
public static void main(String[] args) throws InterruptedException {
//1.构建计数器对象
Counter counter=new Counter();
//2.构建多个线程
List<Thread> list=new ArrayList<>();
for(int i=1;i<=100;i++){
list.add(new Thread(counter));
}
//3.启动线程
for (Thread t:list){
t.start();
}
//4.所有线程结束 输出count 的值
while(Thread.activeCount()>2){//如果存活的线程>1
Thread.yield();//则主线程让出CPU
}
//Thread.sleep(1000);
System.out.println(1);
System.out.println(counter.getCount());
}
}
结果
1
1000
while(Thread.activeCount()>2) 这里需要注意一个死循环的坑
ideaIntelliJ Idea执行用户的代码使用的是反射的方式,与此同时会创建一个Monitor Ctrl-Break的线程用于监控的目的。所以主线程main+监控线层,最后会剩余两个线程。
相同的代码,用java命令执行就没有问题。所以将Thread.activeCount的判断值改为Thread.activeCount > 2即可。