Java:多線程

一、實現(xiàn)多線程

1刽沾、繼承Thread類

方法名 說明
void run() 在線程開啟后本慕,此方法將被調(diào)用執(zhí)行
void start() 使此線程開始執(zhí)行,Java虛擬機(jī)會調(diào)用run方法()
  • run()方法和start()方法的區(qū)別侧漓?
    run():封裝線程執(zhí)行的代碼,直接調(diào)用监氢,相當(dāng)于普通方法的調(diào)用
    start():啟動線程布蔗;然后由JVM調(diào)用此線程的run()方法

  • 實現(xiàn)步驟

    • 定義一個類MyThread繼承Thread類
    • 在MyThread類中重寫run()方法
    • 創(chuàng)建MyThread類的對象
    • 啟動線程
//實現(xiàn)多線程
 class MyThread extends Thread {
    @Override
    public void run() {
        // 當(dāng)前線程
        String name = Thread.currentThread().getName();
        System.out.println(name);
        //代碼就是線程在開啟之后執(zhí)行的代碼
        for (int i = 0; i < 3; i++) {
            System.out.println("線程開啟了" + i);
        }
    }
}
// 使用
public class Demo {
    public static void main(String[] args) {
        // 當(dāng)前線程
        String name = Thread.currentThread().getName();
        System.out.println(name);

        //創(chuàng)建一個線程對象
        MyThread t1 = new MyThread();
      //表示的僅僅是創(chuàng)建對象,用對象去調(diào)用方法,并沒有開啟線程
        t1.run();
        //開啟一條線程
        t1.start();
    }
}

2、實現(xiàn)Runnable接口

方法名 說明
Thread(Runnable target) 分配一個新的Thread對象
Thread(Runnable target, String name) 分配一個新的Thread對象
  • 實現(xiàn)步驟

    • 定義一個類MyRunnable實現(xiàn)Runnable接口
    • 在MyRunnable類中重寫run()方法
    • 創(chuàng)建MyRunnable類的對象
    • 創(chuàng)建Thread類的對象浪腐,把MyRunnable對象作為構(gòu)造方法的參數(shù)
    • 啟動線程
// 實現(xiàn)Runnable接口
class MyRunnable implements Runnable{
    @Override
    public void run() {
        // 當(dāng)前線程
        String name = Thread.currentThread().getName();
        System.out.println(name);

        //線程啟動后執(zhí)行的代碼
        for (int i = 0; i < 3; i++) {
            System.out.println( "Runnable:" + i);
        }
    }
}
public class Demo {
    public static void main(String[] args) {
        // 當(dāng)前線程
        String name = Thread.currentThread().getName();
        System.out.println(name);

        //創(chuàng)建了一個參數(shù)的對象
        MyRunnable mr = new MyRunnable();
        //創(chuàng)建了一個線程對象,并把參數(shù)傳遞給這個線程.
        //在線程啟動之后,執(zhí)行的就是參數(shù)里面的run方法
        Thread t1 = new Thread(mr);
        //開啟線程
        t1.start();
    }
}

3.實現(xiàn)Callable接口

方法名 說明
V call() 計算結(jié)果纵揍,如果無法計算結(jié)果,則拋出一個異常
FutureTask(Callable<V> callable) 創(chuàng)建一個 FutureTask议街,一旦運(yùn)行就執(zhí)行給定的 Callable
V get() 如有必要泽谨,等待計算完成,然后獲取其結(jié)果
  • 實現(xiàn)步驟

    • 定義一個類MyCallable實現(xiàn)Callable接口
    • 在MyCallable類中重寫call()方法
    • 創(chuàng)建MyCallable類的對象
    • 創(chuàng)建Future的實現(xiàn)類FutureTask對象特漩,把MyCallable對象作為構(gòu)造方法的參數(shù)
    • 創(chuàng)建Thread類的對象吧雹,把FutureTask對象作為構(gòu)造方法的參數(shù)
    • 啟動線程
    • 再調(diào)用get方法,就可以獲取線程結(jié)束之后的結(jié)果涂身。
/ 實現(xiàn)Callable接口
class MyCallable implements Callable<String> {
    @Override
    public String call() throws Exception {
        // 當(dāng)前線程
        String name = Thread.currentThread().getName();
        System.out.println(name);

        for (int i = 0; i < 3; i++) {
            System.out.println("Callable:" + i);
        }
        //返回值就表示線程運(yùn)行完畢之后的結(jié)果
        return "答應(yīng)";
    }
}
public class Demo {
    public static void main(String[] args) throws ExecutionException, InterruptedException {
        // 當(dāng)前線程
        String name = Thread.currentThread().getName();
        System.out.println(name);

        //線程開啟之后需要執(zhí)行里面的call方法
        MyCallable mc = new MyCallable();
        //可以獲取線程執(zhí)行完畢之后的結(jié)果.也可以作為參數(shù)傳遞給Thread對象
        FutureTask<String> ft = new FutureTask<>(mc);

        //創(chuàng)建線程對象
        Thread t1 = new Thread(ft);
        //開啟線程
        t1.start();

        String s = ft.get();
        System.out.println(s);
    }
}

4.三種實現(xiàn)方式的對比

  • 實現(xiàn)Runnable雄卷、Callable接口
    • 好處: 擴(kuò)展性強(qiáng),實現(xiàn)該接口的同時還可以繼承其他的類
    • 缺點: 編程相對復(fù)雜蛤售,不能直接使用Thread類中的方法
  • 繼承Thread類
    • 好處: 編程比較簡單丁鹉,可以直接使用Thread類中的方法
    • 缺點: 可以擴(kuò)展性較差妒潭,不能再繼承其他的類

二、多線程的使用

1.設(shè)置和獲取線程名稱

方法名 說明
void setName(String name) 將此線程的名稱更改為等于參數(shù)name
String getName() 返回此線程的名稱
Thread currentThread() 返回對當(dāng)前正在執(zhí)行的線程對象的引用
static void sleep(long millis) 使當(dāng)前正在執(zhí)行的線程停留(暫停執(zhí)行)指定的毫秒數(shù)
final int getPriority() 返回此線程的優(yōu)先級
final void setPriority(int newPriority) 更改此線程的優(yōu)先級線程默認(rèn)優(yōu)先級是5揣钦;線程優(yōu)先級的范圍是:1-10
void setDaemon(boolean on) 將此線程標(biāo)記為守護(hù)線程雳灾,當(dāng)運(yùn)行的線程都是守護(hù)線程時,Java虛擬機(jī)將退出
public class MyRunnable implements Runnable {
    @Override
    public void run() {
        for (int i = 0; i < 30; i++) {
            try {
                // 不同線程睡眠時間不同冯凹,觀察控制臺打印
                String name = Thread.currentThread().getName();
                if (name.equals("1")){
                    Thread.sleep(10);
                }else{
                    Thread.sleep(20);
                }
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(Thread.currentThread().getName() + "---" + i);
        }
    }
}

public class Demo {
    public static void main(String[] args) throws InterruptedException {
        MyRunnable mr = new MyRunnable();

        //線程是有默認(rèn)名字的,格式:Thread-編號
        // 設(shè)置線程名
        Thread t1 = new Thread(mr,"1");
        Thread t2 = new Thread(mr,"2");
      
  //設(shè)置優(yōu)先級: 1 - 10 默認(rèn)值:5
        t1.setPriority(10);
        t2.setPriority(1);
        
        t1.start();
        t2.start();
    }
}

2.線程鎖:synchronized

  • 安全問題出現(xiàn)的條件
    • 是多線程環(huán)境
    • 有共享數(shù)據(jù)
    • 有多條語句操作共享數(shù)據(jù)
  • 同步代碼塊格式:
  synchronized(任意對象) { 
    多條語句操作共享數(shù)據(jù)的代碼 
  }

synchronized(任意對象):就相當(dāng)于給代碼加鎖了佑女,任意對象就可以看成是一把鎖


public class MyThread extends Thread {
    private static int ticketCount = 100;
    private static Object obj = new Object();

    @Override
    public void run() {
        while(true){
            synchronized (obj){ //就是當(dāng)前的線程對象.
                if(ticketCount <= 0){
                    //賣完了
                    break;
                }else{
                    try {
                        Thread.sleep(100);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    ticketCount--;
                    System.out.println(Thread.currentThread().getName() + "在賣票,還剩下" + ticketCount + "張票");
                }
            }
        }
    }
}
public class Demo {
    public static void main(String[] args) {
        MyThread t1 = new MyThread();
        MyThread t2 = new MyThread();

        t1.setName("窗口一");
        t2.setName("窗口二");

        t1.start();
        t2.start();
    }
}

2.線程鎖:ReentrantLock

方法名 說明
ReentrantLock() 創(chuàng)建一個ReentrantLock的實例
  • 加鎖解鎖方法

    方法名 說明
    void lock() 獲得鎖
    void unlock() 釋放鎖
public class Ticket implements Runnable {
    //票的數(shù)量
    private int ticket = 100;
    private Object obj = new Object();
    private ReentrantLock lock = new ReentrantLock();

    @Override
    public void run() {
        while (true) {
            lock.lock();
            if (ticket <= 0) {
                //賣完了
                break;
            } else {
                ticket--;
                System.out.println(Thread.currentThread().getName() + "在賣票,還剩下" + ticket + "張票");
            }
            lock.unlock();

        }
    }
}
public class Demo {
    public static void main(String[] args) {
        Ticket ticket = new Ticket();

        Thread t1 = new Thread(ticket);
        Thread t2 = new Thread(ticket);
        Thread t3 = new Thread(ticket);

        t1.setName("窗口一");
        t2.setName("窗口二");
        t3.setName("窗口三");

        t1.start();
        t2.start();
        t3.start();
    }
}

3..生產(chǎn)者消費者

  • Object類的等待和喚醒方法

    方法名 說明
    void wait() 導(dǎo)致當(dāng)前線程等待,直到另一個線程調(diào)用該對象的 notify()方法或 notifyAll()方法
    void notify() 喚醒正在等待對象監(jiān)視器的單個線程
    void notifyAll() 喚醒正在等待對象監(jiān)視器的所有線程
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末谈竿,一起剝皮案震驚了整個濱河市团驱,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌空凸,老刑警劉巖嚎花,帶你破解...
    沈念sama閱讀 206,126評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異呀洲,居然都是意外死亡紊选,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,254評論 2 382
  • 文/潘曉璐 我一進(jìn)店門道逗,熙熙樓的掌柜王于貴愁眉苦臉地迎上來兵罢,“玉大人,你說我怎么就攤上這事滓窍÷舸剩” “怎么了?”我有些...
    開封第一講書人閱讀 152,445評論 0 341
  • 文/不壞的土叔 我叫張陵吏夯,是天一觀的道長此蜈。 經(jīng)常有香客問我,道長噪生,這世上最難降的妖魔是什么裆赵? 我笑而不...
    開封第一講書人閱讀 55,185評論 1 278
  • 正文 為了忘掉前任,我火速辦了婚禮跺嗽,結(jié)果婚禮上战授,老公的妹妹穿的比我還像新娘。我一直安慰自己桨嫁,他們只是感情好植兰,可當(dāng)我...
    茶點故事閱讀 64,178評論 5 371
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著瞧甩,像睡著了一般钉跷。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上肚逸,一...
    開封第一講書人閱讀 48,970評論 1 284
  • 那天爷辙,我揣著相機(jī)與錄音彬坏,去河邊找鬼。 笑死膝晾,一個胖子當(dāng)著我的面吹牛栓始,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播血当,決...
    沈念sama閱讀 38,276評論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼幻赚,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了臊旭?” 一聲冷哼從身側(cè)響起落恼,我...
    開封第一講書人閱讀 36,927評論 0 259
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎离熏,沒想到半個月后佳谦,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,400評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡滋戳,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,883評論 2 323
  • 正文 我和宋清朗相戀三年钻蔑,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片奸鸯。...
    茶點故事閱讀 37,997評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡咪笑,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出娄涩,到底是詐尸還是另有隱情窗怒,我是刑警寧澤,帶...
    沈念sama閱讀 33,646評論 4 322
  • 正文 年R本政府宣布钝满,位于F島的核電站兜粘,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏弯蚜。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,213評論 3 307
  • 文/蒙蒙 一剃法、第九天 我趴在偏房一處隱蔽的房頂上張望碎捺。 院中可真熱鬧,春花似錦贷洲、人聲如沸收厨。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,204評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽诵叁。三九已至,卻和暖如春钦椭,著一層夾襖步出監(jiān)牢的瞬間拧额,已是汗流浹背碑诉。 一陣腳步聲響...
    開封第一講書人閱讀 31,423評論 1 260
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留侥锦,地道東北人进栽。 一個月前我還...
    沈念sama閱讀 45,423評論 2 352
  • 正文 我出身青樓,卻偏偏與公主長得像恭垦,于是被迫代替她去往敵國和親快毛。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,722評論 2 345

推薦閱讀更多精彩內(nèi)容