java多线程安全问题,4个窗口卖100张票

首先看下如果没加同步代码块会造成什么样的线程安全问题:

public class Demo014 {
	
	public static void main(String[] args) {
		new Ticket().start();
		new Ticket().start();
		new Ticket().start();
		new Ticket().start();
	}

}

class Ticket extends Thread {
	private static int ticket = 100;
	
	public void run() {
		while(true) {
			
				if(ticket <= 0) {
					break;
				}
				
				//Thread.sleep(10),模拟有多段代码执行
				try {
					Thread.sleep(10);	//可能造成:线程1睡,线程2睡,线程3睡,线程4睡
				} catch (InterruptedException e) {
					
					e.printStackTrace();
				}
				System.out.println(getName() + "...这是第" + ticket-- + "号票");
			
		}
	}
}

结果如下:

分析:线程Thread-3进入到while循环的时候,ticket为1,然后遇到Thread.sleep(10),此刻还未执行到ticket--,ticket还是为1。与此同时线程Thread-2和Thread-1进入,因为ticket为1所以未执行break。

           随后Thread-3执行ticket--,ticket变为1;然后Thread-2和Thread-1同时执行ticket--,所以出现两个ticket为-1的情况。

解决:加同步代码块

public class Demo014 {
	
	public static void main(String[] args) {
		new Ticket().start();
		new Ticket().start();
		new Ticket().start();
		new Ticket().start();
	}

}

class Ticket extends Thread {
	private static int ticket = 100;
	//private static Object obj = new Object();		//如果用引用数据类型成员变量当作锁对象,必须是静态的
	public void run() {
		while(true) {
			synchronized(Ticket.class) {
				if(ticket <= 0) {
					break;
				}
				
				//Thread.sleep(10),模拟有多段代码执行
				try {
					Thread.sleep(10);	//可能造成:线程1睡,线程2睡,线程3睡,线程4睡
				} catch (InterruptedException e) {
					
					e.printStackTrace();
				}
				System.out.println(getName() + "...这是第" + ticket-- + "号票");
			}
		}
	}
}

        

猜你喜欢

转载自blog.csdn.net/xldmx/article/details/88734800
今日推荐