首先看下如果没加同步代码块会造成什么样的线程安全问题:
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-- + "号票");
}
}
}
}