一般在Linux下面启动一个服务,比如samba服务,使用命令service smb start, 还有一个就是通过/etc/init.d/smb start. 为什么能通过这个命令启动一个或者服务。
这和两个脚本有关系。我们通过which service可以知道service在/sbin/service ,smb 在/etc/init.d/smb。 我们近距离观察会发现这两个文件实际上是脚本。重点观察smb文件,会发现一些很有意思的事。
start函数里面有一个daemon。daemon是干什么的?
Daemon是长时间运行的进程,通常在系统启动后就运行,在系统关闭时才结束。一般说Daemon程序在后台运行,是因为它没有控制终端,无法和前台的用户交互。Daemon程序一般都作为服务程序使用,等待客户端程序与它通信。我们也把运行的Daemon程序称作守护进程。
比如,我们的网络服务程序,可以在完成创建套接口,绑定套接口,设置套接口为监听模式后,变成守护进程进入后台执行而不占用控制终端,这是网络服务程序的常用模式。UNIX下的网络服务程序,如Web Server,FTP,Telnet一般都是由守护进程(Daemon)来实现的。守护进程不占用终端,在后台运行。UNIX的守护进程一般都命名为 *d 的形式,如httpd,telnetd等等。
Daemon程序编写规则
编写Daemon程序有一些基本的规则,以避免不必要的麻烦。
1、首先是程序运行后调用fork,并让父进程退出。子进程获得一个新的进程ID,但继承了父进程的进程组ID。
2、调用setsid创建一个新的session,使自己成为新session和新进程组的leader,并使进程没有控制终端(tty)。
3、改变当前工作目录至根目录,以免影响可加载文件系统。或者也可以改变到某些特定的目录。
4、设置文件创建mask为0,避免创建文件时权限的影响。
5、关闭不需要的打开文件描述符。因为Daemon程序在后台执行,不需要于终端交互,通常就关闭STDIN、STDOUT和STDERR。其它根据实际情况处理。另一个问题是Daemon程序不能和终端交互,也就无法使用printf方法输出信息了。
另一个问题是Daemon程序不能和终端交互,也就无法使用printf方法输出信息了。我们可以使用syslog机制来实现信息的输出,方便程序的调试。
#include
#include
#include
#include
#include
#include
#include
intdaemon_init(void)
{
pid_t pid;
if((pid = fork()) < 0)
return(-1);
elseif(pid != 0)
exit(0); /* parent exit */
/* child continues */
setsid(); /* become session leader */
chdir("/");/* change working directory */
umask(0); /* clear file mode creation mask */
close(0); /* close stdin */
close(1); /* close stdout */
close(2); /* close stderr */
return(0);
}
voidsig_term(intsigno)
{
if(signo == SIGTERM)
/* catched signal sent by kill(1) command */
{
syslog(LOG_INFO, "program terminated.");
closelog();
exit(0);
}
}
intmain(void)
{
if(daemon_init() == -1)
{
printf("can't fork self/n");
exit(0);
}
openlog("daemontest", LOG_PID, LOG_USER);
syslog(LOG_INFO, "program started.");
signal(SIGTERM, sig_term); /* arrange to catch the signal */
while(1)
{
sleep(1); /* put your main program here */
}
return(0);
}
使用如下命令编译该程序: gcc -Wall -o daemontestd daemontest.c编译完成后生成名为daemontest的可执行程序,执行./daemontestd来测试程序的运行。
使用ps axj命令可以显示系统中已运行的daemon程序的信息,包括进程ID、session ID、控制终端等内容。部分显示内容:
PPID PID PGID SID TTY TPGID STAT UID TIME COMMAND
1 17765 17765 17765 ? -1 Ss 0 0:00 ./daemontestd
从中可以看到daemontest程序运行的进程号为17765。
我们再来看看/var/log/messages文件中的信息:
Feb 3 17:09:30 localhost daemontestd[17765]: program started.
显示了我们在程序中希望输出的信息。
使用kill 17765命令来杀死这个进程,/var/log/messages文件中就会有如下的信息:
Feb 3 17:12:26 localhost daemontest[17765]: program terminated.
再使用ps axj命令检查,发现系统中daemontest进程已经没有了。
重点是这个脚本:daemomtest
#!/bin/sh
#
# chkconfig: - 91 35
# description: Starts and stops the daemontest daemon \
#
#
# Source function library.
if [ -f /etc/init.d/functions ] ; then
. /etc/init.d/functions
elif [ -f /etc/rc.d/init.d/functions ] ; then
. /etc/rc.d/init.d/functions
else
exit 1
fi
# Avoid using root's TMPDIR
unset TMPDIR
# Source networking configuration.
. /etc/sysconfig/network
# Check that networking is up.
[ ${NETWORKING} = "no" ] && exit 1
RETVAL=0
start() {
KIND="daemontest"
echo -n $"Starting $KIND services: "
daemon daemonestd -D
RETVAL=$?
echo
[ $RETVAL -eq 0 ] && touch /var/lock/subsys/sipserver || \
RETVAL=1
return $RETVAL
}
stop() {
KIND="daemontest
echo -n $"Shutting down $KIND services: "
killproc daemonestd
RETVAL=$?
echo
[ $RETVAL -eq 0 ] && rm -f /var/lock/subsys/sipserver
return $RETVAL
}
restart() {
stop
start
}
case "$1" in
start)
start
;;
stop)
stop
;;
restart)
restart
;;
reload)
reload
;;
status)
rhstatus
;;
configtest)
configtest
;;
condrestart)
[ -f /var/lock/subsys/smb ] && restart || :
;;
*)
echo $"Usage: $0 {start|stop|restart}"
exit 2
esac
exit $?
将daemontest 拷贝到/etc/init.d/目录下面, 将daemontestd拷贝到/usr/sbin/目录下面 然后就可以通过service daemontestd start开启动