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