He_mengcheng
2021-03-08 11:29
采纳率: 50%
浏览 95
已采纳

使用synchronized锁定静态方法后,第一个线程调用过后第二个线程为什么没有继续调用?

 在学习线程的过程中,做个了一个测试,想试试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();
    }
}

 

  • 写回答
  • 好问题 提建议
  • 关注问题
  • 收藏
  • 邀请回答

4条回答 默认 最新

  • 农夫丶果园 2021-03-08 12:00
    已采纳

    最先拿到锁的线程,执行循环,循环了100次,每次减1,最后刚好减到0,此时循环已经结束了,不会再有下一次循环,所以要到第101次循环才能打印商品售完;

    而第二个拿到锁的线程,也要循环100次, 但在第一次循环时a就已经等于0了,所以打印之后就直接结束了

    已采纳该答案
    评论
    解决 1 无用
    打赏 举报
  • Forrest Gump plus 2021-03-08 11:43

    之前的回答有问题,所以先删了

    评论
    解决 无用
    打赏 举报
  • 被风吹过的回忆 2021-03-08 13:13

    直接在商品已售完打上线程就知道是哪个线程打印的已售完

    评论
    解决 无用
    打赏 举报
  • 三岁丫 2021-03-08 14:03

    是 Thead1 还是 Thread2 输出 '商品已经售完!'  这个不是确定的。因为你是两个单独的线程,并不是一个线程,也就是说两个线程都会竞争 T1.class 这个锁,运气好 Thread1 都抢到了,那么就是 Thread2 打印,如果是 Thread2 抢到了就是 Thread1 打印。你的商品是 100 个,你每个线程也只买 100 个,所以刚好能抢完,所以只会有一个打印。

    评论
    解决 无用
    打赏 举报

相关推荐 更多相似问题