在学习线程的过程中,做个了一个测试,想试试synchronized加在静态方法上时的情况。我开启了2个线程,当发现静态变量 a 为0时,则会输出,商品已经售完的语句,但是我在日志中只看到一次输出,理论上应该是2输出才对吧?测试代码见下方,求大佬帮忙看看原因!
public class T1 {
public static int a = 100;
public synchronized static void ct () {
try {
for (int i = 1; i <=100 ; i++) {
if(a==0){
System.out.println("商品已经售完!");
return;
}
a--;
System.out.println(Thread.currentThread().getName()+ ": a = " + a);
}
//Thread.sleep(10000);
} catch (Exception e) {
e.printStackTrace();
}
}
public static void main(String[] args) {
//第一个线程开始
new Thread(() ->{
T1.ct();
}).start();
//第二个线程等待第一个线程结束,理论上应该也是会输出 "商品已经售完" 才对吧?为什么日志没
//输出呢?
new Thread(() ->{
T1.ct();
}).start();
}
}
最先拿到锁的线程,执行循环,循环了100次,每次减1,最后刚好减到0,此时循环已经结束了,不会再有下一次循环,所以要到第101次循环才能打印商品售完;
而第二个拿到锁的线程,也要循环100次, 但在第一次循环时a就已经等于0了,所以打印之后就直接结束了
之前的回答有问题,所以先删了
直接在商品已售完打上线程就知道是哪个线程打印的已售完
是 Thead1 还是 Thread2 输出 '商品已经售完!' 这个不是确定的。因为你是两个单独的线程,并不是一个线程,也就是说两个线程都会竞争 T1.class 这个锁,运气好 Thread1 都抢到了,那么就是 Thread2 打印,如果是 Thread2 抢到了就是 Thread1 打印。你的商品是 100 个,你每个线程也只买 100 个,所以刚好能抢完,所以只会有一个打印。