導航:首頁 > 操作系統 > linux守護進程編寫

linux守護進程編寫

發布時間:2022-07-17 12:47:37

1. 如何編寫linux Daemon後台程序

守護進程的編程要點 :
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下必須顯式等待子進程結束才能釋放僵屍進程。

2. 如何正確編寫linux守護進程

1、守護進程,也就是通常說的Daemon進程,是Linux中的後台服務進程。它是一個生存期較長的進程,通常獨立於控制終端並且周期性地執行某種任務或等待處理某些發生的事件。如果想讓某個進程不因為用戶或終端或其他地變化而受到影響,那麼就必須把這個進程變成一個守護進程。
2、創建守護進程步驟
1)創建子進程,父進程退出
之後的所有工作都在子進程中完成,而用戶在Shell終端里則可以執行其他命令,從而在形式上做到了與控制終端的脫離。
在Linux中父進程先於子進程退出會造成子進程成為孤兒進程,而每當系統發現一個孤兒進程時,就會自動由1號進程(init)收養它,這樣,原先的子進程就會變成init進程的子進程。
2)在子進程中創建新會話
進程組:是一個或多個進程的集合。進程組有進程組ID來唯一標識。除了進程號(PID)之外,進程組ID也是一個進程的必備屬性。每個進程組都有一個組長進程,其組長進程的進程號等於進程組ID。且該進程組ID不會因組長進程的退出而受到影響。
會話周期:會話期是一個或多個進程組的集合。通常,一個會話開始於用戶登錄,終止於用戶退出,在此期間該用戶運行的所有進程都屬於這個會話期。
(1)pid_t setsid(void);
setsid() creates a new session if the calling process is not a process group leader. The calling process will be the only process in this new process group and in this new session.
setsid函數用於創建一個新的會話,並擔任該會話組的組長。調用setsid有下面的3個作用:
① 讓進程擺脫原會話的控制
② 讓進程擺脫原進程組的控制
③ 讓進程擺脫原控制終端的控制
有以下三個結果:
(a)成為新會話的首進程
(b)成為一個新進程組的組長進程
(c)沒有控制終端。
有些人建議在此時再次調用fork,並使父進程終止。第二個子進程作為守護進程繼續運行。這樣就保證了該守護進程不是會話首進程。
setsid函數能夠使進程完全獨立出來,從而擺脫其他進程的控制。
setsid()調用成功後,進程成為新的會話組長和新的進程組長,並與原來的登錄會話和進程組脫離。由於會話過程對控制終端的獨占性,進程同時與控制終端脫離。 子進程可以自己組成一個新的進程組,即調用setpgrp()與原進程組脫離關系,產生一個新的進程組,進程組號與它的進程號相同.這樣,父進程退出運行後就不會影響子進程的當前運行.
3)改變當前目錄為根目錄
使用fork創建的子進程繼承了父進程的當前工作目錄;進程活動時,其工作目錄所在的文件系統不能卸下。通常的做法是讓"/"作為守護進程的當前工作目錄,也可以是其他目錄,如/tmp,使用chdir。
4)重設文件許可權掩碼
文件許可權掩碼是指屏蔽掉文件許可權中的對應位。比如,有個文件許可權掩碼是050,它就屏蔽了文件組擁有者的可讀與可執行許可權。mask = mask & ~050
通常,把文件許可權掩碼設置為0,umask(0)。
5)關閉文件描述符
用fork函數新建的子進程會從父進程那裡繼承已經打開了的文件描述符。這些被打開的文件可能永遠不會被守護進程讀寫,但它們一樣消耗系統資源,而且可能導致所在的文件系統無法卸下。
在上面的第二步之後,守護進程已經與所屬的控制終端失去了聯系。因此從終端輸入的字元不可能達到守護進程,守護進程中用常規方法(如printf)輸出的字元也不可能在終端上顯示出來。所以,文件描述符為0、1和2 的3個文件(常說的輸入、輸出和報錯)已經失去了存在的價值,也應被關閉。
for(i=0;i<MAXFILE;i++)
close(i);
6)守護進程退出處理
當用戶需要外部停止守護進程運行時,往往會使用 kill命令停止該守護進程。所以,守護進程中需要編碼來實現kill發出的signal信號處理,達到進程的正常退出。
signal(SIGTERM, sigterm_handler);
void sigterm_handler(int arg)
{
_running = 0;
}
7)處理SIGCHLD信號
處理SIGCHLD信號並不是必須的。但對於某些進程,特別是伺服器進程往往在請求到來時生成子進程處理請求。如果父進程不等待子進程結束,子進程將成為僵屍進程(zombie)從而佔用系統資源。如果父進程等待子進程結束,將增加父進程的負擔,影響伺服器進程的並發性能。在Linux下可以簡單地將 SIGCHLD信號的操作設為SIG_IGN。
signal(SIGCHLD,SIG_IGN);
這樣,內核在子進程結束時不會產生僵屍進程。

3. linux編寫一個程序,要求運行後成為守護進程,每隔3秒修改一個本機的IP地址,並在屏幕上顯示IP地址信息。

守護進程與終端已經脫離關系,無法把輸出列印到屏幕上,一下代碼可以幫你實現這個功能,需要在終端上使用ifconfig命令查看ip的變化
執行的時候需要使用超級用戶許可權,因為修改ip需要超級用戶許可權才能修改
需要有一個/tmp/ip_list.txt文件,每行存儲一個需要改變的ip地址,例如:

192.168.1.2
192.168.1.3
192.168.1.4
192.168.1.5
192.168.1.6
192.168.1.7
192.168.1.8
192.168.1.9
192.168.1.10

代碼如下:

#include <unistd.h>#include <stdio.h>#include <stdlib.h>#include <string.h>#include <sys/types.h>#include <sys/stat.h>#include <fcntl.h>#include <signal.h>void sigterm_handler(int arg){exit(1);}int main(int argc, char * argv[]){pid_t pid;FILE *fp;char line[128];char cmd[128];pid = fork();if (pid < 0) {perror("fork error");exit(-1);} else if (pid > 0) {/* father */printf("father die\n");exit(-1);}/* child */if (setsid() == -1)perror("setsid() error");if (chdir("/") == -1)perror("chdir error");printf("child: pid = %d\n", getpid());signal(SIGTERM, sigterm_handler);fp = fopen("/tmp/ip_list.txt", "r");if (fp == NULL) {perror("fopen");exit(-1);}umask(0);close(STDIN_FILENO);close(STDOUT_FILENO);close(STDERR_FILENO);while (1) {memset(line, 0, sizeof(line));if (NULL == fgets(line, sizeof(line) - 1, fp)) {rewind(fp);continue;}/* check ip format here if nessesary *//* ... */memset(cmd, 0, sizeof(cmd));snprintf(cmd, sizeof(cmd) - 1, "ifconfig eth0 %s", line);system(cmd);sleep(3);}fclose(fp);return 0;}

4. linux下守護進程編寫問題

因為終端都沒了所以才需要重定向

這三個描述符默認是用終端的, 終端沒了之後把它們重定向到文件這樣它們就仍然可用

所謂「依靠終端才起作用」那是默認情況

5. 如何以守護進程在linux系統下執行

編寫守護進程程序的要點:
(1)讓程序在後台執行。方法是調用fork()產生一個子進程,然後使父進程退出。
(2)調用setsid()創建一個新對話期。控制終端、登錄會話和進程組通常是從父進程繼承下來的,守護進程要擺脫它們,不受它們的影響,方法是調用setsid()使進程成為一個會話組長。setsid()調用成功後,進程成為新的會話組長和進程組長,並與原來的登錄會話、進程組和控制終端脫離。
(3)禁止進程重新打開控制終端。經過以上步驟,進程已經成為一個無終端的會話組長,但是它可以重新申請打開一個終端。為了避免這種情況發生,可以通過使進程不再是會話組長來實現。再一次通過fork()創建新的子進程,使調用fork的進程退出。
(4)關閉不再需要的文件描述符。子進程從父進程繼承打開的文件描述符。如不關閉,將會浪費系統資源,造成進程所在的文件系統無法卸下以及引起無法預料的錯誤。首先獲得最高文件描述符值,然後用一個循環程序,關閉0到最高文件描述符值的所有文件描述符。
(5)將當前目錄更改為根目錄。
(6)子進程從父進程繼承的文件創建屏蔽字可能會拒絕某些許可權。為防止這一點,使用unmask(0)將屏蔽字清零。
(7)處理SIGCHLD信號。對於伺服器進程,在請求到來時往往生成子進程處理請求。如果父進程不等待子進程結束,子進程將成為僵屍進程(zombie),從而佔用系統資源。如果父進程等待子進程結束,將增加父進程的負擔,影響伺服器進程的並發性能。在Linux下可以簡單地將SIGCHLD信號的操作設為SIG_IGN。這樣,子進程結束時不會產生僵屍進程。

6. 如何在Linux下用c語言創建守護進程並監控系統運行期間的所有進程

這跟execvp函數的實現方式有關:
int execvp(const char *file ,char * const argv []);

execvp()會從PATH 環境變數所指的目錄中查找符合參數file的文件名,找到後便執行該文件,然後將第二個參數argv傳給該欲執行的文件。如果執行成功則函數不會返回,執行失敗則直接返回-1,失敗原因存於errno中。

之所以顯示「fail to exec」,是因為在PATH環境變數所指的目錄中沒有名為「hello」的程序。建議進行如下操作:
1、運行「echo $PATH」,查看一下PATH環境變數指向那些目錄
2、編寫一個輸出「hello world」的程序,並命名為hello,即執行命令:
gcc -o hello hello.c
3、把名為」hello「的程序拷貝到PATH變數所指的其中一個目錄中

7. Linux 簡單的守護進程 程序源代碼是什麼怎麼寫

//根據麥子學院視頻改編。。。。 #include #include #include #include #include #include #include int main(int argc,char**argv) { int num=0; //sundy print log char content[128]=""; //1,創建一個子進程,退出父進程 pid_t pid=fork(); if(pid0) { exit(0); } //2,設置新的會話 pid_t ret=setsid(); if(ret0) { sprintf(content, "sundy print log = %d\n", num); write(fd,content,strlen(content)); close(fd); } num++; sleep(3); } return 0; }

閱讀全文

與linux守護進程編寫相關的資料

熱點內容
閩政通無法請求伺服器是什麼 瀏覽:48
怎麼做積木解壓神器 瀏覽:203
王者榮耀解壓玩具抽獎 瀏覽:49
12位是由啥加密的 瀏覽:868
程序員編迷你世界代碼 瀏覽:895
php取現在時間 瀏覽:246
單片機高吸收 瀏覽:427
怎麼區分五代頭是不是加密噴頭 瀏覽:244
hunt測試伺服器是什麼意思 瀏覽:510
2013程序員考試 瀏覽:641
畢業論文是pdf 瀏覽:736
伺服器跑網心雲劃算嗎 瀏覽:471
單片機定時器計數初值的計算公式 瀏覽:801
win7控制台命令 瀏覽:567
貓咪成年app怎麼升級 瀏覽:692
360有沒有加密軟體 瀏覽:315
清除cisco交換機配置命令 瀏覽:751
華為刪除交換機配置命令 瀏覽:473
shell打包命令 瀏覽:827
加密狗插上輸不了密碼 瀏覽:187