以下操作均时基于 Centos 6.8 操作。
一、现象
在平时工作中,当需要修改主机名时,我们一般会这样操作:
第一步,通过 hostname 命令临时修改主机名。
hostname kwang-test01
第二步,修改配置文件,保证机器重启时主机名不会变。
$ cat /etc/sysconfig/network
NETWORKING=yes
HOSTNAME=kwang-test01
NOZEROCONF=yes
以上操作确实是修改主机名的正确姿势,也能达到预期的效果,但为什么要这样操作呢,知其然也要知其所以然,下面我们来看看原因。
二、透过现象看本质
2.1 hostname 临时修改主机名
当我们执行 hostname 命令时,会临时修改 Linux Kernel 中一个同为 hostname 的内核参数,而 Linux Kernel 中 hostname 参数保存在 /proc/sys/kernel/hostname 中。
2.2 修改 /etc/sysconfig/network 配置永久修改主机名
有人可能会困惑,为什么永久修改主机名需要修改 /etc/sysconfig/network 的 HOSTNAME 参数?回答这个问题前,先让我们看看 Linux 启动时脚本的一段代码:
HOSTNAME=$(/bin/hostname)
set-mif [ -f /etc/sysconfig/network ]; then./etc/sysconfig/networkfi
if [ -z "$HOSTNAME" -o "$HOSTNAME" = "(none)" ]; thenHOSTNAME=localhostfi
可以看出 Linux 的启动逻辑:首先会读取 /etc/sysconfig/network 中的 HOSTNAME 参数,然后将系统主机名配置成获取的 HOSTNAME 参数。
注意,我们也说了,只有在 Linux 在启动时才会加载 /etc/sysconfig/network 配置,但平时执行 hostname 命令系统是如何知道主机名临时修改了呢?我们继续往下看。
2.3 更进一步
通过 hostname 命令获取的值跟 /etc/sysconfig/network 文件中的 HOSTNAME 有一定的关联,但是没有必然联系,只有在 Linux 启动时才会与配置文件的 HOSTNAME 值有保持一致,启动相互不影响。进一步了解,我们发现通过 hostname 命令获取的值并不是直接从 /etc/sysconfig/network 获取,而是从 Linux Kernel 的内核参数 /proc/sys/kernel/hostname 获取,这一点我们可以从下面实操看出:
# hostname //当前主机名
kwang_test01
#cat /proc/sys/kernel/hostname //修改内核参数
kwang_test01
#echo "kwang_test01_change" > /proc/sys/kernel/hostname //修改内核参数
# cat /proc/sys/kernel/hostnamekwang_test01_change
#hostname //修改后主机名,发现主机名已修改
kwang_test01_change
#cat /etc/sysconfig/network //并发现这个配置的 HOSTNAME 值没有变
NETWORKING=yes
HOSTNAME=kwang-test01
NOZEROCONF=yes
结论:
hostname 命令获得的值是从 /proc/sys/kernel/hostname 获取的,与 /etc/sysconfig/network 配置中的 HOSTNAME 没有直接关联;
/proc/sys/kernel/hostname 内核参数的初始值在 Linux 启动时从 /etc/sysconfig/network 配置中加载,启动后该值通过 root 账号可以修改。
三、疑惑
最近在遇到一个奇怪的现象,/proc/sys/kernel/hostname 中的值被定时修改了,没有人为操作,系统也没有重启,暂时没有解决,后续解决了更新。