對于一位合格的Java開發(fā)工程師來說孔庭,多線程和JVM是必須要熟練掌握的尺上,因此我計劃對Java的多線程部分整理出一個系列的文章,希望讀者朋友們有所收獲圆到。我們之所以需要了解多線程怎抛,是因為它在工作中確實發(fā)揮著巨大的作用。在這個系列中會更加偏向?qū)嶋H操作芽淡,能夠和我們?nèi)粘5墓ぷ鲀?nèi)容結(jié)合起來马绝。
先來說一下線程的故事,隨著多核心CPU的出現(xiàn)和普及挣菲,并發(fā)編程也就開始流行起來富稻,對于如何使用多核心CPU掷邦,多線程和多進程是比較普遍的兩種方式。首先要明確一下進程和線程的區(qū)別椭赋,進程是系統(tǒng)進行資源分配和調(diào)度的一個獨立單位抚岗,線程是CPU調(diào)度和分派的基本單位,線程的出現(xiàn)可以讓一個進程有了并發(fā)的能力哪怔。
很多性能卓越的項目是基于多線程的宣蔚,也有很多性能卓越的項目是基于多進程的。MySQL是基于多線程的认境,PostGRESQL和Oracle則是基于多進程的胚委,Nginx也是基于多進程的。很多項目在Windows下基于多線程叉信,在Linux上基于多進程亩冬,之所以會有這樣的實現(xiàn)方式,也跟操作系統(tǒng)的支持情況和發(fā)展歷程有關(guān)硼身。
對于Java項目來說硅急,很多都是基于多線程的,而且多線程也是我們本次介紹的重點鸠姨。在Java中铜秆,要實現(xiàn)多線程,有很多種方式讶迁,我們這里先介紹最常見的兩種方式连茧。
第一種是繼承Thread類,然后重寫其run()方法巍糯,然后通過調(diào)用我們的類的start()方法就可以啟動這個線程了啸驯。
第二種是實現(xiàn)Runnable接口,然后重寫其run()方法祟峦,然后通過new一個Thread類的子類罚斗,把我們編寫的類的實例當(dāng)做參數(shù)傳給Thread類,然后調(diào)用Thread類的對象的start()方法宅楞。
一般來說针姿,我們優(yōu)先選擇實現(xiàn)Runnable接口的方式,因為畢竟Java單繼承的特性厌衙,繼承Thread類可能會對我們的結(jié)構(gòu)設(shè)計造成一定的影響距淫。其實在內(nèi)部Thread類也實現(xiàn)了Runable接口,而且我們調(diào)用的start()方法在內(nèi)部也調(diào)用了run()方法婶希。
我們來看一個具體的例子吧榕暇,我們編寫代碼如下:
package com.mengzhidu.java.thread.demo;
/**
* 通過繼承自Thread類的方式來創(chuàng)建多線程
* 如果讀著朋友們多運行幾次這個例子,會發(fā)現(xiàn)顯示順序會略有不同
*/
public class Demo1 {
public static void main(String[] args) {
HiThread hi1 = new HiThread();
HiThread hi2 = new HiThread();
hi1.start();
hi2.start();
hi();
}
private static class HiThread extends Thread {
@Override
public void run() {
hi();
}
}
private static void hi() {
System.out.println(Thread.currentThread().getName() + " says: hi");
}
}
在上面的代碼中,我們的HiThread繼承自Thread類彤枢,然后重寫了其run()方法狰晚,然后我們新建了該類的兩個實例,然后分別調(diào)用其start方法缴啡,這樣我們就新建了兩個線程壁晒,再加上我們main方法所在的線程,一共有三個線程調(diào)用了hi()方法盟猖,所以會輸出三次問候讨衣。
這里的Thread.currentThread()是獲取當(dāng)前的線程,然后getName()是獲取線程的名稱式镐,如果我們沒有給線程設(shè)置名稱,系統(tǒng)就會給所在的線程設(shè)置一個默認(rèn)的名稱固蚤。
然后我們運行一下上面的例子娘汞,我們會看到如下的內(nèi)容:
我們多運行幾次,會發(fā)現(xiàn)輸出的結(jié)果可能會略有不同夕玩,比如有可能會是下面這樣:
對于我們繼承Thread類來實現(xiàn)多線程的范例你弦,我們就介紹到這里啦。
我們還可以通過實現(xiàn)Runnable接口的方式來創(chuàng)建多線程燎孟,我們這里可以把上面的例子來重新實現(xiàn)一下禽作,直接上代碼吧,如下所示:
package com.mengzhidu.java.thread.demo;
/**
* 通過實現(xiàn)Runnable方法的方式來創(chuàng)建多線程
*/
public class Demo2 {
public static void main(String[] args) {
Thread t1 = new Thread(new HiRunnable());
Thread t2 = new Thread(new HiRunnable());
t1.start();
t2.start();
hi();
}
private static class HiRunnable implements Runnable {
public void run() {
hi();
}
}
private static void hi() {
System.out.println(Thread.currentThread().getName() + " says: hi");
}
}
這里的HiRunnable是實現(xiàn)了Runnable接口的一個類揩页,但是多線程的創(chuàng)建還是依賴Thread這個類旷偿,然后我們通過調(diào)用它的start()方法來啟動這個線程來執(zhí)行。然后執(zhí)行的時候效果是類似的爆侣,這里就不再截圖展示了萍程。
對于多線程的創(chuàng)建,這里就整理到這里了兔仰,在實際工作中茫负,我們通常通過線程池的方式來創(chuàng)建多線程,這樣可以更好的實現(xiàn)線程的復(fù)用乎赴。