1. 程式人生 > >linux系統程式設計之程序(八):守護程序詳解及建立,daemon()使用

linux系統程式設計之程序(八):守護程序詳解及建立,daemon()使用

一,守護程序概述

Linux Daemon(守護程序)是執行在後臺的一種特殊程序。它獨立於控制終端並且週期性地執行某種任務或等待處理某些發生的事件。它不需要使用者輸入就能執行而且提供某種服務,不是對整個系統就是對某個使用者程式提供服務。Linux系統的大多數伺服器就是通過守護程序實現的。常見的守護程序包括系統日誌程序syslogd、 web伺服器httpd、郵件伺服器sendmail和資料庫伺服器mysqld等。

守護程序一般在系統啟動時開始執行,除非強行終止,否則直到系統關機都保持執行。守護程序經常以超級使用者(root)許可權執行,因為它們要使用特殊的埠(1-1024)或訪問某些特殊的資源。

一個守護程序的父程序是init程序,因為它真正的父程序在fork出子程序後就先於子程序exit退出了,所以它是一個由init繼承的孤兒程序。守護程序是非互動式程式,沒有控制終端,所以任何輸出,無論是向標準輸出裝置stdout還是標準出錯裝置stderr的輸出都需要特殊處理。

守護程序的名稱通常以d結尾,比如sshd、xinetd、crond等

二,建立守護程序步驟

首先我們要了解一些基本概念:

程序組 :

  • 每個程序也屬於一個程序組
  • 每個程序主都有一個程序組號,該號等於該程序組組長的PID號 .
  • 一個程序只能為它自己或子程序設定程序組ID號

會話期:

會話期(session)是一個或多個程序組的集合。

setsid()函式可以建立一個對話期:

 如果,呼叫setsid的程序不是一個程序組的組長,此函式建立一個新的會話期

(1)此程序變成該對話期的首程序

(2)此程序變成一個新程序組的組長程序。

(3)此程序沒有控制終端,如果在呼叫setsid前,該程序有控制終端,那麼與該終端的聯絡被解除。 如果該程序是一個程序組的組長,此函式返回錯誤。

(4)為了保證這一點,我們先呼叫fork()然後exit(),此時只有子程序在執行

現在我們來給出建立守護程序所需步驟:

編寫守護程序的一般步驟步驟:

(1)在父程序中執行fork並exit推出;

(2)在子程序中呼叫setsid函式建立新的會話;

(3)在子程序中呼叫chdir函式,讓根目錄 ”/” 成為子程序的工作目錄;

(4)在子程序中呼叫umask函式,設定程序的umask為0;

(5)在子程序中關閉任何不需要的檔案描述符

說明:

1. 在後臺執行。
為避免掛起控制終端將Daemon放入後臺執行。方法是在程序中呼叫fork使父程序終止,讓Daemon在子程序中後臺執行。
if(pid=fork())
exit(0);//是父程序,結束父程序,子程序繼續
2. 脫離控制終端,登入會話和程序組
有必要先介紹一下Linux中的程序與控制終端,登入會話和程序組之間的關係:程序屬於一個程序組,程序組號(GID)就是程序組長的程序號(PID)。登入會話可以包含多個程序組。這些程序組共享一個控制終端。這個控制終端通常是建立程序的登入終端。
控制終端,登入會話和程序組通常是從父程序繼承下來的。我們的目的就是要擺脫它們,使之不受它們的影響。方法是在第1點的基礎上,呼叫setsid()使程序成為會話組長:
setsid();
說明:當程序是會話組長時setsid()呼叫失敗。但第一點已經保證程序不是會話組長。setsid()呼叫成功後,程序成為新的會話組長和新的程序組長,並與原來的登入會話和程序組脫離。由於會話過程對控制終端的獨佔性,程序同時與控制終端脫離。
3. 禁止程序重新開啟控制終端
現在,程序已經成為無終端的會話組長。但它可以重新申請開啟一個控制終端。可以通過使程序不再成為會話組長來禁止程序重新開啟控制終端:
if(pid=fork())
exit(0);//結束第一子程序,第二子程序繼續(第二子程序不再是會話組長)
4. 關閉開啟的檔案描述符
程序從建立它的父程序那裡繼承了開啟的檔案描述符。如不關閉,將會浪費系統資源,造成程序所在的檔案系統無法卸下以及引起無法預料的錯誤。按如下方法關閉它們:
for(i=0;i 關閉開啟的檔案描述符close(i);>
5. 改變當前工作目錄
程序活動時,其工作目錄所在的檔案系統不能卸下。一般需要將工作目錄改變到根目錄。對於需要轉儲核心,寫執行日誌的程序將工作目錄改變到特定目錄如/tmpchdir("/")
6. 重設檔案建立掩模
程序從建立它的父程序那裡繼承了檔案建立掩模。它可能修改守護程序所建立的檔案的存取位。為防止這一點,將檔案建立掩模清除:umask(0);
7. 處理SIGCHLD訊號
處理SIGCHLD訊號並不是必須的。但對於某些程序,特別是伺服器程序往往在請求到來時生成子程序處理請求。如果父程序不等待子程序結束,子程序將成為殭屍程序(zombie)從而佔用系統資源。如果父程序等待子程序結束,將增加父程序的負擔,影響伺服器程序的併發效能。在Linux下可以簡單地將SIGCHLD訊號的操作設為SIG_IGN。
signal(SIGCHLD,SIG_IGN);
這樣,核心在子程序結束時不會產生殭屍程序。這一點與BSD4不同,BSD4下必須顯式等待子程序結束才能釋放殭屍程序。

三,建立守護程序

在建立之前我們先了解setsid()使用:

  #include <unistd.h>

       pid_t setsid(void);

DESCRIPTION
       setsid()  creates a new session if the calling process is not a process
       group leader
The calling process is the leader of  the  new  session,
       the  process group leader of the new process group, and has no control-
       ling tty
The process group ID and session ID of the  calling  process
       are set to the PID of the calling process
The calling process will be
       the only process in this new process group and in this new session
.

//呼叫程序必須是非當前程序組組長,呼叫後,產生一個新的會話期,且該會話期中只有一個程序組,且該程序組組長為呼叫程序,沒有控制終端,新產生的group ID 和 session ID 被設定成呼叫程序的PID

RETURN VALUE
       On success, the (new) session ID of the calling  process  is  returned.
       On  error,  (pid_t) -1  is  returned,  and errno is set to indicate the
       error.

現在根據上述步驟建立一個守護程序:

以下程式是建立一個守護程序,然後利用這個守護程序每個一分鐘向daemon.log檔案中寫入當前時間

#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <time.h>
#include <fcntl.h>
#include <string.h>
#include <sys/stat.h>

#define ERR_EXIT(m) \
do\
{\
    perror(m);\
    exit(EXIT_FAILURE);\
}\
while (0);\

void creat_daemon(void);
int main(void)
{
    time_t t;
    int fd;
    creat_daemon();
    while(1){
        fd = open("daemon.log",O_WRONLY|O_CREAT|O_APPEND,0644);
        if(fd == -1)
            ERR_EXIT("open error");
        t = time(0);
        char *buf = asctime(localtime(&t));
        write(fd,buf,strlen(buf));
        close(fd);
        sleep(60);
            
    }
    return 0;
}
void creat_daemon(void)
{
    pid_t pid;
    pid = fork();
    if( pid == -1)
        ERR_EXIT("fork error");
    if(pid > 0 )
        exit(EXIT_SUCCESS);
    if(setsid() == -1)
        ERR_EXIT("SETSID ERROR");
    chdir("/");
    int i;
    for( i = 0; i < 3; ++i)
    {
        close(i);
        open("/dev/null", O_RDWR);
        dup(0);
        dup(0);
    }
    umask(0);
    return;
}

結果:

QQ截圖20130713184143

結果顯示:當我一普通使用者執行a.out時,程序表中並沒有出現新建立的守護程序,但當我以root使用者執行時,成功了,並在/目錄下建立了daemon.log檔案,cat檢視後確實每個一分鐘寫入一次。為什麼只能root執行,那是因為當我們建立守護程序時,已經將當前目錄切換我/目錄,所以當我之後建立daemon.log檔案是其實是在/目錄下,那肯定不行,因為普通使用者沒有許可權,或許你會問那為啥沒報錯呢?其實是有出錯,只不過我們在建立守護程序時已經將標準輸入關閉並重定向到/dev/null,所以看不到錯誤資訊。

四,利用庫函式daemon()建立守護程序

其實我們完全可以利用daemon()函式建立守護程序,其函式原型:

#include <unistd.h>

int daemon(int nochdir, int noclose);


DESCRIPTION
       The daemon() function is for programs wishing to detach themselves from
       the controlling terminal and run in the background as system daemons.

       If nochdir is zero, daemon()  changes  the  process’s  current  working
       directory to the root directory ("/"); otherwise,

       If  noclose is zero, daemon() redirects standard input, standard output
       and standard error to /dev/null; otherwise,  no  changes  are  made  to
       these file descriptors.

功能:建立一個守護程序

引數:

nochdir:=0將當前目錄更改至“/”

noclose:=0將標準輸入、標準輸出、標準錯誤重定向至“/dev/null”

返回值:

成功:0

失敗:-1

現在我們利用daemon()改寫剛才那個程式:

#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <time.h>
#include <fcntl.h>
#include <string.h>
#include <sys/stat.h>

#define ERR_EXIT(m) \
do\
{\
    perror(m);\
    exit(EXIT_FAILURE);\
}\
while (0);\

void creat_daemon(void);
int main(void)
{
    time_t t;
    int fd;
    if(daemon(0,0) == -1)
        ERR_EXIT("daemon error");
    while(1){
        fd = open("daemon.log",O_WRONLY|O_CREAT|O_APPEND,0644);
        if(fd == -1)
            ERR_EXIT("open error");
        t = time(0);
        char *buf = asctime(localtime(&t));
        write(fd,buf,strlen(buf));
        close(fd);
        sleep(60);
            
    }
    return 0;
}

當daemon(0,0)時:

QQ截圖20130713190523

結果同剛才一樣,也是隻有root才能成功,普通使用者執行時看不到錯誤資訊

現在讓daemon(0,1),就是不關閉標準輸入輸出結果:

QQ截圖20130713190932

可以看到錯誤資訊

現在讓daemon(1,0),就是不重定向,結果如下:

QQ截圖20130713191221

這次普通使用者執行成功了,以為沒有切換到/目錄下,有許可權

其實我們可以利用我們剛才寫的建立守護程序程式預設daemon()實現:

程式碼如下:

#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <time.h>
#include <fcntl.h>
#include <string.h>
#include <sys/stat.h>

#define ERR_EXIT(m) \
do\
{\
    perror(m);\
    exit(EXIT_FAILURE);\
}\
while (0);\

void creat_daemon(int nochdir, int noclose);
int main(void)
{
    time_t t;
    int fd;
    creat_daemon(0,0);
    while(1){
        fd = open("daemon.log",O_WRONLY|O_CREAT|O_APPEND,0644);
        if(fd == -1)
            ERR_EXIT("open error");
        t = time(0);
        char *buf = asctime(localtime(&t));
        write(fd,buf,strlen(buf));
        close(fd);
        sleep(60);
            
    }
    return 0;
}
void creat_daemon(int nochdir, int noclose)
{
    pid_t pid;
    pid = fork();
    if( pid == -1)
        ERR_EXIT("fork error");
    if(pid > 0 )
        exit(EXIT_SUCCESS);
    if(setsid() == -1)
        ERR_EXIT("SETSID ERROR");
    if(nochdir == 0)
        chdir("/");
    if(noclose == 0){
            int i;
    for( i = 0; i < 3; ++i)
    {
        close(i);
        open("/dev/null", O_RDWR);
        dup(0);
        dup(0);
    }

    umask(0);
    return;
}

相關推薦

linux系統程式設計程序守護程序建立daemon()使用

一,守護程序概述 Linux Daemon(守護程序)是執行在後臺的一種特殊程序。它獨立於控制終端並且週期性地執行某種任務或等待處理某些發生的事件。它不需要使用者輸入就能執行而且提供某種服務,不是對整個系統就是對某個使用者程式提供服務。Linux系統的大多數伺服器就是通過守護程序實現的。常見的守護程序包括系

linux系統程式設計訊號訊號的阻塞與未決

/*************************************************************************     > File Name: process_.c     > Author: Simba     > Mail: [email 

linux守護程序建立daemon()使用

Linux Daemon(守護程序)是執行在後臺的一種特殊程序。它獨立於控制終端並且週期性地執行某種任務或等待處理某些發生的事件。它不需要使用者輸入就能執行而且提供某種服務,不是對整個系統就是對某個使用者程式提供服務。Linux系統的大多數伺服器就是通過守護程序實現的。常見的守護程序包括系統日誌程序sysl

Linux系統程式設計學習筆記1-檔案的I/O操作

概述 在Linux系統下,通常以一個非負整數來代指一個開啟的檔案,這些檔案可以包括終端,socket,裝置,普通檔案等等。規定的三個標準的檔案描述符為0,1,2下面分別介紹(在互動式shell中,這些檔案描述符通常指向shell執行所在的終端): 檔案描述

ASP.NET MVC 排球計分程序 排球計分程序的演示

png 運行 logs net 插入 bsp img 多條 mage 運行主頁 插入一些球員信息 單擊提交進入統計界面 統計界面可以在中途查詢信息,也可以實時的為場上的比賽計分 添加第一條信息 單擊提交後文本框會清空 插入到數據庫裏的信息會在下方顯示

胡八一Java多執行緒

多執行緒的優勢:多程序執行需要獨立的記憶體空間,而多執行緒可以共享記憶體,從而提高了執行緒的執行效率。 建立執行緒一般使用兩種方式: 1、繼承Thread類: import java.io.IOException; public class Test extends

Linux小小白入門教程閱讀文字命令

以下操作在Linux終端進行。Linux因為許可權非常嚴格,所以暫時所有的命令操作全部是在/home資料夾下的/yangjw資料夾下進行。/yangjw資料夾就是登入使用者名稱所在的資料夾,出了此資料

Linux網路程式設計學習筆記7---5種I/O模型select輪詢

本文主要介紹5種I/O模型,select函式以及利用select實現C/S模型。 1、5種I/O模型 (1)阻塞I/O: 一直等到資料到來,才會將資料從核心中拷貝到使用者空間中。 (2)非阻塞I/O: 每過一段時

Linux進階部署

經過長期的開發工作,在專案中經常會需要將打包好的專案部署到Linux伺服器上,不過,在此之前需要掌握一些Linux常用命令比如ls、ll、ps -ef|grep java 、cp、tail 等等,詳細的可以百度查一下,還是很多的。接下來,就說一下關於專案的部署。 首先需要一個遠端訪問Linu

Selenium2+Python3.6實戰定位下拉菜單出錯如何解決?用select或xpath定位。

排查 會有 有時 ide 導入 python3 很好 沒有 元素 在登錄界面,有時候會有幾種不同的角色,針對不同角色定位到的信息是不一樣的。查詢資料知道定位下拉框的元素有兩種方式:Xpath和select。 但是使用xpath定位時,user定位到了,登錄的時候卻是調用的a

(轉)javaSpringIOC註解裝配Bean

pos work 多個 public pre tor not 註解裝配 creat 在這裏我們要詳細說明一下利用Annotation-註解來裝配Bean。 因為如果你學會了註解,你就再也不願意去手動配置xml文件了,下面就看看Annotation的魅力所在吧。 先

系統吞吐量、TPSQPS、使用者併發

軟體效能測試的基本概念和計算公式 一、軟體效能的關注點 對一個軟體做效能測試時需要關注那些效能呢? 我們想想在軟體設計、部署、使用、維護中一共有哪些角色的參與,然後再考慮這些角色各自關注的效能點是什麼,作為一個軟體效能測試工程師,我們又該關注什麼? 首先,開發軟體的目的是為了讓使用者

作業系統程序作業排程常見演算法

分享一下我老師大神的人工智慧教程!零基礎,通俗易懂!http://blog.csdn.net/jiangjunshow 也歡迎大家轉載本篇文章。分享知識,造福人民,實現我們中華民族偉大復興!        

linux】Valgrind工具集命令列

一、使用方法 usage: valgrind [options] prog-and-args 使用方法:valgrind [引數選項] 程式和引數 二、選擇工具 tool-selection option, with default in [ ]: 工具選擇選項,預設值在[]

Ansibleplaybook劇本介紹和配置

前言 在上篇博文中講解了ansible的多個常用的模組,這些模組讓ansible具有了管理,部署後端主機的能力,但是一個一個命令的執行明顯很浪費時間,那麼能不能有一個檔案類似於shell指令碼那樣可以把複雜的、重複的命令,簡單化、程式流程化起來呢?答案是肯定的,playbook劇本就

Go遊戲伺服器開發的一些思考Docker橋接網路固定IP

為什麼需要Docker的橋接網路 有時我們需要把Docker容器暴露到某個網段,這樣就可以把一個Docker容器看成一臺物理機。這時就需要用到Docker的橋接網路。 比如 在做Redis叢集時,通常現有的Redis管理軟體會通過ssh來管理一組Redis。

windows程式設計14滑鼠訊息

關於滑鼠的一些細節知識: 通常,我們發訊息時,都是對一個特定的視窗,但是對於滑鼠訊息卻不然:只要滑鼠跨越視窗或者在某視窗中按下滑鼠按鍵,那麼視窗訊息處理程式就會收到滑鼠訊息,而不管該視窗是否活動或者是否擁有輸入焦點。滑鼠訊息一個有21種:10個顯示區域訊息,11個非顯示區域

java程式設計師的大資料5HDFS壓縮與壓縮

背景 好久沒有更新了,原因是公司專案上線,差點被祭天。在這種驚心動魄的時候還是要抽時間做一點自己喜歡做的事情的,然而進度比預期慢了許多。 正式開始 接下來就開始記錄最近的學習成果啦! 在Hadoop叢集中,網路資源是非常珍貴的。因此對檔案進行壓縮是非

activiti自定義流程整合整合自定義表單建立模型

本來在建立了表單之後應該是表單列表和預覽功能,但是我看了看整合的程式碼,和之前沒有用angularjs的基本沒有什麼變化,一些極小的變動也只是基於angularjs的語法,因此完全可以參考之前說些的表單列表展示相關的內容,這裡也就直接進入到下一個步驟,建立流程模型了。

linux文字處理三劍客awk命令

簡介 awk是一個強大的文字分析工具,相對於grep的查詢,sed的編輯,awk在其對資料分析並生成報告時,顯得尤為強大。簡單來說awk就是把檔案逐行的讀入,以空格為預設分隔符將每行切片,切開的部分再進行各種分析處理。 awk有3個不同版本: awk、nawk和gaw