首页 > 解决方案 > 为什么这段代码运行一段时间后总是什么也不打印?

问题描述

一段时间后我运行 main 方法,它总是什么也不打印。

但是如果我取消注释“System.out.println(111);” 和“System.out.println(222);”,我认为它会正常运行。

我获取了“soldWorker”线程和“refundWorker”线程的转储,他们停在了if法官面前。

public class ProducerAndConsumer2 {

    public static void main(String[] args) {
        SyncTikcet2 syncTikcet = new SyncTikcet2(1);

        Runnable syncSoldTicket = () -> syncTikcet.soldTicket();
        Runnable syncRefundTicket = () -> syncTikcet.refund();

        new Thread(syncSoldTicket, "soldWorker").start();
        new Thread(syncRefundTicket, "refundWorker").start();
    }
 }

class SyncTikcet2 {

    private Integer ticketNum;

    public SyncTikcet2(Integer ticketNum) {
        this.ticketNum = ticketNum;
    }


    public void soldTicket() {
        while (true) {
            if (ticketNum > 0) {
                ticketNum = ticketNum - 1;
                System.out.println("sold ticket,the num is" + ticketNum);
            } else {
                // System.out.println(111);
            }
        }
    }

    public void refund() {
        while (true) {
            if (ticketNum == 0) {
                ticketNum = ticketNum + 1;
                System.out.println("refund ticket,the num is" + ticketNum);
            } else  {
                // System.out.println(222);
            }
        }
    }

    public Integer getTicketNum() {
        return ticketNum;
    }

    public void setTicketNum(Integer ticketNum) {
        this.ticketNum = ticketNum;
    }
}

标签: javamultithreading

解决方案


如果您没有使该字段变为易失性,则线程没有理由重新读取它没有更改的值。

尝试使用private volatile int ticketNum

我建议使用原语 eg int,除非您必须使用 Object 引用。它不仅更快,而且还清楚地表明您不期望值是null.


推荐阅读