Linux平台下的service程序编写指南
Hu Dennis Sep 24, 2010转载注明出处http://blog.csdn.net/gobitan
摘要:本文主要介绍了如何编写一个service服务所涉及的两个主要方面。1)将一个普通程序装成daemon程序;2)编写service方式的shell脚本。并以C语言为例,基于Red hat 企业版5.3平台演示了一个service服务程序从代码编写到脚本调试以及测试运行全过程。
(一) Service介绍
Service是一种被称为守护进程(daemon)的程序。它通常一旦启动后就在后台持续运行,通常在等待什么输入或者监控系统有什么变化。例如Apache服务器有一个叫httpd的守护进程监听80端口以等待http请求。
Service程序通常通过service命令来启动或停止等。例如apache服务的启动可通过”service httpd start”来启动。
通过” chkconfig --list”可以查看系统当前所有的service服务。
通过” service --status-all”可以查看系统当前所有service服务的状态。
要自己编写一个类似httpd的service方式的服务应该包括两个部分:(1)将普通程序包装成daemon程序;(2)编写service控制脚本来管理daemon程序。
因此,每个service服务在/etc/rc.d/目录下都对应一个可执行的脚本。如apache的httpd对应/etc/init.d/httpd。
参考资料:http://bobpeers.com/linux/services.php
(二)如何将普通程序包装成daemon程序
这里仅介绍一种较为简单的方式,关于其原理及更详细的介绍请参见本节参考资料。
Linux提供了一个名叫daemon的函数来初始化环境,如下:
int daemon(int nochdir, int noclose);
使用该函数需加入#include <stdlib.h>头文件包含。调用该函数之后,其后的程序将会以daemon方式运行。
下面以Hello world为例,因为daemon程序是一个持续运行程序,为了测试运行本文以while+sleep来模拟。
- #include <stdio.h>
- #include <stdlib.h>
- int main()
- {
- daemon(0,0);
- while(1)
- {
- printf("Hellow World!/n");
- sleep(100000);
- }
- }
保存上面的测试程序为dennis.c,执行如下编译命令得到dennisd守护进程程序。
#gcc -c dennis.c
# gcc -o dennisd dennis.o
实际运行中,例如你编写了一个tcp服务器程序,可以程序写成一个函数(如tcp_server),然后将while部分更换为你的函数tcp_server()即可。如下:
- #include <stdio.h>
- #include <stdlib.h>
- int main()
- {
- daemon(0,0);
- tcp_server();
- }
参考资料:http://www.netzmafia.de/skripten/unix/linux-daemon-howto.html
(三)如何编写service控制脚本
首先脚本必须放在/etc/init.d/目录下,因此本文在先创建/etc/init.d/dennisd脚本文件。内容如下:
- #!/bin/bash
- #
- # Description: This shell script takes care of starting and stopping dennis
- # Hu Dennis created on Sep. 24th, 2010
- #
- # Source function library
- . /etc/init.d/functions
- #the service name for example: dennis
- SNAME=dennisd
- #the full path and name of the daemon program
- #Warning: The name of executable file must be identical with service name
- PROG=/usr/bin/$SNAME
- # start function
- start() {
- #check the daemon status first
- if [ -f /var/lock/subsys/$SNAME ]
- then
- echo "$SNAME is already started!"
- exit 0;
- else
- action "Starting $SNAME ..." $PROG
- [ $? -eq 0 ] && touch /var/lock/subsys/$SNAME
- exit 0;
- fi
- }
- #stop function
- stop() {
- echo "Stopping $SNAME ..."
- killproc $SNAME
- rm -rf /var/lock/subsys/$SNAME
- }
- case "$1" in
- start)
- start
- ;;
- stop)
- stop
- ;;
- reload|restart)
- stop
- start
- ;;
- status)
- status $SNAME
- ;;
- *)
- echo $"Usage: $0 {start|stop|restart|status}"
- exit 1
- esac
#chmod +x dennisd
(四)常见调试错误
在调试该脚本的时候常见的集中错误如下:
(1)dennisd: unrecognized service:先检查/etc/rc.d/init.d/dennisd是否存在,然后再看该文件是否有可执行权限。
(2)env: /etc/init.d/dennisd: No such file or directory:这种情况一般是windows与linux的格式兼容问题,执行dos2unix将dos格式转换程序linux格式(ubuntu下为fromdos命令)。
(3)Usage: status [-p pidfile] {program}: status后面的参数不对
(4)dennisd dead but subsys locked:可执行程序的名字需要与service名字保持一致。
(五)测试运行
将第二步编译得到的dennisd程序拷贝至/usr/bin/目录下,然后执行如下命令测试:
(1)启动服务
[root init.d]# service dennisd start
Starting dennisd ... [ OK ]
通过ps命名确认dennisd已经启动:
[root init.d]# ps -ef|grep dennisd
root 3885 1 0 14:30 ? 00:00:00 /usr/bin/dennisd
(2)查看状态
[root init.d]# service dennisd status
dennisd (pid 3885) is running...
(3)停止服务
[root init.d]# service dennisd stop
Stopping dennisd ...
再执行status命令查看状态
[root init.d]# service dennisd status
dennisd is stopped
(4)多次启动
[root init.d]# service dennisd start
Starting dennisd ... [ OK ]
[root init.d]# service dennisd start
dennisd is already started!
如果服务已经启动,系统会提示而不会启动多个。
(六)结束语
本文以C语言为例,在linux环境下演示了一个service服务程序从代码编写到脚本调试以及测试运行全过程,希望对有此需求的朋友有所帮助!
Hu Dennis 2010-9-24 于成都 Email: dennis.hu.cd at gmail.com