linux 守護進程 daemon


Daemon程序簡介 
       Daemon是長時間運行的進程,通常在系統啟動后就運行,在系統關閉時才結束。一般說Daemon程序在后台運行,是因為它沒有控制終端,無法和前台的用戶交互。Daemon程序一般都作為服務程序使用,等待客戶端程序與它通信。我們也把運行的Daemon程序稱作守護進程。
      比如,我們的網絡服務程序,可以在完成創建套接口,綁定套接口,設置套接口為監聽模式后,變成守護進程進入后台執行而不占用控制終端,這是網絡服務程序的常用模式。UNIX下的網絡服務程序,如Web  Server,FTP,Telnet一般都是由守護進程(Daemon)來實現的。守護進程不占用終端,在后台運行。UNIX的守護進程一般都命名為  *d 的形式,如httpd,telnetd等等。 
      守護進程一旦脫離了終端,退出就成了問題。使用 ps axj 查出進程ID然后 kill ID 之。

Daemon程序編寫規則
   編寫Daemon程序有一些基本的規則,以避免不必要的麻煩。
   1、首先是程序運行后調用fork,並讓父進程退出。子進程獲得一個新的進程ID,但繼承了父進程的進程組ID。
   2、調用setsid創建一個新的session,使自己成為新session和新進程組的leader,並使進程沒有控制終端(tty)。
   3、改變當前工作目錄至根目錄,以免影響可加載文件系統。或者也可以改變到某些特定的目錄。
   4、設置文件創建mask為0,避免創建文件時權限的影響。
   5、關閉不需要的打開文件描述符。因為Daemon程序在后台執行,不需要於終端交互,通常就關閉STDIN、STDOUT和STDERR。其它根據實際情況處理。另一個問題是Daemon程序不能和終端交互,也就無法使用printf方法輸出信息了。

    另一個問題是Daemon程序不能和終端交互,也就無法使用printf方法輸出信息了。我們可以使用syslog機制來實現信息的輸出,方便程序的調試。

 

下面是一個daemon程序的例子:(daemontest.c) 

 

[c-sharp]  view plain  copy
 
  1. #include <unistd.h>  
  2. #include <sys/types.h>  
  3. #include <sys/stat.h>  
  4. #include <stdlib.h>  
  5. #include <stdio.h>  
  6. #include <syslog.h>  
  7. #include <signal.h>  
  8. int daemon_init(void)   
  9. {   
  10.     pid_t pid;   
  11.   if((pid = fork()) < 0)   
  12.     return(-1);   
  13.   else if(pid != 0)   
  14.     exit(0); /* parent exit */   
  15. /* child continues */   
  16.   setsid(); /* become session leader */   
  17.   chdir("/"); /* change working directory */   
  18.   umask(0); /* clear file mode creation mask */   
  19.   close(0); /* close stdin */   
  20.   close(1); /* close stdout */   
  21.   close(2); /* close stderr */   
  22.   return(0);   
  23. }  
  24.    
  25. void sig_term(int signo)   
  26. {   
  27.     if(signo == SIGTERM)   
  28. /* catched signal sent by kill(1) command */   
  29.   {   
  30.     syslog(LOG_INFO, "program terminated.");   
  31.   closelog();   
  32.   exit(0);   
  33.   }   
  34. }  
  35.    
  36. int main(void)   
  37. {   
  38.     if(daemon_init() == -1)   
  39.   {   
  40.     printf("can't fork self/n");   
  41.     exit(0);   
  42.   }   
  43.   openlog("daemontest", LOG_PID, LOG_USER);   
  44.   syslog(LOG_INFO, "program started.");   
  45.   signal(SIGTERM, sig_term); /* arrange to catch the signal */   
  46.   while(1)   
  47.   {   
  48.     sleep(1); /* put your main program here */   
  49.   }   
  50.   return(0);   
  51. }  

 
      使用如下命令編譯該程序: gcc -Wall -o daemontest daemontest.c編譯完成后生成名為daemontest的可執行程序,執行./daemontest來測試程序的運行。
      使用ps axj命令可以顯示系統中已運行的daemon程序的信息,包括進程ID、session ID、控制終端等內容。部分顯示內容:
 PPID   PID  PGID   SID TTY      TPGID STAT   UID   TIME COMMAND
    1 17765 17765 17765 ?           -1 Ss       0   0:00 ./daemontest
      從中可以看到daemontest程序運行的進程號為17765。
      我們再來看看/var/log/messages文件中的信息:     
Feb  3 17:09:30 localhost daemontest[17765]: program started.
      顯示了我們在程序中希望輸出的信息。
      使用kill 17765命令來殺死這個進程,/var/log/messages文件中就會有如下的信息:
Feb  3 17:12:26 localhost daemontest[17765]: program terminated.
      再使用ps axj命令檢查,發現系統中daemontest進程已經沒有了。

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM