守护进程和守护线程
對(duì)于JAVA而言,一般一個(gè)應(yīng)用程序只有一個(gè)進(jìn)程——JVM。除非在代碼里面另外派生或者開啟了新進(jìn)程。
而線程,當(dāng)然是由進(jìn)程開啟的。當(dāng)開啟該線程的進(jìn)程離開時(shí),線程也就不復(fù)存在了。
所以,對(duì)于JAVA而言,線程是完全可以由自己的API自由產(chǎn)生。例如new Thread().但是進(jìn)程就不同,必須通過調(diào)用OS的API,如Runtime.getRuntime.exec()。 所以說,進(jìn)程是OS級(jí)別的概念。
守護(hù)線程和用戶線程的區(qū)別:
二者其實(shí)基本上是一樣的。唯一的區(qū)別在于JVM何時(shí)離開。
用戶線程:當(dāng)存在任何一個(gè)用戶線程未離開,JVM是不會(huì)離開的。
守護(hù)線程:如果只剩下守護(hù)線程未離開,JVM是可以離開的。
在Java中,制作守護(hù)線程非常簡(jiǎn)單,直接利用.setDaemon(true)
在linux或者unix操作系統(tǒng)中在系統(tǒng)的引導(dǎo)的時(shí)候會(huì)開啟很多服務(wù),這些服務(wù)就叫做守護(hù)進(jìn)程。為了增加靈活性,root可以選擇系統(tǒng)開啟的模式,這些模式叫做運(yùn)行級(jí)別,每一種運(yùn)行級(jí)別以一定的方式配置系統(tǒng)。 守護(hù)進(jìn)程是脫離于終端并且在后臺(tái)運(yùn)行的進(jìn)程。守護(hù)進(jìn)程脫離于終端是為了避免進(jìn)程在執(zhí)行過程中的信息在任何終端上顯示并且進(jìn)程也不會(huì)被任何終端所產(chǎn)生的終端信息所打斷。
守護(hù)進(jìn)程簡(jiǎn)介 守護(hù)進(jìn)程,也就是通常說的Daemon進(jìn)程,是Linux中的后臺(tái)服 務(wù)進(jìn)程。它是一個(gè)生存期較長的進(jìn)程,通常獨(dú)立于控制終端并且周期性地執(zhí)行某種任務(wù)或等待處理某些發(fā)生的事件。守護(hù)進(jìn)程常常在系統(tǒng)引導(dǎo)裝入時(shí)啟動(dòng),在系統(tǒng)關(guān) 閉時(shí)終止。Linux系統(tǒng)有很多守護(hù)進(jìn)程,大多數(shù)服務(wù)都是通過守護(hù)進(jìn)程實(shí)現(xiàn)的,同時(shí),守護(hù)進(jìn)程還能完成許多系統(tǒng)任務(wù),例如,作業(yè)規(guī)劃進(jìn)程crond、打印 進(jìn)程lqd等(這里的結(jié)尾字母d就是Daemon的意思)。
由于在Linux中,每一個(gè)系統(tǒng)與用戶進(jìn)行交流的界面稱為終端,每一個(gè)從此終端開始運(yùn)行的進(jìn)程都會(huì)依附于這個(gè)終端,這個(gè)終端就稱為這些進(jìn)程的控 制終端,當(dāng)控制終端被關(guān)閉時(shí),相應(yīng)的進(jìn)程都會(huì)自動(dòng)關(guān)閉。但是守護(hù)進(jìn)程卻能夠突破這種限制,它從被執(zhí)行開始運(yùn)轉(zhuǎn),直到整個(gè)系統(tǒng)關(guān)閉時(shí)才退出。如果想讓某個(gè)進(jìn) 程不因?yàn)橛脩艋蚪K端或其他地變化而受到影響,那么就必須把這個(gè)進(jìn)程變成一個(gè)守護(hù)進(jìn)程。
創(chuàng)建守護(hù)進(jìn)程
?創(chuàng)建子進(jìn)程,父進(jìn)程退出
這是編寫守護(hù)進(jìn)程的第一步。由于守護(hù)進(jìn)程是脫離控制終端的,因此,完成第一步后就會(huì)在Shell終端里造成一程序已經(jīng)運(yùn)行完畢的假象。之后的所有工作都在子進(jìn)程中完成,而用戶在Shell終端里則可以執(zhí)行其他命令,從而在形式上做到了與控制終端的脫離。
在Linux中父進(jìn)程先于子進(jìn)程退出會(huì)造成子進(jìn)程成為孤兒進(jìn)程,而每當(dāng)系統(tǒng)發(fā)現(xiàn)一個(gè)孤兒進(jìn)程時(shí),就會(huì)自動(dòng)由1號(hào)進(jìn)程(init)收養(yǎng)它,這樣,原先的子進(jìn)程就會(huì)變成init進(jìn)程的子進(jìn)程。
在子進(jìn)程中創(chuàng)建新會(huì)話
這個(gè)步驟是創(chuàng)建守護(hù)進(jìn)程中最重要的一步,雖然它的實(shí)現(xiàn)非常簡(jiǎn)單,但它的意義卻非常重大。在這里使用的是系統(tǒng)函數(shù)setsid,在具體介紹setsid之前,首先要了解兩個(gè)概念:進(jìn)程組和會(huì)話期
進(jìn)程組:是一個(gè)或多個(gè)進(jìn)程的集合。進(jìn)程組有進(jìn)程組ID來唯一標(biāo)識(shí)。除了進(jìn)程號(hào)(PID)之外,進(jìn)程組ID也是一個(gè)進(jìn)程的必備屬性。每個(gè)進(jìn)程組都有一個(gè)組長進(jìn)程,其組長進(jìn)程的進(jìn)程號(hào)等于進(jìn)程組ID。且該進(jìn)程組ID不會(huì)因組長進(jìn)程的退出而受到影響。
會(huì)話周期:會(huì)話期是一個(gè)或多個(gè)進(jìn)程組的集合。通常,一個(gè)會(huì)話開始于用戶登錄,終止于用戶退出,在此期間該用戶運(yùn)行的所有進(jìn)程都屬于這個(gè)會(huì)話期。
接下來就可以具體介紹setsid的相關(guān)內(nèi)容:
(1)setsid函數(shù)作用:
setsid函數(shù)用于創(chuàng)建一個(gè)新的會(huì)話,并擔(dān)任該會(huì)話組的組長。調(diào)用setsid有下面的3個(gè)作用:
讓進(jìn)程擺脫原會(huì)話的控制
讓進(jìn)程擺脫原進(jìn)程組的控制
讓進(jìn)程擺脫原控制終端的控制
那么,在創(chuàng)建守護(hù)進(jìn)程時(shí)為什么要調(diào)用setsid函數(shù)呢?由于創(chuàng)建守護(hù)進(jìn)程的第一步調(diào)用了fork函數(shù)來創(chuàng)建子進(jìn)程,再將父進(jìn)程退出。由于在調(diào)用了fork函數(shù)時(shí),子進(jìn)程全盤拷貝了父進(jìn)程的會(huì)話期、進(jìn)程組、控制終端等,雖然父進(jìn)程退出了,但會(huì)話期、進(jìn)程組、控制終端等并沒有改變,因此,這還不是真正意義上的獨(dú)立開來,而setsid函數(shù)能夠使進(jìn)程完全獨(dú)立出來,從而擺脫其他進(jìn)程的控制。
改變當(dāng)前目錄為根目錄
這一步也是必要的步驟。使用fork創(chuàng)建的子進(jìn)程繼承了父進(jìn)程的當(dāng)前工作目錄。由于在進(jìn)程運(yùn)行中,當(dāng)前目錄所在的文件系統(tǒng)(如“/mnt/usb”)是不能卸載的, 這對(duì)以后的使用會(huì)造成諸多的麻煩(比如系統(tǒng)由于某種原因要進(jìn)入單用戶模式)。因此,通常的做法是讓"/"作為守護(hù)進(jìn)程的當(dāng)前工作目錄,這樣就可以避免上述 的問題,當(dāng)然,如有特殊需要,也可以把當(dāng)前工作目錄換成其他的路徑,如/tmp。改變工作目錄的常見函數(shù)式chdir。
重設(shè)文件權(quán)限掩碼
文件權(quán)限掩碼是指屏蔽掉文件權(quán)限中的對(duì)應(yīng)位。比如,有個(gè)文件權(quán)限掩碼是050,它就屏蔽了文件組擁有者的可讀與可執(zhí)行權(quán)限。由于使用fork函數(shù)新建的子進(jìn)程繼承了父進(jìn)程的文件權(quán)限掩碼,這就給該子進(jìn)程使用文件帶來了諸多的麻煩。因此,把文件權(quán)限掩碼設(shè)置為0,可以大大增強(qiáng)該守護(hù)進(jìn)程的靈活性。設(shè)置文件權(quán)限掩碼的函數(shù)是umask。在這里,通常的使用方法為umask(0)。
關(guān)閉文件描述符
同文件權(quán)限碼一樣,用fork函數(shù)新建的子進(jìn)程會(huì)從父進(jìn)程那里繼承一些已經(jīng)打開了的文件。這些被打開的文件可能永遠(yuǎn)不會(huì)被守護(hù)進(jìn)程讀寫,但它們一樣消耗系統(tǒng)資源,而且可能導(dǎo)致所在的文件系統(tǒng)無法卸下。
在上面的第二步之后,守護(hù)進(jìn)程已經(jīng)與所屬的控制終端失去了聯(lián)系。因此從終端輸入的字符不可能達(dá)到守護(hù)進(jìn)程,守護(hù)進(jìn)程中用常規(guī)方法(如printf)輸出的字符也不可能在終端上顯示出來。所以,文件描述符為0、1和2 的3個(gè)文件(常說的輸入、輸出和報(bào)錯(cuò))已經(jīng)失去了存在的價(jià)值,也應(yīng)被關(guān)閉。通常按如下方式關(guān)閉文件描述符:
===============================
for(i=0;i<MAXFILE;i++)
close(i);
===============================
守護(hù)進(jìn)程退出處理
當(dāng)用戶需要外部停止守護(hù)進(jìn)程運(yùn)行時(shí),往往會(huì)使用 kill命令停止該守護(hù)進(jìn)程。所以,守護(hù)進(jìn)程中需要
編碼來實(shí)現(xiàn)kill發(fā)出的signal信號(hào)處理,達(dá)到進(jìn)程的正常退出。
===============================
signal(SIGTERM, sigterm_handler);
void sigterm_handler(int arg)
{
_running = 0;
}
===============================
這樣,一個(gè)簡(jiǎn)單的守護(hù)進(jìn)程就建立起來了。
實(shí)現(xiàn)守護(hù)進(jìn)程的完整實(shí)例(每隔10s在/tmp/dameon.log中寫入一句話):
#include<stdio.h>#include<stdlib.h>#include<string.h>#include<fcntl.h>#include<sys/types.h>#include<unistd.h>#include<sys/wait.h>#include <signal.h>#define MAXFILE 65535volatile sig_atomic_t _running = 1;int main(){pid_t pc;int i,fd,len;char *buf="this is a Dameon\n";len = strlen(buf);pc = fork(); //第一步if(pc<0){printf("error fork\n");exit(1);}else if(PC>0)exit(0);setsid(); //第二步chdir("/"); //第三步umask(0); //第四步for(i=0;i<MAXFILE;i++) //第五步close(i);signal(SIGTERM, sigterm_handler);while( _running ){if((fd=open("/tmp/dameon.log",O_CREAT|O_WRONLY|O_APPEND,0600))<0){perror("open");exit(1);}write(fd,buf,len);close(fd);usleep(10*1000); //10毫秒}}void sigterm_handler(int arg){_running = 0;}linux守護(hù)進(jìn)程列表
amd:自動(dòng)安裝NFS(網(wǎng)絡(luò)文件系統(tǒng))守侯進(jìn)程
apmd:高級(jí)電源治理
Arpwatch:記錄日志并構(gòu)建一個(gè)在LAN接口上看到的以太網(wǎng)地址和ip地址對(duì)數(shù)據(jù)庫
Autofs:自動(dòng)安裝治理進(jìn)程automount,與NFS相關(guān),依靠于NIS
Bootparamd:引導(dǎo)參數(shù)服務(wù)器,為LAN上的無盤工作站提供引導(dǎo)所需的相關(guān)信息
crond:linux下的計(jì)劃任務(wù)
Dhcpd:啟動(dòng)一個(gè)DHCP(動(dòng)態(tài)IP地址分配)服務(wù)器
Gated:網(wǎng)關(guān)路由守候進(jìn)程,使用動(dòng)態(tài)的OSPF路由選擇協(xié)議
Httpd:WEB服務(wù)器
Inetd:支持多種網(wǎng)絡(luò)服務(wù)的核心守候程序
Innd:Usenet新聞服務(wù)器
Linuxconf:答應(yīng)使用本地WEB服務(wù)器作為用戶接口來配置機(jī)器
Lpd:打印服務(wù)器
Mars-nwe:mars-nwe文件和用于Novell的打印服務(wù)器
Mcserv:Midnight命令文件服務(wù)器
named:DNS服務(wù)器
netfs:安裝NFS、Samba和NetWare網(wǎng)絡(luò)文件系統(tǒng)
network:激活已配置網(wǎng)絡(luò)接口的腳本程序
nfs:打開NFS服務(wù)
nscd:nscd(Name Switch Cache daemon)服務(wù)器,用于NIS的一個(gè)支持服務(wù),它高速緩存用戶口令和組成成員關(guān)系
portmap:RPC portmap治理器,與inetd類似,它治理基于RPC服務(wù)的連接
postgresql:一種SQL數(shù)據(jù)庫服務(wù)器
routed:路由守候進(jìn)程,使用動(dòng)態(tài)RIP路由選擇協(xié)議
rstatd:一個(gè)為LAN上的其它機(jī)器收集和提供系統(tǒng)信息的守候程序
ruserd:遠(yuǎn)程用戶定位服務(wù),這是一個(gè)基于RPC的服務(wù),它提供關(guān)于當(dāng)前記錄到LAN上一個(gè)機(jī)器日志中的用戶信息
rwalld:激活rpc.rwall服務(wù)進(jìn)程,這是一項(xiàng)基于RPC的服務(wù),答應(yīng)用戶給每個(gè)注冊(cè)到LAN機(jī)器上的其他終端寫消息
rwhod:激活rwhod服務(wù)進(jìn)程,它支持LAN的rwho和ruptime服務(wù)
sendmail:郵件服務(wù)器sendmail
smb:Samba文件共享/打印服務(wù)
snmpd:本地簡(jiǎn)單網(wǎng)絡(luò)治理候進(jìn)程
squid:激活代理服務(wù)器squid
syslog:一個(gè)讓系統(tǒng)引導(dǎo)時(shí)起動(dòng)syslog和klogd系統(tǒng)日志守候進(jìn)程的腳本
xfs:X Window字型服務(wù)器,為本地和遠(yuǎn)程X服務(wù)器提供字型集
xntpd:網(wǎng)絡(luò)時(shí)間服務(wù)器
ypbind:為NIS(網(wǎng)絡(luò)信息系統(tǒng))客戶機(jī)激活ypbind服務(wù)進(jìn)程
yppasswdd:NIS口令服務(wù)器
ypserv:NIS主服務(wù)器
gpm:管鼠標(biāo)的
identd:AUTH服務(wù),在提供用戶信息方面與finger類似
總結(jié)
- 上一篇: 号称“飞行硫酸” 专家提醒:隐翅虫飞到身
- 下一篇: 靓汤飞抵韩国