转自:Linux如何实现开机启动程序详解
我们假设大家已经熟悉其它操作系统的引导过程,了解硬件的自检引导步骤,就只从Linux操作系统的引导加载程序(对个人电脑而言通常是LILO)开始,介绍Linux开机引导的步骤。
Linux开机引导的步骤
-
加载内核
LILO启动之后,如果你选择了Linux作为准备引导的操作系统,第一个被加载的东西就是内核。请记住此时的计算机内存中还不存在任何操作系统,PC(因为它们天然的设计缺陷)也还没有办法存取机器上全部的内存。因此,内核就必须完整地加载到可用RAM的第一个兆字节之内。为了实现这个目的,内核是被压缩了的。这个文件的头部包含着必要的代码,先设置CPU进入安全模式(以此解除内存限制),再对内核的剩余部分进行解压缩。 -
执行内核
内核在内存中解压缩之后,就可以开始运行了。此时的内核只知道它本身内建的各种功能,也就是说被编译为模块的内核部分还不能使用。最基本的是,内核必须有足够的代码设置自己的虚拟内存子系统和根文件系统(通常就是ext2文件系统)。一旦内核启动运行,对硬件的检测就会决定需要对哪些设备驱动程序进行初始化。从这里开始,内核就能够挂装根文件系统(这个过程类似于Windows识别并存取C盘的过程)。内核挂装了根文件系统之后,将启动并运行一个叫做init的程序。注意:在这里我们故意略去了Linux内核启动的许多细节,这些细节只有内核开发人员才感兴趣。 -
init进程
init进程是非内核进程中第一个被启动运行的,因此它的进程编号PID的值总是1。init读它的配置文件<code>/etc/inittab</code>,决定需要启动的运行级别(Runlevel)。从根本上说,运行级别规定了整个系统的行为,每个级别(分别由0到6的整数表示)满足特定的目的。
如果定义了initdefault级别,这个值就直接被选中,否则需要由用户输入一个代表运行级别的数值。输入代表运行级别的数字之后,init根据<code>/etc/inittab</code>文件中的定义执行一个命令脚本程序。
缺省的运行级别取决于安装阶段对登录程序的选择:是使用基于文本的,还是使用基于X-Window的登录程序。
rc命令脚本程序我们已经知道,当运行级别发生改变时,将由<code>/etc/inittab</code>文件定义需要运行哪一个命令脚本程序。这些命令脚本程序负责启动或者停止该运行级别特定的各种服务。由于需要管理的服务数量很多,因此需要使用rc命令脚本程序。其中,最主要的一个是<code>/etc/rc.d/rc</code>,它负责为每一个运行级别按照正确的顺序调用相应的命令脚本程序。我们可以想象,这样一个命令脚本程序很容易变得难以控制!为了防止这类事件的发生,需要使用精心设计的方案。
对每一个运行级别来说,在<code>/etc/rc.d</code>子目录中都有一个对应的下级目录。这些运行级别的下级子目录的命名方法是<code>rcX.d</code>,其中的X就是代表运行级别的数字。比如说,运行级别3的全部命令脚本程序都保存在<code>/etc/rc.d/rc3.d</code>子目录中。
在各个运行级别的子目录中,都建立有到<code>/etc/rc.d/init.d</code>子目录中命令脚本程序的符号链接,但是,这些符号链接并不使用命令脚本程序在<code>/etc/rc.d/init.d</code>子目录中原来的名字。
如果命令脚本程序是用来启动一个服务的,其符号链接的名字就以字母S打头;如果命令脚本程序是用来关闭一个服务的,其符号链接的名字就以字母K打头。许多情况下,这些命令脚本程序的执行顺序都很重要。如果没有先配置网络接口,就没有办法使用DNS服务解析主机名!为了安排它们的执行顺序,在字母S或者K的后面紧跟着一个两位数字,数值小的在数值大的前面执行。比如:<code>/etc/rc.d/rc3.d/S50inet</code>就会在<code>/etc/rc.d/rc3.d/S55named</code>之前执行(S50inet配置网络设置,55named启动DNS服务器)。存放在<code>/etc/rc.d/init.d</code>子目录中的、被符号链接上的命令脚本程序是真正的实干家,是它们完成了启动或者停止各种服务的操作过程。当<code>/etc/rc.d/rc</code>运行通过每个特定的运行级别子目录的时候,它会根据数字的顺序依次调用各个命令脚本程序执行。它先运行以字母K打头的命令脚本程序,然后再运行以字母S打头的命令脚本程序。对以字母K打头的命令脚本程序来说,会传递Stop参数;类似地对以字母S打头的命令脚本程序来说,会传递Start参数。编写自己的rc命令脚本在维护Linux系统运转的日子里,肯定会遇到需要系统管理员对开机或者关机命令脚本进行修改的情况。
有两种方法可以用来实现修改的目的:
如果所做的修改只在引导开机的时候起作用,并且改动不大的话,可以考虑简单地编辑一下/etc/rc.d/rc.local脚本。这个命令脚本程序是在引导过程的最后一步被执行的。
如果所做的修改比较细致,或者还要求关闭进程使之明确地停止运行,则需要在/etc/rc.d/init.d子目录中添加一个命令脚本程序。这个命令脚本程序必须可以接受Start和Stop参数并完成相应的操作。
第一种方法,编辑/etc/rc.d/rc.local脚本,当然是两种方法中比较简单的。如果想在这个命令脚本程序中添加内容,只需要使用喜欢的编辑器程序打开它,再把打算执行的命令附加到文件的末尾就可以了。这对一两行的修改来说的确很便利。如果确实需要使用一个命令脚本程序,这时必须选择第二个方法。编写一个rc命令脚本程序的过程并不像想象中那么困难。我们下面就给出一个例子,看看它是怎样实现的(顺便说一句,你可以把我们的例子当作范本,按照自己的需要进行修改和添加)。假设你打算每隔60分钟调用一个特殊的程序来弹出一条消息,提醒自己需要从键盘前面离开休息一会儿,命令脚本程序将包括下面几个部分:
- 关于这个命令脚本程序功能的说明(这样就不会在一年之后忘记它);
- 在试图运行它之前验证这个命令脚本程序确实存在;
- 接受start和stop参数并执行要求的动作。
参数给定后,我们就可以编写命令的脚本程序。这个程序很简单,大家可以自己编写一下,我在这里就不给出了。编写好新的命令脚本程序之后,再从相关的运行级别子目录中加上必要的符号链接,来控制这个命令脚本程序的启动或者停止。在我的印象中,只想让它在运行级别3或者运行级别5中启动,原因是我认为只有这两个运行级别才是日常工作的地方。最后,希望这个命令脚本程序在进入运行级别6(重启动)的时候被关闭。
激活或者禁止服务项目有的时候会发现,在引导的时候并不需要某个特定的服务被启动。如果你正在考虑使用Linux替换Windows NT的文件和打印服务器,就更是如此。我们已经知道,在特定的运行级别子目录中给符号链接改个名称,就可以让该服务不被启动,如把其名称的第一个字母由S改为K。一旦熟练掌握了命令行和符号链接,就会发现这是激活或者禁止服务的最快办法。在学习这个改名方法的时候,可能会觉得图形化的操作界面ksysv比较容易掌握。虽然它原来是设计使用在KDE环境里的,但在Red HatLinux 7.2下缺省安装的GNOME环境里也运行得很好。如果想启动它,只需简单地打开一个xterm窗口,并输入ksysv命令就可以了。屏幕上会出现一个窗口,其中列出了能够修改的全部参数,需要时还包括在线帮助。警告:如果是在一个现实中的系统上学习本文的知识,要多多运用常识。当试着对启动脚本程序进行修改的时候,要记住所做的修改可能会造成你的系统不能正常工作,而且无法采用重启动的方法恢复。不要在正常运转的系统上实验新的设置,对你准备修改的文件要全部进行备份。最重要的是,在手边要准备一张引导盘以防不测。
具体操作步骤
<code>/etc/rc.local</code>这是一个最简单的方法,编辑“/etc/rc.local”,把启动程序的shell命令输入进去即可(要输入命令的全路径),类似于windows下的“启动”。
使用命令 <code>vim /etc/rc.local </code>,然后在文件最后一行添加要执行程序的全路径。例如,每次开机时要执行一个haha.sh,这个脚本放在/opt下面,那就可以在<code>/etc/rc.local</code>中加一行<code>/opt/./haha.sh</code>,或者两行<code>cd /opt</code>和<code>./haha.sh</code>。crontab(类似于windows的任务计划服务)
通过crontab可以设定程序的执行时间表,例如让程序在每天的8点,或者每个星期一的10点执行一次。
- crontab -l 列出时间表
- crontab -e编辑时间表
- crontab -d删除时间表
- “-l”没什么可说的,就是一个查看而已
- “-e”是编辑,和vi没什么差别(其实就是用vi编辑一个特定文件)
- “-d”基本不用,因为它把该用户所有的时间表都删除了,一般都是用“-e”编辑把不要了的时间表逐行删除;
那到底该如何编辑呢?
crontab文件的格式是:<code>M H D m d CMD</code>。一个6个字段,其中最后一个CMD就是所要执行的程序,如haha.sh。
- M:分钟(0-59)
- H:小时(0-23)
- D:日期(1-31)
- m:月份(1-12)
- d:一个星期中的某天(0-6,0代表周日)
这5个时间字段用空格隔开,其值可以是一个数字,也可以用逗号隔开的多个数字(或其他) ,如果不需设置,则默认为** * *。 例如,每天的8点5分执行haha.sh,就是<code>5 8 * * * /opt/./haha.sh</code>。 好像和“开机程序自动启动”扯远了,现在回归正题。其实上面介绍的crontab的功能已经具备了开机自动启动的能力,可以写一个监测脚本,每5分钟执行一次(<code>/5 * * * * ./haha.sh</code>),如果程序不在了就重新启动一次。
- 注册系统服务
操作系统自带的服务,如ssh,ftp等等,开机都是自动启动的,我们也可以通过这种方式让自己开发的程序提高“身价”。比如我想把某个已经安装了的服务添加为系统服务,可以执行以下命令:
- <code>chkconfig --add 服务名称 </code> (首先,添加为系统服务,注意add前面有两个横杠)
- <code>chkconfig -leve 启动级别 服务名 on</code>(说明,3级别代表在命令行模式启动,5级别代表在图形界面启动,on表示开启)
- <code>chkconfig -leve 启动级别 服务名 off</code>说明,off表示关闭自启动)
- <code>chkconfig -level 3 mysql on</code>说明:让mysql服务在命令行模式,随系统启动
- <code>chkconfig --add 服务名称</code> 来删除系统服务
如果要查看哪些服务被添加为系统服务可以使用命令 :
<code>ntsysv </code>或者<code>chkconfig --list</code>如果要查看哪些程序被添加为自启动,可以使用命令 :
<code>cat /etc/rc.local </code> (查看这个文件中添加了哪些程序路径)
** 下面举例说说,如何把一个shell脚本添加为系统服务,并跟随系统启动:**
可以看到<code>/etc/rc.d/init.d</code>下有很多的文件,每个文件都是可以看到内容的,其实都是一些shell脚本。系统服务的启动就是通过<code>/etc/rc.d/init.d</code>中的脚本文件实现的。我们也可以写一个自己的脚本放在这里。
脚本文件的内容也很简单,类似于这个样子(例如起个名字叫做“hahad”):
<pre>
. /etc/init.d/functions
start() {
echo "Starting my process "
cd /opt
./haha.sh
}
stop() {
killall haha.sh
echo "Stoped"
}
</pre>
写了脚本文件之后事情还没有完,继续完成以下几个步骤:
- <code>chmod +x hahad</code> ** # **增加执行权限
- <code>chkconfig --add hahad</code> ** # ** 把hahad添加到系统服务列表
- <code>chkconfig hahad on</code> ** # **设定hahad的开关(on/off)
- <code>chkconfig --list hahad</code> ** # **就可以看到已经注册了hahad的服务
这时候才完成了全部工作。
本文转自:Linux如何实现开机启动程序详解