在java中如果要?jiǎng)?chuàng)建線程的話,一般有兩種方式:1)繼承Thread類;2)實(shí)現(xiàn)Runnable接口。
站在用戶的角度思考問題,與客戶深入溝通,找到巫山網(wǎng)站設(shè)計(jì)與巫山網(wǎng)站推廣的解決方案,憑借多年的經(jīng)驗(yàn),讓設(shè)計(jì)與互聯(lián)網(wǎng)技術(shù)結(jié)合,創(chuàng)造個(gè)性化、用戶體驗(yàn)好的作品,建站類型包括:成都做網(wǎng)站、網(wǎng)站設(shè)計(jì)、企業(yè)官網(wǎng)、英文網(wǎng)站、手機(jī)端網(wǎng)站、網(wǎng)站推廣、申請(qǐng)域名、網(wǎng)站空間、企業(yè)郵箱。業(yè)務(wù)覆蓋巫山地區(qū)。
1.繼承Thread類
繼承Thread類的話,必須重寫run方法,在run方法中定義需要執(zhí)行的任務(wù)。
123456789101112
class MyThread extends Thread{ private static int num = 0; public MyThread(){ num++; } @Override public void run() { System.out.println("主動(dòng)創(chuàng)建的第"+num+"個(gè)線程"); }}
創(chuàng)建好了自己的線程類之后,就可以創(chuàng)建線程對(duì)象了,然后通過(guò)start()方法去啟動(dòng)線程。注意,不是調(diào)用run()方法啟動(dòng)線程,run方法中只是定義需要執(zhí)行的任務(wù),如果調(diào)用run方法,即相當(dāng)于在主線程中執(zhí)行run方法,跟普通的方法調(diào)用沒有任何區(qū)別,此時(shí)并不會(huì)創(chuàng)建一個(gè)新的線程來(lái)執(zhí)行定義的任務(wù)。
1234567891011121314151617181920
public class Test { public static void main(String[] args) { MyThread thread = new MyThread(); thread.start(); }} class MyThread extends Thread{ private static int num = 0; public MyThread(){ num++; } @Override public void run() { System.out.println("主動(dòng)創(chuàng)建的第"+num+"個(gè)線程"); }}
在上面代碼中,通過(guò)調(diào)用start()方法,就會(huì)創(chuàng)建一個(gè)新的線程了。為了分清start()方法調(diào)用和run()方法調(diào)用的區(qū)別,請(qǐng)看下面一個(gè)例子:
1234567891011121314151617181920212223
public class Test { public static void main(String[] args) { System.out.println("主線程ID:"+Thread.currentThread().getId()); MyThread thread1 = new MyThread("thread1"); thread1.start(); MyThread thread2 = new MyThread("thread2"); thread2.run(); }} class MyThread extends Thread{ private String name; public MyThread(String name){ this.name = name; } @Override public void run() { System.out.println("name:"+name+" 子線程ID:"+Thread.currentThread().getId()); }
在java中如果要?jiǎng)?chuàng)建線程的話,一般有兩種方式:
1.繼承Thread類;
2.實(shí)現(xiàn)Runnable接口。
線程安全:
線程安全就是多線程訪問時(shí),采用了加鎖機(jī)制,當(dāng)一個(gè)線程訪問該類的某個(gè)數(shù)據(jù)時(shí),進(jìn)行保護(hù),其他線程不能進(jìn)行訪問直到該線程讀取完,其他線程才可使用。不會(huì)出現(xiàn)數(shù)據(jù)不一致或者數(shù)據(jù)污染。 線程不安全就是不提供數(shù)據(jù)訪問保護(hù),有可能出現(xiàn)多個(gè)線程先后更改數(shù)據(jù)造成所得到的數(shù)據(jù)是臟數(shù)據(jù)。
一般說(shuō)來(lái),確保線程安全的方法有這幾個(gè):競(jìng)爭(zhēng)與原子操作、同步與鎖、可重入、過(guò)度優(yōu)化。
個(gè)人認(rèn)為,保證線程安全,無(wú)外乎保持線程同步的方式,
如:@synchronized、NSLock、dispatch_semaphore、NSCondition、pthread_mutex、OSSpinLock。
然而:
OSSpinLock和dispatch_semaphore的效率遠(yuǎn)遠(yuǎn)高于其他。
@synchronized和NSConditionLock效率較差。
鑒于OSSpinLock的不安全,所以我們?cè)陂_發(fā)中如果考慮性能的話,建議使用dispatch_semaphore。
如果不考慮性能,只是圖個(gè)方便的話,那就使用@synchronized。
如何保證呢:
1.使用線程安全的類;
使用synchronized同步代碼塊,或者用Lock鎖;
由于線程安全問題,使用synchronized同步代碼塊 ? ? 原理:當(dāng)兩個(gè)并發(fā)線程訪問同一個(gè)對(duì)象object中的這個(gè)synchronized(this)同步代碼塊時(shí),一個(gè)時(shí)間內(nèi)只能有一個(gè)線程得到執(zhí)行。 ? ? 另一個(gè)線程必須等待當(dāng)前線程執(zhí)行完這個(gè)代碼塊以后才能執(zhí)行該代碼塊。
多線程并發(fā)情況下,線程共享的變量改為方法局部級(jí)變量;
作者:squirrels
鏈接:
來(lái)源:簡(jiǎn)書
著作權(quán)歸作者所有。商業(yè)轉(zhuǎn)載請(qǐng)聯(lián)系作者獲得授權(quán),非商業(yè)轉(zhuǎn)載請(qǐng)注明出處。
Java線程類也是一個(gè)object類,它的實(shí)例都繼承自java.lang.Thread或其子類。 可以用如下方式用java中創(chuàng)建一個(gè)線程:
Tread thread = new Thread();
執(zhí)行該線程可以調(diào)用該線程的start()方法:
thread.start();
在上面的例子中,我們并沒有為線程編寫運(yùn)行代碼,因此調(diào)用該方法后線程就終止了。
編寫線程運(yùn)行時(shí)執(zhí)行的代碼有兩種方式:一種是創(chuàng)建Thread子類的一個(gè)實(shí)例并重寫run方法,第二種是創(chuàng)建類的時(shí)候?qū)崿F(xiàn)Runnable接口。接下來(lái)我們會(huì)具體講解這兩種方法:
創(chuàng)建Thread的子類
創(chuàng)建Thread子類的一個(gè)實(shí)例并重寫run方法,run方法會(huì)在調(diào)用start()方法之后被執(zhí)行。例子如下:
public class MyThread extends Thread {
public void run(){
System.out.println("MyThread running");
}
}
可以用如下方式創(chuàng)建并運(yùn)行上述Thread子類
MyThread myThread = new MyThread();
myTread.start();
一旦線程啟動(dòng)后start方法就會(huì)立即返回,而不會(huì)等待到run方法執(zhí)行完畢才返回。就好像run方法是在另外一個(gè)cpu上執(zhí)行一樣。當(dāng)run方法執(zhí)行后,將會(huì)打印出字符串MyThread running。
你也可以如下創(chuàng)建一個(gè)Thread的匿名子類:
Thread thread = new Thread(){
public void run(){
System.out.println("Thread Running");
}
};
thread.start();
當(dāng)新的線程的run方法執(zhí)行以后,計(jì)算機(jī)將會(huì)打印出字符串”Thread Running”。
實(shí)現(xiàn)Runnable接口
第二種編寫線程執(zhí)行代碼的方式是新建一個(gè)實(shí)現(xiàn)了java.lang.Runnable接口的類的實(shí)例,實(shí)例中的方法可以被線程調(diào)用。下面給出例子:
public class MyRunnable implements Runnable {
public void run(){
System.out.println("MyRunnable running");
}
}
為了使線程能夠執(zhí)行run()方法,需要在Thread類的構(gòu)造函數(shù)中傳入 MyRunnable的實(shí)例對(duì)象。示例如下:
Thread thread = new Thread(new MyRunnable());
thread.start();
當(dāng)線程運(yùn)行時(shí),它將會(huì)調(diào)用實(shí)現(xiàn)了Runnable接口的run方法。上例中將會(huì)打印出”MyRunnable running”。
同樣,也可以創(chuàng)建一個(gè)實(shí)現(xiàn)了Runnable接口的匿名類,如下所示:
Runnable myRunnable = new Runnable(){
public void run(){
System.out.println("Runnable running");
}
}
Thread thread = new Thread(myRunnable);
thread.start();
創(chuàng)建子類還是實(shí)現(xiàn)Runnable接口?
對(duì)于這兩種方式哪種好并沒有一個(gè)確定的答案,它們都能滿足要求。就我個(gè)人意見,我更傾向于實(shí)現(xiàn)Runnable接口這種方法。因?yàn)榫€程池可以有效的管理實(shí)現(xiàn)了Runnable接口的線程,如果線程池滿了,新的線程就會(huì)排隊(duì)等候執(zhí)行,直到線程池空閑出來(lái)為止。而如果線程是通過(guò)實(shí)現(xiàn)Thread子類實(shí)現(xiàn)的,這將會(huì)復(fù)雜一些。
有時(shí)我們要同時(shí)融合實(shí)現(xiàn)Runnable接口和Thread子類兩種方式。例如,實(shí)現(xiàn)了Thread子類的實(shí)例可以執(zhí)行多個(gè)實(shí)現(xiàn)了Runnable接口的線程。一個(gè)典型的應(yīng)用就是線程池。
常見錯(cuò)誤:調(diào)用run()方法而非start()方法
創(chuàng)建并運(yùn)行一個(gè)線程所犯的常見錯(cuò)誤是調(diào)用線程的run()方法而非start()方法,如下所示:
Thread newThread = new Thread(MyRunnable());
newThread.run(); //should be start();
起初你并不會(huì)感覺到有什么不妥,因?yàn)閞un()方法的確如你所愿的被調(diào)用了。但是,事實(shí)上,run()方法并非是由剛創(chuàng)建的新線程所執(zhí)行的,而是被創(chuàng)建新線程的當(dāng)前線程所執(zhí)行了。也就是被執(zhí)行上面兩行代碼的線程所執(zhí)行的。想要讓創(chuàng)建的新線程執(zhí)行run()方法,必須調(diào)用新線程的start方法。
線程名
當(dāng)創(chuàng)建一個(gè)線程的時(shí)候,可以給線程起一個(gè)名字。它有助于我們區(qū)分不同的線程。例如:如果有多個(gè)線程寫入System.out,我們就能夠通過(guò)線程名容易的找出是哪個(gè)線程正在輸出。例子如下:
MyRunnable runnable = new MyRunnable();
Thread thread = new Thread(runnable, "New Thread");
thread.start();
System.out.println(thread.getName());
需要注意的是,因?yàn)镸yRunnable并非Thread的子類,所以MyRunnable類并沒有g(shù)etName()方法??梢酝ㄟ^(guò)以下方式得到當(dāng)前線程的引用:
Thread.currentThread();
因此,通過(guò)如下代碼可以得到當(dāng)前線程的名字:
String threadName = Thread.currentThread().getName();
線程代碼舉例:
這里是一個(gè)小小的例子。首先輸出執(zhí)行main()方法線程名字。這個(gè)線程JVM分配的。然后開啟10個(gè)線程,命名為1~10。每個(gè)線程輸出自己的名字后就退出。
public class ThreadExample {
public static void main(String[] args){
System.out.println(Thread.currentThread().getName());
for(int i=0; i10; i++){
new Thread("" + i){
public void run(){
System.out.println("Thread: " + getName() + "running");
}
}.start();
}
}
}
需要注意的是,盡管啟動(dòng)線程的順序是有序的,但是執(zhí)行的順序并非是有序的。也就是說(shuō),1號(hào)線程并不一定是第一個(gè)將自己名字輸出到控制臺(tái)的線程。這是因?yàn)榫€程是并行執(zhí)行而非順序的。Jvm和操作系統(tǒng)一起決定了線程的執(zhí)行順序,他和線程的啟動(dòng)順序并非一定是一致的。