java多線程編程之使用thread類創(chuàng)建線程
在Java中創(chuàng)建線程有兩種方法:使用Thread類和使用Runnable接口。在使用Runnable接口時(shí)需要建立一個(gè)Thread實(shí)例。因此,無(wú)論是通過(guò)Thread類還是Runnable接口建立線程,都必須建立Thread類或它的子類的實(shí)例。Thread類的構(gòu)造方法被重載了八次,構(gòu)造方法如下:
public Thread( );
public Thread(Runnable target);
public Thread(String name);
public Thread(Runnable target, String name);
public Thread(ThreadGroup group, Runnable target);
public Thread(ThreadGroup group, String name);
public Thread(ThreadGroup group, Runnable target, String name);
public Thread(ThreadGroup group, Runnable target, String name, long stackSize);
Runnable target
實(shí)現(xiàn)了Runnable接口的類的實(shí)例。要注意的是Thread類也實(shí)現(xiàn)了Runnable接口,因此,從Thread類繼承的類的實(shí)例也可以作為target傳入這個(gè)構(gòu)造方法。
String name
線程的名子。這個(gè)名子可以在建立Thread實(shí)例后通過(guò)Thread類的setName方法設(shè)置。如果不設(shè)置線程的名子,線程就使用默認(rèn)的線程名:Thread-N,N是線程建立的順序,是一個(gè)不重復(fù)的正整數(shù)。
ThreadGroup group
當(dāng)前建立的線程所屬的線程組。如果不指定線程組,所有的線程都被加到一個(gè)默認(rèn)的線程組中。關(guān)于線程組的細(xì)節(jié)將在后面的章節(jié)詳細(xì)討論。
long stackSize
線程棧的大小,這個(gè)值一般是CPU頁(yè)面的整數(shù)倍。如x86的頁(yè)面大小是4KB。在x86平臺(tái)下,默認(rèn)的線程棧大小是12KB。
一個(gè)普通的Java類只要從Thread類繼承,就可以成為一個(gè)線程類。并可通過(guò)Thread類的start方法來(lái)執(zhí)行線程代碼。雖然Thread類的子類可以直接實(shí)例化,但在子類中必須要覆蓋Thread類的run方法才能真正運(yùn)行線程的代碼。下面的代碼給出了一個(gè)使用Thread類建立線程的例子:
package mythread;
public class Thread1 extends Thread
{
public void run()
{
System.out.println(this.getName());
}
public static void main(String[] args)
{
System.out.println(Thread.currentThread().getName());
Thread1 thread1 = new Thread1();
Thread1 thread2 = new Thread1 ();
thread1.start();
thread2.start();
}
}
上面的代碼建立了兩個(gè)線程:thread1和thread2。上述代碼中的005至行是Thread1類的run方法。當(dāng)在014和015行調(diào)用start方法時(shí),系統(tǒng)會(huì)自動(dòng)調(diào)用run方法。在007行使用this.getName()輸出了當(dāng)前線程的名字,由于在建立線程時(shí)并未指定線程名,因此,所輸出的線程名是系統(tǒng)的默認(rèn)值,也就是Thread-n的形式。在011行輸出了主線程的線程名。
上面代碼的運(yùn)行結(jié)果如下:
main
Thread-0
Thread-1
從上面的輸出結(jié)果可以看出,第一行輸出的main是主線程的名子。后面的Thread-1和Thread-2分別是thread1和thread2的輸出結(jié)果。
注意:任何一個(gè)Java程序都必須有一個(gè)主線程。一般這個(gè)主線程的名子為main。只有在程序中建立另外的線程,才能算是真正的多線程程序。也就是說(shuō),多線程程序必須擁有一個(gè)以上的線程。
Thread類有一個(gè)重載構(gòu)造方法可以設(shè)置線程名。除了使用構(gòu)造方法在建立線程時(shí)設(shè)置線程名,還可以使用Thread類的setName方法修改線程名。要想通過(guò)Thread類的構(gòu)造方法來(lái)設(shè)置線程名,必須在Thread的子類中使用Thread類的public Thread(String name)構(gòu)造方法,因此,必須在Thread的子類中也添加一個(gè)用于傳入線程名的構(gòu)造方法。下面的代碼給出了一個(gè)設(shè)置線程名的例子:
package mythread;
public class Thread2 extends Thread
{
private String who;
public void run()
{
System.out.println(who + ":" + this.getName());
}
public Thread2(String who)
{
super();
this.who = who;
}
public Thread2(String who, String name)
{
super(name);
this.who = who;
}
public static void main(String[] args)
{
Thread2 thread1 = new Thread2 ("thread1", "MyThread1");
Thread2 thread2 = new Thread2 ("thread2");
Thread2 thread3 = new Thread2 ("thread3");
thread2.setName("MyThread2");
thread1.start();
thread2.start();
thread3.start();
}
在類中有兩個(gè)構(gòu)造方法:
第011行:public sample2_2(String who)
這個(gè)構(gòu)造方法有一個(gè)參數(shù):who。這個(gè)參數(shù)用來(lái)標(biāo)識(shí)當(dāng)前建立的線程。在這個(gè)構(gòu)造方法中仍然調(diào)用Thread的默認(rèn)構(gòu)造方法public Thread( )。
第016行:public sample2_2(String who, String name)
這個(gè)構(gòu)造方法中的who和第一個(gè)構(gòu)造方法的who的含義一樣,而name參數(shù)就是線程的名名。在這個(gè)構(gòu)造方法中調(diào)用了Thread類的public Thread(String name)構(gòu)造方法,也就是第018行的super(name)。
在main方法中建立了三個(gè)線程:thread1、thread2和thread3。其中thread1通過(guò)構(gòu)造方法來(lái)設(shè)置線程名,thread2通過(guò)setName方法來(lái)修改線程名,thread3未設(shè)置線程名。
運(yùn)行結(jié)果如下:
thread1:MyThread1
thread2:MyThread2
thread3:Thread-1
從上面的輸出結(jié)果可以看出,thread1和thread2的線程名都已經(jīng)修改了,而thread3的線程名仍然為默認(rèn)值:Thread-1。thread3的線程名之所以不是Thread-2,而是Thread-1,這是因?yàn)樵?26行已經(jīng)指定了thread2的Name,因此,啟動(dòng)thread3時(shí)就將thread3的線程名設(shè)為Thread-1。因此就會(huì)得到上面的輸出結(jié)果。
注意:在調(diào)用start方法前后都可以使用setName設(shè)置線程名,但在調(diào)用start方法后使用setName修改線程名,會(huì)產(chǎn)生不確定性,也就是說(shuō)可能在run方法執(zhí)行完后才會(huì)執(zhí)行setName。如果在run方法中要使用線程名,就會(huì)出現(xiàn)雖然調(diào)用了setName方法,但線程名卻未修改的現(xiàn)象。
Thread類的start方法不能多次調(diào)用,如不能調(diào)用兩次thread1.start()方法。否則會(huì)拋出一個(gè)IllegalThreadStateException異常。
相關(guān)文章
在Java程序中使用數(shù)據(jù)庫(kù)的新方法
這篇文章主要介紹了在Java程序中使用數(shù)據(jù)庫(kù)的新方法,講述了Java8以來(lái)數(shù)據(jù)庫(kù)API的一些新特性,需要的朋友可以參考下2015-07-07使用Mybatis-plus策略自動(dòng)更新數(shù)據(jù)庫(kù)時(shí)間失敗問(wèn)題解決
這篇文章主要介紹了使用Mybatis-plus策略自動(dòng)更新數(shù)據(jù)庫(kù)時(shí)間失敗問(wèn)題解決,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-10-10java equals和=,==的區(qū)別詳細(xì)介紹
這篇文章主要介紹了java equals和=,==的區(qū)別,學(xué)習(xí)Java的朋友對(duì)equals 和== 這個(gè)概念開(kāi)始使用的時(shí)候會(huì)有疑問(wèn),很難辨別如何正確使用,這里幫大家詳細(xì)講解該知識(shí)點(diǎn),希望大家能掌握,有需要的小伙伴可以參考下2016-10-10Java實(shí)現(xiàn)表單提交(支持多文件同時(shí)上傳)
本文介紹了Java、Android實(shí)現(xiàn)表單提交(支持多文件同時(shí)上傳)的方法,具有一定的參考價(jià)值,下面跟著小編一起來(lái)看下吧2017-01-01java環(huán)境配好后jar文件打開(kāi)命令框閃退(無(wú)打開(kāi)方式,無(wú)反應(yīng))解決辦法
在Java開(kāi)發(fā)中我們經(jīng)常會(huì)遇到運(yùn)行Jar包時(shí)閃退的問(wèn)題,下面這篇文章主要給大家介紹了關(guān)于java環(huán)境配好后jar文件打開(kāi)命令框閃退(無(wú)打開(kāi)方式,無(wú)反應(yīng))的解決辦法,文中通過(guò)圖文介紹的非常詳細(xì),需要的朋友可以參考下2024-04-04SpringBoot整合Sa-Token實(shí)現(xiàn)登錄認(rèn)證的示例代碼
本文主要介紹了SpringBoot整合Sa-Token實(shí)現(xiàn)登錄認(rèn)證的示例代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-01-01使用 Spring Boot 內(nèi)嵌容器 Undertow創(chuàng)建服務(wù)器的方法
Undertow是一個(gè)非常輕量并高性能的web server,它來(lái)自 JBoss。支持blocking和non-blocking兩種NIO API。接下來(lái)通過(guò)本文給大家介紹使用Spring Boot 內(nèi)嵌容器 Undertow創(chuàng)建服務(wù)器的方法,感興趣的朋友一起看看吧2017-11-11