Java多線程的線程守護(hù)兩個(gè)例子發(fā)布者:本站 時(shí)間:2020-05-06 14:05:54
核心說(shuō)明:理解這句話下面可以不看了。
守護(hù)與其他線程同時(shí)執(zhí)行,當(dāng)正在運(yùn)行的線程都是守護(hù)線程時(shí),Java 虛擬機(jī)退出。
守護(hù)線程與普通線程寫(xiě)法上基本么啥區(qū)別,調(diào)用線程對(duì)象的方法setDaemon(true),則可以將其設(shè)置為守護(hù)線程。
守護(hù)線程使用的情況較少,但并非無(wú)用,舉例來(lái)說(shuō),JVM的垃圾回收、內(nèi)存管理等線程都是守護(hù)線程。還有就是在做數(shù)據(jù)庫(kù)應(yīng)用時(shí)候,使用的數(shù)據(jù)庫(kù)連接池,連接池本身也包含著很多后臺(tái)線程,監(jiān)控連接個(gè)數(shù)、超時(shí)時(shí)間、狀態(tài)等等。
setDaemon方法的詳細(xì)說(shuō)明:
public final void setDaemon(boolean on)將該線程標(biāo)記為守護(hù)線程或用戶線程。當(dāng)正在運(yùn)行的線程都是守護(hù)線程時(shí),Java 虛擬機(jī)退出。
該方法必須在啟動(dòng)線程前調(diào)用。
該方法首先調(diào)用該線程的 checkAccess 方法,且不帶任何參數(shù)。這可能拋出 SecurityException(在當(dāng)前線程中)。
參數(shù):
on - 如果為 true,則將該線程標(biāo)記為守護(hù)線程。
拋出:
IllegalThreadStateException - 如果該線程處于活動(dòng)狀態(tài)。
SecurityException - 如果當(dāng)前線程無(wú)法修改該線程。
另請(qǐng)參見(jiàn):
代碼如下 | |
isDaemon(), checkAccess() /** * Java線程:線程的調(diào)度-守護(hù)線程 * * @author leizhimin 2009-11-4 9:02:40 */ public class Test { public static void main(String[] args) { Thread t1 = new MyCommon(); Thread t2 = new Thread(new MyDaemon()); t2.setDaemon(true); //設(shè)置為守護(hù)線程 t2.start(); t1.start(); } } class MyCommon extends Thread { public void run() { for (int i = 0; i < 5; i++) { System.out.println("線程1第" + i + "次執(zhí)行!"); try { Thread.sleep(7); } catch (InterruptedException e) { e.printStackTrace(); } } } } class MyDaemon implements Runnable { public void run() { for (long i = 0; i < 9999999L; i++) { System.out.println("后臺(tái)線程第" + i + "次執(zhí)行!"); try { Thread.sleep(7); } catch (InterruptedException e) { e.printStackTrace(); } } } } |
后臺(tái)線程第0次執(zhí)行!
線程1第0次執(zhí)行!
線程1第1次執(zhí)行!
后臺(tái)線程第1次執(zhí)行!
后臺(tái)線程第2次執(zhí)行!
線程1第2次執(zhí)行!
線程1第3次執(zhí)行!
后臺(tái)線程第3次執(zhí)行!
線程1第4次執(zhí)行!
后臺(tái)線程第4次執(zhí)行!
后臺(tái)線程第5次執(zhí)行!
后臺(tái)線程第6次執(zhí)行!
后臺(tái)線程第7次執(zhí)行!
Process finished with exit code 0
從上面的執(zhí)行結(jié)果可以看出:
前臺(tái)線程是保證執(zhí)行完畢的,后臺(tái)線程還沒(méi)有執(zhí)行完畢就退出了。
實(shí)際上:JRE判斷程序是否執(zhí)行結(jié)束的標(biāo)準(zhǔn)是所有的前臺(tái)執(zhí)線程行完畢了,而不管后臺(tái)線程的狀態(tài),因此,在使用后臺(tái)縣城時(shí)候一定要注意這個(gè)問(wèn)題。
后臺(tái)線程(守護(hù)線程)
所謂的后臺(tái)線程,是指在程序運(yùn)行的時(shí)候在后臺(tái)提供一種通用服務(wù)的線程,并且這種線程并不屬于程序中不可或缺的部分。因此當(dāng)所有的非后臺(tái)線程結(jié)束時(shí),程序也就終止了,同時(shí)會(huì)殺死所有后臺(tái)線程。反過(guò)來(lái)說(shuō),只要有任何非后臺(tái)線程(用戶線程)還在運(yùn)行,程序就不會(huì)終止。后臺(tái)線程在不執(zhí)行finally子句的情況下就會(huì)終止其run方法。后臺(tái)線程創(chuàng)建的子線程也是后臺(tái)線程。
下面是一個(gè)后臺(tái)線程的示例:
代碼如下 | |
<span style="font-size:16px;">package demo.thread; import java.util.concurrent.TimeUnit; public class DaemonDemo implements Runnable { @Override public void run() { try { while (true) { Thread.sleep(1000); System.out.println("#" + Thread.currentThread().getName()); } } catch (InterruptedException e) { e.printStackTrace(); } finally {// 后臺(tái)線程不執(zhí)行finally子句 System.out.println("finally "); } } public static void main(String[] args) { for (int i = 0; i < 10; i++) { Thread daemon = new Thread(new DaemonDemo()); // 必須在start之前設(shè)置為后臺(tái)線程 daemon.setDaemon(true); daemon.start(); } System.out.println("All daemons started"); try { TimeUnit.MILLISECONDS.sleep(1000); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } } } </span> |
運(yùn)行結(jié)果:
All daemons started
#Thread-2
#Thread-3
#Thread-1
#Thread-0
#Thread-9
#Thread-6
#Thread-8
#Thread-5
#Thread-7
#Thread-4
分析:從結(jié)果可以看出,十個(gè)子線程并沒(méi)有無(wú)線循環(huán)的打印,而是在主線程(main())退出后,JVM強(qiáng)制關(guān)閉所有后臺(tái)線程。而不會(huì)有任何希望出現(xiàn)的確認(rèn)形式,如finally子句不執(zhí)行。
選擇我們,優(yōu)質(zhì)服務(wù),不容錯(cuò)過(guò)
1. 優(yōu)秀的網(wǎng)絡(luò)資源,強(qiáng)大的網(wǎng)站優(yōu)化技術(shù),穩(wěn)定的網(wǎng)站和速度保證
2. 15年上海網(wǎng)站建設(shè)經(jīng)驗(yàn),優(yōu)秀的技術(shù)和設(shè)計(jì)水平,更放心
3. 全程省心服務(wù),不必?fù)?dān)心自己不懂網(wǎng)絡(luò),更省心。
------------------------------------------------------------
24小時(shí)聯(lián)系電話:021-58370032