0%

【Linux技术分享】systemctl和service启动服务

Linux开启自启服务器

网上很多介绍systemctl和service的区别和使用,我觉得都没有说到点子上,本文介绍这两个自启动命令,并且加自己经常使用具体方法。

systemctl

  1. 背景说明

    systemd,当系统启动之后systemd就会成功系统的第一个进程,而我们这个systemctl命令就是systemd命令中最重要的命令之一,用于管理系统。

    systemctl命令在centos6及以下版本是不支持的,调用这个命令会报命令未找到 or command not found错误,但是在7以上Linux为了提高系统的启动速度,尽可能的增加多个进程同时启动,所以就支持了systemd,也就是systemctl命令。

  2. 调用方法

    systemctl 命令 说明
    systemctl stop 停止
    systemctl start 开始
    systemctl status 查看状态
    systemctl mark 注销
    systemctl suspend 进入睡眠
    systemctl hibermate 进入休眠
    systemctl get-default 获得当前运行级别
  3. 具体调用逻辑

    关键词daemon:后台服务进程,常驻。

    文件保存路径主要是在以下三个路径,我自己写的service文件通常会用脚本,拷贝到下面三个路径中。

    • /usr/lib/systemd/system
    • /run/systemd/system
    • /etc/systemd/system

    unit文件分很多种,我主要讲一下service文件和Target文件,这两个文件在写linux应用的时候用的比较多一点。

    1. service:文件扩展名为.service,用于定义系统服务,文件内容类似于desktop文件。

      [unit]段常用的选项

      • Description:描述信息,意义性描述
      • After:定义unit的启动顺序,晚于某服务启动
      • Requies:依赖其他的units,当依赖的服务没启动,是不能启动此服务的,强依赖
      • Waints:和上面一样,弱依赖
      • Conficts:定义units冲突关系(不太明白)

      [Service]段常用选项

      • Type:用于定义影响ExecStart和相关参数功能unit进程启动类型

        1. simple:默认值,这个daemon主要由execstart后面所写的字符串来启动,启动后常驻内存。
        2. forking:由 ExecStart 启动的程序通过 spawns 延伸出其他子程序来作为此 daemon 的主要服务。原生的父程序在启动结束后就会终止运行,ps:类似于管道启动
        3. oneshot:与simple类似的启动方式类似,但是工作完就结束,不常驻内存
        4. dbus:与simple类似的启动方式类似,但要设置DbusName=XX,daemon程序才会运行。
        5. notify:暂时不了解
        6. 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
#!/bin/sh

echo 1 | evtest >>/home/listen.txt

注意:脚本启动都是已root权限启动,方便调试。非root权限无法启动service服务

service

  1. 背景说明

    service本质为进程,是linux最早控制服务的一种手段,并不是所有linux都有这个命令,主要是在redhat,fedora,mandriva,centos中 ,这个命令的可执行程序在/sbin下(一般进程都是在/usr/bin下)

  2. 调用方法

    其实没什么调用方法,主要是看你service下的逻辑支持是什么实现,就可以怎么实现。
    比如service dbus start,其实调用dbus这个服务文件中start的函数实现。

  3. 具体实现

    看个栗子,比如我本机的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
    #!/bin/sh
    ### 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
    #!/bin/bash
    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,比较规范方便