鍍金池/ 問(wèn)答/Java/ 創(chuàng)建的線程執(zhí)行多次Start方法輸出結(jié)果錯(cuò)誤

創(chuàng)建的線程執(zhí)行多次Start方法輸出結(jié)果錯(cuò)誤

class MyThread implements Runnable{
    private int num = 5;

    public void run(){
        while(num>0) {
            System.out.println(Thread.currentThread().getName()+"--->"+this.num);
             num--;
        }
    }
}
public class test{
    public static void main(String[] args) {
        MyThread mt1 = new MyThread();
        new Thread(mt1,"Number 1").start();
        new Thread(mt1,"Number 2").start();
    }
}

首先這代碼理解很簡(jiǎn)單,但我在編譯運(yùn)行時(shí),對(duì)結(jié)果很好奇,如圖。但為什么會(huì)輸出兩次5?。這和我在書(shū)上看到的答案有出入。

圖片描述

我又新起了第三個(gè)線程new Thread(mt1,"Number 3").start();,結(jié)果又有不同,如圖。
圖片描述

剛剛學(xué)習(xí)java,對(duì)此抱有疑問(wèn)。對(duì)于這幾個(gè)線程來(lái)說(shuō),num不是共享的全局變量嗎?其次,這些線程間不應(yīng)該提供鎖機(jī)制嗎?(我從1234的輸出推斷出的)那怎么會(huì)同時(shí)幾個(gè)線程會(huì)訪問(wèn)同一個(gè)num呢?

希望各位大佬能幫助我這個(gè)菜雞,謝謝。

回答
編輯回答
赱丅呿

1.線程之間并沒(méi)有自動(dòng)提供鎖這么一說(shuō),需要使用java中提供的關(guān)鍵字Sychronized、ReentrantLock類等來(lái)實(shí)現(xiàn)線程同步。
2.調(diào)用start方法只會(huì)讓線程進(jìn)入可執(zhí)行狀態(tài),然后CPU會(huì)隨機(jī)調(diào)度可執(zhí)行狀態(tài)下的線程進(jìn)行執(zhí)行。
3.因?yàn)槭嵌嗑€程,可能出現(xiàn)線程1先打印完結(jié)果,在線程1做減法之前,這時(shí)num還沒(méi)有變化,同時(shí)運(yùn)行的線程2打印了相同的結(jié)果。

2018年5月27日 21:18
編輯回答
心夠野

1、num不是共享的全局變量,你大概理解成單例模式了,但這里要引入緩存的概念,你就直接當(dāng)做每個(gè)線程第一次讀取了num后直接緩存為5,然后進(jìn)行計(jì)算,系統(tǒng)隨機(jī)調(diào)起線程。另外自增和自減不具有原子性。
2、鎖機(jī)制要自己設(shè)置啊。。。上synchronized 上volatile
3、解決方法

class MyThread implements Runnable{
    private int num = 5;
     // private volatile int num = 5;
    synchronized public void run(){
        while(num>0) {
            System.out.println(Thread.currentThread().getName()+"--->"+this.num);
            num--;
        }
        num=7;//這里是為了示例線程鎖
    }
}

4、相關(guān)建議:

A、裝個(gè)IDEA或者Eclipse 
B、看一下《Java多線程編程核心技術(shù)》有條件的話最好支持正版
2017年10月10日 09:31