Linux开启自启服务器
网上很多介绍systemctl和service的区别和使用,我觉得都没有说到点子上,本文介绍这两个自启动命令,并且加自己经常使用具体方法。
systemctl
背景说明
systemd,当系统启动之后systemd就会成功系统的第一个进程,而我们这个systemctl命令就是systemd命令中最重要的命令之一,用于管理系统。
systemctl命令在centos6及以下版本是不支持的,调用这个命令会报
命令未找到
orcommand not found
错误,但是在7以上Linux为了提高系统的启动速度,尽可能的增加多个进程同时启动,所以就支持了systemd,也就是systemctl命令。调用方法
systemctl 命令 说明 systemctl stop 停止 systemctl start 开始 systemctl status 查看状态 systemctl mark 注销 systemctl suspend 进入睡眠 systemctl hibermate 进入休眠 systemctl get-default 获得当前运行级别 具体调用逻辑
关键词daemon:后台服务进程,常驻。
文件保存路径主要是在以下三个路径,我自己写的service文件通常会用脚本,拷贝到下面三个路径中。
- /usr/lib/systemd/system
- /run/systemd/system
- /etc/systemd/system
unit文件分很多种,我主要讲一下service文件和Target文件,这两个文件在写linux应用的时候用的比较多一点。
service:文件扩展名为.service,用于定义系统服务,文件内容类似于desktop文件。
[unit]段常用的选项
- Description:描述信息,意义性描述
- After:定义unit的启动顺序,晚于某服务启动
- Requies:依赖其他的units,当依赖的服务没启动,是不能启动此服务的,强依赖
- Waints:和上面一样,弱依赖
- Conficts:定义units冲突关系(不太明白)
[Service]段常用选项
Type:用于定义影响ExecStart和相关参数功能unit进程启动类型
- simple:默认值,这个daemon主要由execstart后面所写的字符串来启动,启动后常驻内存。
- forking:由 ExecStart 启动的程序通过 spawns 延伸出其他子程序来作为此 daemon 的主要服务。原生的父程序在启动结束后就会终止运行,ps:类似于管道启动
- oneshot:与simple类似的启动方式类似,但是工作完就结束,不常驻内存
- dbus:与simple类似的启动方式类似,但要设置DbusName=XX,daemon程序才会运行。
- notify:暂时不了解
- idle:与simple类似,要执行这个daemon 必须要所有的工作都顺利执行完毕后才会执行。这类的daemon 通常是开机到最后才执行即可的服务
EnvironmentFile:环境配置文件
ExecStart:启动脚本或者命令进程
ExecStop:停止脚本或者命令进程
[install]段常用选项
- Alias:暂时不了解
- RequiredBy:被哪些units所依赖,强
- WantedBy:如上,弱
2. Target:文件扩展为.target,用于模拟实现“运行级别”。
举个实际性的例子,如果我完成了一个程序的编写或者脚本的编写,想让他开机启动并且常驻的话,我就可以自己写一个servcie文件,放入上文中三条路径中,类似下面这个文件。
1
2
3
4
5
6
7
8
9
10
11
[Unit]
Description=test
[Service]
Type=forking
ExecStart=/home/mecry/DemonShell
ExecStop=/bin/kill -HUP $MAINPID
PrivateTmp=true
[Install]
WantedBy=multi-user.target
ExecStart就是我在电脑中目标脚本,脚本中可以写一些监视开机启动的命令,输出到指定文件夹下,比如我监听笔记本闭合事件,开关机时间等,具体日志输出就在/home中的listen.txt文件下。
1
2
3
echo 1 | evtest >>/home/listen.txt
注意:脚本启动都是已root权限启动,方便调试。非root权限无法启动service服务
service
背景说明
service本质为进程,是linux最早控制服务的一种手段,并不是所有linux都有这个命令,主要是在redhat,fedora,mandriva,centos中 ,这个命令的可执行程序在/sbin下(一般进程都是在/usr/bin下)
调用方法
其实没什么调用方法,主要是看你service下的逻辑支持是什么实现,就可以怎么实现。
比如service dbus start
,其实调用dbus这个服务文件中start的函数实现。具体实现
看个栗子,比如我本机的dbus服务,文件存放都存放在
/etc/init.d/
目录下。1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
### BEGIN INIT INFO
# Provides: dbus
# Required-Start: $remote_fs $syslog
# Required-Stop: $remote_fs $syslog
# Default-Start: 2 3 4 5
# Default-Stop:
# Short-Description: D-Bus systemwide message bus
# Description: D-Bus is a simple interprocess messaging system, used
# for sending messages between applications.
### END INIT INFO
# -*- coding: utf-8 -*-
# Debian init.d script for D-BUS
# Copyright © 2003 Colin Walters <walters@debian.org>
# Copyright © 2005 Sjoerd Simons <sjoerd@debian.org>
set -e
DAEMON=/usr/bin/dbus-daemon
UUIDGEN=/usr/bin/dbus-uuidgen
UUIDGEN_OPTS=--ensure
NAME=dbus
DAEMONUSER=messagebus
PIDDIR=/var/run/dbus
PIDFILE=$PIDDIR/pid
DESC="system message bus"
test -x $DAEMON || exit 0
. /lib/lsb/init-functions
# Source defaults file; edit that file to configure this script.
PARAMS=""
if [ -e /etc/default/dbus ]; then
. /etc/default/dbus
fi
create_machineid() {
# Create machine-id file
if [ -x $UUIDGEN ]; then
$UUIDGEN $UUIDGEN_OPTS
fi
}
start_it_up()
{
if [ ! -d $PIDDIR ]; then
mkdir -p $PIDDIR
chown $DAEMONUSER $PIDDIR
chgrp $DAEMONUSER $PIDDIR
fi
if ! mountpoint -q /proc/ ; then
log_failure_msg "Can't start $DESC - /proc is not mounted"
return
fi
if [ -e $PIDFILE ]; then
if $0 status > /dev/null ; then
log_success_msg "$DESC already started; not starting."
return
else
log_success_msg "Removing stale PID file $PIDFILE."
rm -f $PIDFILE
fi
fi
create_machineid
log_daemon_msg "Starting $DESC" "$NAME"
start-stop-daemon --start --quiet --pidfile $PIDFILE \
--exec $DAEMON -- --system $PARAMS
log_end_msg $?
}
shut_it_down()
{
log_daemon_msg "Stopping $DESC" "$NAME"
start-stop-daemon --stop --retry 5 --quiet --oknodo --pidfile $PIDFILE \
--user $DAEMONUSER
# We no longer include these arguments so that start-stop-daemon
# can do its job even given that we may have been upgraded.
# We rely on the pidfile being sanely managed
# --exec $DAEMON -- --system $PARAMS
log_end_msg $?
rm -f $PIDFILE
}
reload_it()
{
create_machineid
log_action_begin_msg "Reloading $DESC config"
dbus-send --print-reply --system --type=method_call \
--dest=org.freedesktop.DBus \
/ org.freedesktop.DBus.ReloadConfig > /dev/null
# hopefully this is enough time for dbus to reload it's config file.
log_action_end_msg $?
}
case "$1" in
start)
start_it_up
;;
stop)
shut_it_down
;;
reload|force-reload)
reload_it
;;
restart)
shut_it_down
start_it_up
;;
status)
status_of_proc -p $PIDFILE $DAEMON $NAME && exit 0 || exit $?
;;
*)
echo "Usage: /etc/init.d/$NAME {start|stop|reload|restart|force-reload|status}" >&2
exit 2
;;
esac说明:实际上dbus的service,就是一个.sh脚本,如果有C或者sh的基础就很好弄懂是什么意思,比如
service dbus start
,service进程就会从/etc/init.d/
目录下找到dbus这个服务,这个sh脚本的case
参数传入start
,调用具体的实现逻辑。再举一个简单点的例子,和实际用法。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
SERVERNAME= "DemonShell"
start()
{
echo "start $SERVERNAME"
/home/mecry/DemonShell
echo "start $SERVERNAME ok!"
exit 0;
}
stop()
{
echo "stop $SERVERNAME"
killall $SERVERNAME
echo "stop $SERVERNAME ok!"
}
case "$1" in
start)
start
;;
stop)
stop
;;
restart)
stop
start
;;
*)
echo "usage: $0 start|stop|restart"
exit 0;
esac
exit和上面的systemctl一样,
/home/mecry/DemonShell
是我脚本存放位置,开机控制其启动,调用service也可以控制启动和关闭此脚本,非常方便。
systemctl和service优点缺点
建议使用systemctl,比较规范方便