高并发下可能回出现的超卖问题
运行一下代码,模拟超卖问题,运行后会发现会出现重复编号的票
public class Test {
public static void main(String[] args) {
/**
* 模拟出售电影票
*/
PiaoThread piaoThread = new PiaoThread();
new Thread(piaoThread,"1号窗口").start();
new Thread(piaoThread,"2号窗口").start();
new Thread(piaoThread,"3号窗口").start();
new Thread(piaoThread,"4号窗口").start();
}
}
class PiaoThread implements Runnable {
//多个窗口使用的是同一个对象,所以成员变量是共享的
private int piao = 100;
@Override
public void run() {
while (true){
if (piao < 1){ break;}
System.out.println(Thread.currentThread().getName()+"卖出了第"+piao+"编号的票");
piao--;
}
}
}
同步代码块可以解决超卖问题
public class Test {
public static void main(String[] args) {
/**
* 模拟出售电影票
*/
PiaoThread piaoThread = new PiaoThread();
new Thread(piaoThread,"1号窗口").start();
new Thread(piaoThread,"2号窗口").start();
new Thread(piaoThread,"3号窗口").start();
new Thread(piaoThread,"4号窗口").start();
}
}
class PiaoThread implements Runnable {
//多个窗口使用的是同一个对象,所以成员变量是共享的
private int piao = 100;
@Override
public void run() {
while (true){
//悲观锁
// synchronized (锁对象),
// 需要是同一个对象,这里都是使用的一个对象,所以可以使用this
synchronized (this){
if (piao < 1){ break;}
System.out.println(Thread.currentThread().getName()+"卖出了第"+piao+"编号的票");
piao--;
}
}
}
}
同步方法,我们把需要同步的代码块抽取出来,idea抽取方法快捷键ctrl+alt+m
同步方法的锁对象:
如果是非静态同步方法,锁对象是this
如果是静态方法那么锁对象是该方法所在类的字节码对象(类名.class)
场景:a线程使用了同步代码块,b线程使用的同步方法,这时候ab线程需要同步那么就需要知道同步方法的锁对象是谁。
public class Test {
public static void main(String[] args) {