线程安全问题的举例
这里我们将举一个多个窗口卖票的例子(在这个例题中每个窗口都是一个分线程,共享资源是票)
class Window implements Runnable{
private int ticket=100;
/*
这的ticket就是共享资源,这里由于是实现了Runnable接口的方式,实现了Runable接口的方式就会形成天然的数据共享,所以这里就不需要使用static关键字修饰ticket
*/
@Override
public void run(){ //重写了Runnable接口中的run()方法
while(true){
if(ticket>0) {
try {
Thread.sleep(100); //这里我们在这里使用sleep()来增大发生错票和重票的概率,我
//sleep()方法是一个静态方法并且会抛出一个编译时异常
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName() + "正在卖票,票号为:" + ticket);
ticket--;
}else{
break;
}
}
}
}
public class WindowTest {
public static void main(String[] args) {
Window window = new Window(); //创建了Runnable接口的实现类的对象
Thread thread=new Thread(window); //我们使用创建的Runnable接口的实现类的对象作为参数来 //作为创建线程的构造方法的实参
Thread thread1=new Thread(window);
Thread thread2 =new Thread(window);
thread.setName("窗口一"); //这里我们对线程进行了改名
thread1.setName("窗口二");
thread2.setName("窗口三");
thread.start(); //这里我们开启了这三个线程
thread1.start();
thread2.start();
}
}
在这里可能会发生线程的安全问题,因为这里我们创建了三个线程共享了一份共享资源(就是这里的ticket就是共享资源)
这里的案例运行之后可能会出现卖出重票和错票的情况,这两种情况都是线程的安全问题
-
我们在这个程序中加入了一个sleep()方法,这个方法可以提高发生线程安全问题的概率,也就可以看起来更加的明显,这里并不是说我们加了sleep()方法之后就会出现错票和重票,使用sleep()方法仅仅只是提高了发生问题的概率,具体的这个程序有没有线程安全漏洞和加不加sleep()方法没有任何的关系
-
这里重票的发生是由于我们先进行了输出(也就是先进行了卖票的操作),然后才进行了票数的更改
-
这里错票的发生是由于我们最后假如当就剩一张票的时候这时候假如我们的这三个窗口都抢到了这个票的操作权,这个时候这三个窗口都会进行卖票,这时候就会出现卖的票数为0或者负数的情况