在linux中开发守护程序

linux下的守护程序(daemon)对应于windows下的服务程序。长期运行于后台。通常不提供UI支持,随系统启动而启动。

守护程序的启动和停止指令通常为(以apache2为例):service apache2 start/stop

在前面的博客中有一篇《基于linux TCP的select服务器》,本文以该服务器的代码为基础,将其改造为一个daemon程序,实现用service *** start / stop来控制其启动和停止。

并进而实现开机启动。

在《unix环境高级编程》中,守护程序的创建异常复杂。主要包含以下步骤:

(1)创建子进程,父进程退出。

(2)在子进程中中创建新会话

(3)设置当前目录为根目录

(4)重设文件权限掩码

(5)关闭文件描述符

因为《unix环境高级编程》和《Linux/UNIX系统编程手册》中均提供了例子。故本文不对此过程作演示。

既然创建daemon程序的流程已固定,是否我们每次都要造轮子呢?

好在linux中提供了一个系统调用可以直接使进程变为守护程序:int daemon(int nochdir, int noclose);

下面来看守护程序的代码:

#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <errno.h>
#include <sys/select.h>
#include <unistd.h>
#include <stdlib.h>
#include <unordered_map>
#include <string>
#include <iostream>
using namespace std;

//int->fd
//string->ip&port
unordered_map<int, string> clients_addr;

int main(int argc, char const *argv[])
{
    daemon(0,0);
    int listenfd = socket(AF_INET, SOCK_STREAM | SOCK_NONBLOCK | SOCK_CLOEXEC, IPPROTO_TCP );
    struct sockaddr_in serv;
    serv.sin_family = AF_INET;
    serv.sin_port = htons(5000);
    serv.sin_addr.s_addr = inet_addr("127.0.0.1");

    int on = 1;
    int connfd;
    if (setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on)) < 0)
    {
        perror("setsockopt");
        exit(EXIT_FAILURE);
    }

    if (bind(listenfd, (struct sockaddr *)&serv, sizeof(serv)) < 0)
    {
        perror("bind");
        exit(EXIT_FAILURE);
    }

    if (listen(listenfd, 3) < 0)
    {
        perror("listen");
        exit(EXIT_FAILURE);
    }

    sockaddr_in peer_addr;
    socklen_t peer_addr_len = sizeof(peer_addr);
    int maxfd = listenfd;

    fd_set fdall, fdread;
    FD_ZERO(&fdall);
    FD_SET(listenfd, &fdall);
    // printf("listen fd = %d\n",listenfd);
    while (1)
    {
        fdread = fdall;
        int ret = select(maxfd + 1, &fdread, NULL, NULL, NULL);
        if (ret == -1)
        {
            perror("select");
            exit(EXIT_FAILURE);
        }

        if (ret == 0)
        {
            continue;
        }

        for (int i = 0; i <= maxfd; i++)
        {
            if (FD_ISSET(i, &fdread))
            {
                if (i == listenfd)
                {
                    connfd = accept4(listenfd, (struct sockaddr *)&peer_addr, &peer_addr_len, SOCK_CLOEXEC | SOCK_NONBLOCK);
                    char ip_str_buf[1024];
                    sprintf(ip_str_buf, "%s:%d", inet_ntoa(peer_addr.sin_addr), ntohs(peer_addr.sin_port));

                    clients_addr[connfd] = ip_str_buf;
                    cout << clients_addr[connfd] << "-->" << "login" << endl;

                    FD_SET(connfd, &fdall);
                    if (maxfd < connfd)
                    {
                        maxfd = connfd;
                    }
                }
                else
                {
                    char msg_buf[1024] = {0x00};
                    ret = recv(i, msg_buf, sizeof(msg_buf), 0);
                    if (ret == -1)
                    {
                        perror("recv");
                        exit(EXIT_FAILURE);
                    }

                    if (ret == 0)
                    {
                        close(i);
                        FD_CLR(i, &fdall);
                        cout << clients_addr[i] << "-->" << "log out" << endl;
                    }
                    else
                    {
                        cout << clients_addr[i] << "-->" << msg_buf << endl;
                        ret = send(i, msg_buf, sizeof(msg_buf), 0);
                    }

                }
            }

        }

    }

    return 0;
}

和之前博文中代码没有大的区别,只是在程序入口处增加了一个daemon系统调用。

代码中需要注意的是:
(1)没有给出客户端代码。客户端测试可用telnet或netcat。(建议使用netcat,因为telnet只支持TCP,而netcat对TCP/UDP都提供了支持)
(2)使用了C++11标准中的容器unordered_map(哈希表)来存储ip地址。这意味着编译时需要添加-std=c++11或-std=c++0x选项。在使用只支持C++03/98的旧编译器时可使用std::map(红黑树)来替代本文中的unordered_map。
此时服务端代码已经完成。下一步是实现service *** start/stop命令对其进行管理的支持。
linux通过shell脚本来完成此工作。
在etc/init.d中创建一个shell脚本daemon_test (不要忘记添加可执行权限!)
#!bin/bash
prog=/home/vagrant/learn_linux/daemon/daemon_test     
lock=/home/vagrant/learn_linux/daemon/test.lock

start(){                                 
        $prog start
        echo "daemon_test starting"
        touch $lock
}
stop(){
        sudo  pkill daemon_test
        rm -rf $lock
}
status(){                        
        if [ -e $lock ];then
            echo "daemon_test running"
        else
            echo "daemon_test already stoped"
        fi
}
restart(){              
        stop
        start        
}
case "$1" in        
"start")
        start      
        ;;
"stop")            
        stop
        ;;
"status")             
        status
        ;;
"restart")            
        restart
        ;;
*)                
       # echo "??:$0 start|stop|status|restart"
        ;;
esac
 
脚本代码中需要注意的地方:
(1)脚本中的prog是进程文件的绝对路径,lock是锁定文件的绝对路径。
(2)在stop函数中通常使用killproc指令来结束进程。我的系统(ubuntu 14.04.3)对这个命令没有支持。因此使用pkil来代替。
此时就可以用service daemon_test start/stop来管理守护程序了。

还剩下最后一步:将守护程序设置为开机启动。
有三种办法:(1)RHEL/centos支持的chkconfig。(2)debian/ubuntu支持的update-rc.d (3)即将成为linux事实标准的systemd
我的系统(ubuntu 14.04.3)只提供了对(2)的支持。故暂选择使用 update-rc.d :
update-rc.d daemon_test defaults 99
如此守护程序daemon_test便可开机启动了。
在程序作了改动或代码重新编译之后,要先删掉开机启动项然后重新添加才能生效。删除指令如下:
update-rc.d -f daemon_test remove
至此本文的目的已达到。
PS:(1)笔者曾接触过windows下服务程序的开发。仅仅从windows 服务与linux daemon的开发对比来看,不得不说还是linux对程序员更友好。。。
	(2)本文中前面的守护进程代码没有对信号做处理,错误处理也可抽出来。因仅作示例之用,没做优化。
	(3)文中使用的编译器是gcc(g++) 4.6.3。
	(4)本文中提到的服务/守护程序不等于通常所说客户/服务器模型中的“服务端程序”。两者不是一个概念。
	(5) daemon调用不是posix标准的一部分,最早出现在bsd 4.4中。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值