CPU的一次操作必须是原子性的,想象一下多个窗口卖同一时段同一个电影的电影票的例子(共享数据源),当一个线程再买票,计算剩余票数和显示剩余票数的时候,其他窗口不能进行这些操作,如果多个窗口同时操作就会出现问题,例如窗口2卖完票剩下100张,窗口2先卖票,然后计算计算剩余票数为100张,在显示剩余票数前,窗口1也卖出去了一张票,此时实际剩下99张,窗口2显示的时候还是刚才计算的100张。如何解决线程安全问题呢?
首先,判断线程安全问题出现的条件: A:是否是多线程环境
B:是否有共享数据
C:是否有多条语句操作共享数据
解决方案:只需要限制可能会出现问题的语句在一个线程执行时,其他线程不能够执行就不会出现问题,例如当窗口2在完成买票,计算剩余票数和显示剩余票数这一系列动作前,其他窗口等待,这样就不会出现问题了,java有相应的同步机制 同步代码块:
synchronized(对象){
需要同步的代码;
} 注意:
同步可以解决安全问题的根本原因就在那个对象上。该对象如同锁的功能。
多个线程必须是同一把锁。
public class SellTicket implements Runnable {
// 定义101张票
private int tickets = 101;
//创建锁对象
private Object obj = new Object();
@Override
public void run() {
while (true) {
synchronized (obj) {
if (tickets > 0) {
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()
+ "正在出售第" + (tickets--) + "张票");
}
}
}
}
}
//测试类
public class SellTicketDemo {
public static void main(String[] args) {
// 创建资源对象
SellTicket st = new SellTicket();//只能创建一个对象去启动线程,这样才可以共享数据
// 创建三个线程对象
Thread t1 = new Thread(st, "窗口1");
Thread t2 = new Thread(st, "窗口2");
Thread t3 = new Thread(st, "窗口3");
// 启动线程
t1.start();
t2.start();
t3.start();
}
}