linux笔记:开机自动运行程序(代码片段)

NaisuXu NaisuXu     2022-12-28     254

关键词:

目的

开机自动运行程序,或者说系统启动时自动运行程序,这是经常会需要用到的功能。在linux中实现随系统启动运行程序的功能通常有三种(或者说两种)方法。本篇文章将对相关内容做个简单的介绍。

rc.local

rc.local 是linux中的一个文件,通常路径为 /etc/rc.local/etc/rc.d/rc.local (当然也可能没有,详见后面章节内容)。linux系统启动过程后期阶段会读取该文件并执行其中的命令,我们可以把开机自动运行的程序在这里通过命令调用运行。


在上面演示中我在 rc.local 文件中添加了一行命令,当我重启系统之后该条命令被执行了。

rc.local 文件在编辑的时候需要注意两点:

  • 必须正确退出,通常在rc.local文件最后写上 exit 0 ,不然系统或程程序可能无法正确启动;
  • 如果要执行耗时操作最后将它放入后台(比如在,命令后面加 (空格) &),不然系统启动过程会阻塞在这里;

SysVinit

SysVinit是linux中常见的一个启动程序,它会在系统启动过程中去执行 /etc/init.d/ 目录下的脚本,这里面的这些脚本所运行的程序你可以简单理解为服务程序。你可以编写自己的服务程序放到这个目录中,然后启用它,那么系统启动时脚本中的程序就会运行。

/etc/init.d/ 目录下的脚本常见格式如下:

#!/bin/sh

# 下面段落注释在添加为服务过程中会被读取,条目基本不能少,不需要的内容可以为空
### BEGIN INIT INFO
# Provides:          servicename
# Required-Start:
# Required-Stop:
# Should-Start:
# Should-Stop:
# Default-Start:     2 3 4 5
# Default-Stop:
# Short-Description:
# Description:
### END INIT INFO

set -e # 遇到错误时退出脚本

fun_start() 

fun_stop() 

fun_status() 

# $1是传入的第二个参数,比如/etc/init.d/filename start这条命令,$1就是start
case $1 in
    start)
        fun_start
        ;;
    stop)
        fun_stop
        ;;
    restart)
        # $0是自身路径,即/etc/init.d/filename
        $0 stop
        $0 start
        ;;
    status)
        fun_status
        ;;
    *)
    # 上面命令都不是就打印个提示
	echo "Usage: /etc/init.d/filename start|stop|restart|status"
	exit 1
esac

exit 0

上面的脚本很好理解,就是运行脚本时根据传入的参数选择执行case/esac中的语句,比如把文件名为naisu的脚本文件放到/etc/init.d/路径下,使用 /etc/init.d/naisu start 命令时就会运行脚本中 start) 和 ;; 间的代码。通常这之间的操作可以封装成函数调用,不过不封装直接在这里写也可以。

下面是个最简单的脚本演示:

上面只是最基本的脚本而已,还没有成为服务,没有设置为开机运行,接下来还需要对此进行设置:

  • 根据前面介绍的格式补充脚本中 ### BEGIN INIT INFO### END INIT INFO 部分注释
  • Ubuntu、Debian等系统中使用下面方式设置服务
    • update-rc.d naisu.sh defaults 设置服务开机启动
    • update-rc.d -f naisu.sh remove 关闭服务开机启动
  • Red Hat、CentOS等系统中使用下面方式设置服务
    • chkconfig naisu.sh --add 将脚本添加为服务
    • chkconfig naisu.sh on 设置服务开机启动
    • chkconfig naisu.sh off 关闭服务开机启动


上面演示中可以看到将脚本设置为服务,开机启动时执行了start操作。

设置为服务后可以使用 service name status 来查看状态,不管有没有在脚本中写status相关的操作都可以:

除此之外还可以使用 service name startservice name stop 等操作来执行脚本。

Systemd

Systemd是目前linux中越来越常见的一个启动程序,它会在系统启动过程中去读取放在 /etc/systemd/system 目录下的配置文件并安装其中内容执行程序。我们在使用的时候并不是把配置文件放在这里的,而是放在 /usr/lib/systemd/system//lib/systemd/system/ 目录下。这样我们在使能该配置文件的时候会生成一个到 /etc/systemd/system 的软链接,它就会开机运行了;当我们禁用该配置文件的时候会取消该软链接。

我们这里需要编写的配置文件的文件名通常被命名为 name.service ,其格式如下:

[Unit]
Description=name

[Service]
Type=forking
ExecStart=path

[Install]
WantedBy=multi-user.target

这个配置文件官方的名称是Unit文件,Systemd管理的是一个个的Unit,而服是其中的一种Unit。服务的Unit文件内存主要分为三个部分:

[Unit] 部分主要用于描述该服务以及该服务与其它服务间的关系,常用的可选字段主要如下:

字段说明
Description当前Unit的文本描述
Requires当前Unit依赖的其他Unit,如果它们没有运行,当前Unit会启动失败
Before如果该字段指定的Unit也要启动,那么必须在当前Unit之后启动
After如果该字段指定的Unit也要启动,那么必须在当前Unit之前启动

[Service] 部分主要用于描述服务启动相关的一些内容,常用的可选字段主要如下:

字段说明
Type当前服务的启动方式它有很多选项,这里列出部分:
simple 执行ExecStart指定的命令,启动的进程为主进程
forking ExecStart字段将以fork()方式启动,后台服务通常使用此项
oneshot 类似于simple,但只执行一次,Systemd 会等它执行完,才启动其他服务
idle 其他任务执行完毕当前服务才会运行(只对需要向控制台输出信息的服务有效,最大延迟5s)
ExecStart服务启动时执行的命令
RemainAfterExit当服务的所有进程都结束时服务是否标示为 active 状态
yes 标示为active 状态;no 默认值

[Install] 部分主要用于描述服务安装信息,常用的可选字段主要如下:

字段说明
WantedBy当前Unit所在的Target
当使能该Unit时,当前配置文件会生成软链接到/etc/systemd/system目录下面以Target名+.wants后缀构成的子目录中
这里的Target相当于SysVinit中的runlevel,常用值multi-user.target相当于runlevel 2 3 4
RequiredBy当前Unit依赖的Target
Also当前Unit使能时,会被同时使能的其他Unit

Unit文件编写相关更多内容可以参考下面链接:
https://www.freedesktop.org/software/systemd/man/systemd.unit.html
https://www.freedesktop.org/software/systemd/man/systemd.service.html

在准备好要运行的程序,编写好相应的配置文件,将配置文件放到 /usr/lib/systemd/system//lib/systemd/system/ 目录下,这样我们就可以正式使用了:

  • 使用 sudo systemctl enable name.service 可以将其设置为开机启动;
  • 使用 sudo systemctl disable name.service 可以停用开机启动;
  • 使用 systemctl status name.service 可以查看其状态;

除了上面的一些操作, systemctl 还支持更多操作,比如 start stop reload is-enable 等更多操作。

在Systemd中实现rc.local

在有些使用了Systemd的系统中它会有个内置的方式来模拟传统的SysVinit的一些功能,但也可能没有。这时候你要是怀念rc.local的话也可以自己实现它。

  • 创建/etc/rc.local文件,然后按照正常方式添加需要开机运行的程序;
  • 创建Systemd的配置文件 rc.local.service (文件名也可以自定)放到 /usr/lib/systemd/system//lib/systemd/system/ 目录下,配置文件内容如下:
    [Unit]
    Description=Simulate /etc/rc.local
    ConditionPathExists=/etc/rc.local
    After=network.target
    
    [Service]
    Type=forking
    ExecStart=/etc/rc.local
    TimeoutSec=0
    RemainAfterExit=yes
    GuessMainPID=no
    
    [Install]
    WantedBy=multi-user.target
    
  • 使用 sudo systemctl enable rc.local.service 将其设置为开机启动运行;

注意事项

不管用上面哪种方式都有一点需要注意的地方:

系统在启动过程中很多功能可能没有那么快准备就绪,这个时候如果进行一些需要这些功能的操作的话操作可能会失败。

比如你在这个阶段去访问网上的内容,但这个阶段可能系统还未连上网,那你这个操作就会失败。你可以在真正要进行的任务前加个延时,然后整个丢后台去执行。

SysVinit和Systemd的联系与区别

linux在完成最基本的启动后会启动pid为1的程序,这个程序笼统的被称为init,这是一个守护进程(daemon,相当于windows中的服务程序),用于初始化与管理系统相关的一些东西。init只要能实现相应的功能职责就行,但具体是用哪个程序并不强制,这里面比较出名的就是SysVinit和Systemd。

SysVinit以前非常流行,特点概括点来说就是需要初始化启动的项目一步一步执行,上面的rc.local就是用它启动过程中的一个环节,这种方式启动较慢;Systemd是后面出的,各个初始化项目可以并行执行,启动相对要快些,而且功能很强大,所以现在变得越来越流行。

在linux中实现随系统启动运行程序的功能时,如果发现 rc.local & SysVinit 这个方式不工作,那就换 Systemd 试试,反之亦然,不要在一棵树上吊死。

SysVinit和Systemd中一些操作有所差异,可以查看下面图表:




图片来源:https://linoxide.com/systemd-vs-sysvinit-cheatsheet/

总结

在linux中实现随系统启动运行程序的功能还是比较简单的,主要就是上面的一些方式了。

linux开机自动执行命令或自动启动程序(rc.local)(代码片段)

linux开机的最后会执行/etc/rc.local,因此可以在此脚本里面添加shell命令自动执行或者自动启动某个进程。比如自动输出信息:#!/bin/sh-e##rc.local##Thisscriptisexecutedattheendofeachmultiuserrunlevel.#Makesurethatthescriptwill"exit0" 查看详情

linux学习笔记(代码片段)

Linux学习笔记(一)Author:akynazhBlog:akynazh.siteLinux开机CMOS是记录各项硬件参数且嵌入在主板上面的储存器BIOS则是一个写入到主板上的一个固件(固件就是写入到硬件上的一个软件程序)。这个BIOS就是在开机的时候,计算... 查看详情

关于配置sshd的一些操作(代码片段)

...动化配置的一些操作自动化配置服务器,其本质就是开机自动运行脚本开机自动运行脚本的思路1.以某一用户身份去登陆服务器,在linux系统下会默认运行用户家目录下的.bashrc配置文件2.linux或unix系统下,/etc/rc.d/rc.loca... 查看详情

关于配置sshd的一些操作(代码片段)

...动化配置的一些操作自动化配置服务器,其本质就是开机自动运行脚本开机自动运行脚本的思路1.以某一用户身份去登陆服务器,在linux系统下会默认运行用户家目录下的.bashrc配置文件2.linux或unix系统下,/etc/rc.d/rc.loca... 查看详情

linux开机自启动命令

参考技术A  用户想要在linux开机的时候自启动相关的程序。那我们要怎么设置呢?下面由我为大家整理了linux下开机自启动命令的相关知识,希望对大家有所帮助!  linux下开机自启动命令  1.开机启动时自动运行程序  Linu... 查看详情

linux下开机启动脚本程序

...,使用各种编译器运行程序,但是有时候我们需要在系统开机的时候就自动运行这些程序,下面我们就来看一下在Linux下将脚本设定成开机自动运行的方法。Linux在启动时,会自动执行/etc/rc.d目录下的初始化程序,因此我们可以... 查看详情

debian系列-开机启动程序(代码片段)

Debian系列-开机启动程序文章目录Debian系列-开机启动程序摘要1修改/etc/profile2输入密码,以管理员权限运行程序关键字:开机启动、Debian、Linux、profile、etc内容背景:最近项目终于切到Linux下开发了,所以最近的记... 查看详情

linux学习笔记(代码片段)

目录1、Linux基本篇1、Linux目录结构2、vi和vim2.1基本介绍2.2常见的三种模式及转换2.3练习案例3、开机、重启、登录、注销4、用户管理4.1添加用户4.2删除用户4.3查询用户信息4.4切换用户4.5用户组5、实用指令5.1指定运行级别5.2例子&#x... 查看详情

如何在linux嵌入式开发板上开机自动运行用户应用程序

参考技术A开机后自动运行用户的应用程序或启动系统服务的命令保存在开发板根文件系统的/usr/etc/rc.local文件中。有的开发板开机后自动运行图形界面程序,需要按住ctrl+c让开发板进入到linux的SHELL提示符界面。其实可通过注释... 查看详情

linux设置开机自启(代码片段)

文章目录方法1:systemctlenable方法2:/etc/rc.local方法3:chkconfig应用场景方法1:systemctlenable启用开机自动启动的命令systemctlenable关闭开机自动启动的命令systemctldisable示例:设置MySQL开机自动启动systemctlenablemysql 查看详情

linux重要的目录之etc(代码片段)

...码、网关、主机名配置/etc/resolv.confDNS服务器配置/etc/fstab开机自动挂载系统,所有分区开机都会自动挂载/etc/inittab设定 查看详情

关于linux下redis自动化部署的一些笔记(代码片段)

写在前面分享一些安装redis的笔记博文内容涉及:通过源码编译和yum安装redisDemo通过二进制文件和systemd运行redis的配置方式服务管理,配置的文件的简单介绍ansibleredis角色ansible-role-redis编写通过ansible自动化安装理解不足... 查看详情

开机自动运行程序代码?

参考技术A'先加载修改注册表的API函数:PublicDeclareFunctionRegSetValueLib"advapi32.dll"Alias"RegSetValueA"(ByValhKeyAsLong,ByVallpSubKeyAsString,ByValdwTypeAsLong,ByVallpDataAsString,ByValcbDataAsLong)AsLongPublicDeclareFunctionRegCreateKeyLib"advapi32.dll&q... 查看详情

jvm笔记(代码片段)

一、什么是JVM定义JavaVirtualMachine,JAVA程序的运行环境(JAVA二进制字节码的运行环境)好处一次编写,到处运行自动内存管理,垃圾回收机制数组下标越界检查比较JVMJREJDK的区别二、内存结构整体架构1、程序... 查看详情

怎样使程序开机自启动和程序挂掉自动重启(代码片段)

碰到这个问题基本都是属于服务端工程部署的事情,在部署好我们的算法服务后,怎么样保证当服务器关机重启后,我们的算法服务自动重新运行?如果碰到异常bug使得我们的程序挂掉后,怎样让它重新运行&#... 查看详情

怎样使程序开机自启动和程序挂掉自动重启(代码片段)

碰到这个问题基本都是属于服务端工程部署的事情,在部署好我们的算法服务后,怎么样保证当服务器关机重启后,我们的算法服务自动重新运行?如果碰到异常bug使得我们的程序挂掉后,怎样让它重新运行&#... 查看详情

怎样使程序开机自启动和程序挂掉自动重启(代码片段)

碰到这个问题基本都是属于服务端工程部署的事情,在部署好我们的算法服务后,怎么样保证当服务器关机重启后,我们的算法服务自动重新运行?如果碰到异常bug使得我们的程序挂掉后,怎样让它重新运行&#... 查看详情

linux下设置字符界面开机启动及系统启动流程介绍(代码片段)

Linux下设置字符界面开机启动及系统启动流程介绍一、临时设置开机运行目标1.查看当前默认的运行target2.临时切换当前的运行target3.设置下一次开机的target二、centos中重要的target1.systemd介绍2.系统重要的target3.查看系统的服务4.查... 查看详情