Linux守護進程(Daemon)是Linux的後台服務進程,它脫離了與控制終端的關聯,直接由Linux init進程管理其生命周期,即使你關閉了控制台,daemon也能在後台正常工作。
一句話,為Linux開發與控制台無關的,需要在後台長時間不間斷運行的“服務程序”,Daemon技術是非常重要的。
Daemon程序一般用c/c++開發。不過,我今天要講的,不是怎麼用c/c++開發daemon,而是用C#!
一,創建Daemon程序:
用VS新建一個控制台項目,假設名稱是MyDaemon,輸入下邊的代碼:
using System.Runtime.InteropServices; using System.Threading; namespace MyDaemon { class Program { static void Main(string[] args) { int pid = fork(); if (pid != 0) exit(0); //設置“會話組長”,與父進程脫離 setsid(); pid = fork(); if (pid != 0) exit(0); //已經進程“守護進程”工作狀態了! //關閉所有打開的文件描述符 int max = open("/dev/null", 0); for (var i = 0; i <= max; i++) { close(i); } //重設文件掩模 umask(0); //執行你的程序過程 DaemonMain(args); } /// <summary> /// Daemon工作狀態的主方法 /// </summary> /// <param name="aargs"></param> static void DaemonMain(string[] aargs) { //你的工作代碼... //daemon時,控制台輸入輸出流已經關閉 //請不要再用Console.Write/Read等方法
//阻止daemon進程退出 while (true) { Thread.Sleep(1000); } } [DllImport("libc", SetLastError = true)] static extern int fork(); [DllImport("libc", SetLastError = true)] static extern int setsid(); [DllImport("libc", SetLastError = true)] static extern int umask(int mask); [DllImport("libc", SetLastError = true)] static extern int open([MarshalAs(UnmanagedType.LPStr)]string pathname, int flags); [DllImport("libc", SetLastError = true)] static extern int close(int fd); [DllImport("libc", SetLastError = true)] static extern int exit(int code); } }
然後編譯為 MyDaemon.exe。
二,部署和運行:
.net 程序在linux運行,一般都會使用mono這個.net框架,不過,為了簡單方便,我這裡使用 AnyExec來運行這個程序(關於AnyExec,請參閱:不裝mono,你的.NET程序照樣可以在Linux上運行!)。
1,把 MyDeamon.exe放到anyexec的app文件夾;
2,把 "any"這個程序復制為 MyDeamon;
3,運行:見證神奇的時間到了!請你在linux控制終端上輸入: ./MyDaemon,哈哈,怎麼沒有反應? 其實,不是沒有反應,是你這個 MyDaemon程序已經在後台跑起來了!
輸入 “ps -ef”,看看!
看到那個 MyDaemon了吧!這次運行的PID是11618,父進程是的PID是1,1是誰?linux init!
4,退出daemon程序:daemon程序不會與控制台輸入輸出進行交互,所以,用Console.ReadLine之類的方法控制進程的退出是不現實的。那麼,怎麼關閉這個在後台運行的 daemon呢? 最簡辦法就是用ps -ef查出這個進程的PID號,然後用kill命令終止它。比如當前運行的這個 mydaemon的PID號是 11618,你只需要輸入 kill -9 11618,就能終止它的運行。
(本文為宇內流雲原創,經查,暫沒發現網上有類似的技術文章,歡迎轉載,但不要把作者的名字給弄丟了)