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,就能終止它的運行。
(本文為宇內流雲原創,經查,暫沒發現網上有類似的技術文章,歡迎轉載,但不要把作者的名字給弄丟了)