當我覺得對Linux系統下修改hostname已經非常熟悉的時候,今天碰到了幾個個問題,這幾個問題給我好好上了一課,很多知識點,當你覺得你已經掌握的時候,其實你了解的還只是皮毛。技術活,切勿淺嘗則止!
實驗環境:Red Hat Enterprise Linux Server release 5.7 (Tikanga) ,其它版本Linux可能有所不同。請以實際環境為準。
其實我多次修改過hostname,一般只需要修改 /etc/hosts 和 /etc/sysconfig/network 兩個文件下相關配置即可。但是,今天我遇到了兩個問題:
問題1: 為什么/etc/sysconfig/network配置文件中HOSTNAME為localhost.localdomain,但是顯示的hostname為po132345806-a,那到底hostname的配置值放在哪里?
1
1: [root@po132345806-a ~]# more /etc/hosts 2: # Do not remove the following line, or various programs 3: # that require network functionality will fail. 4: 127.0.0.1 localhost.localdomain localhost 5: ::1 localhost6.localdomain6 localhost6 6: [root@po132345806-a ~]# more /etc/sysconfig/network 7: NETWORKING=yes 8: NETWORKING_IPV6=yes 9: HOSTNAME=localhost.localdomain
有圖有真相,免得大家不相信這個現象,當我第一次碰到這種特殊情況時,我也非常納悶。Google了一些資料加上自己的實踐才弄明白
問題2: 修改了hostname后,如何使其立即生效而不用重啟操作系統。
問題3: 修改hostname有幾種方式?
問題4: hostname跟/etc/hosts 下配置有關系嗎?
問題5: 如何查看hostname的值,以那個為準?
?
問題1解答:我一直以為hostname的值配置在/etc/sysconfig/network中,這個文件里面HOSTNAME配置為啥,hostname值就是啥。但是為什么出現上面那種情況呢?難道/etc/sysconfig/network
不是hostname的配置文件,難道還另有其它配置文件?于是我當時實驗了一下修改了/etc/sysconfig/network文件中HOSTNAME為DB-Server,發現
hostname的值依然沒有變化,于是重啟了計算機
1: "/etc/sysconfig/network" 3L, 66C written
?2: 132345806-a ~]# hostname
?3: 806-a.gfg1.esquel.com
?4: 132345806-a ~]# more /proc/sys/kernel/hostname
?5: 806-a.gfg1.esquel.com
?6: 132345806-a ~]# sysctl kernel.hostname
?7: ostname = po132345806-a.gfg1.esquel.com
?8: 132345806-a ~]#
?9: 132345806-a ~]# reboot
?
重啟過后發現居然hostname變為DB-Server了,也就是說修改配置文件/etc/sysconfig/network下的HOSTNAME生效了。那么也就是說應該是有人修改過 kernel.hostname,請看下面實驗
1: [root@DB-Server ~]# more /etc/sysconfig/network
?2:
?3: NETWORKING=yes
?4:
?5: NETWORKING_IPV6=yes
?6:
?7: HOSTNAME=DB-Server.localdomain
?8:
?9: [root@DB-Server ~]# echo Test > /proc/sys/kernel/hostname
?10:
?11: [root@DB-Server ~]# more /etc/proc/sys/kernel/hostname
?12:
?13: /etc/proc/sys/kernel/hostname: No such file or directory
?14:
?15: [root@DB-Server ~]# more /proc/sys/kernel/hostname
?16:
?17: Test
?18:
?19: [root@DB-Server ~]# /etc/init.d/network restart
?20:
?21: Shutting down interface eth0: [ OK ]
?22:
?23: Shutting down loopback interface: [ OK ]
?24:
?25: Bringing up loopback interface: [ OK ]
?26:
?27: Bringing up interface eth0:
?28:
?29: Determining IP information for eth0... done.
?30:
?31: [ OK ]
?32:
?33: [root@DB-Server ~]# hostname
?34:
?35: Test
?36:
?37: [root@DB-Server ~]#
?38:
注意:其實 /etc/init.d/network restart 沒有什么用。只是當時實驗時以為必須重啟網絡服務。
在SecureCRT新建克隆一個會話發現hostanme已經從DB-Server變為Test了,但是/etc/sysconfig/network的值還是DB-Server.localdomain,并沒有變為Test。
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
[root@ Test ~]# more /etc/sysconfig/network
???2:?
???3: NETWORKING=yes
???4:?
???5: NETWORKING_IPV6=yes
???6:?
???7: HOSTNAME=DB-Server.localdomain
???8:?
???9: [root@ Test ~]# hostname
??10:?
??11: Test
??12:?
??13: [root@ Test ~]# more /etc/hosts
??14:?
??15: # Do not remove the following line, or various programs
??16:?
??17: # that require network functionality will fail.
??18:?
??19: 127.0.0.1 localhost.localdomain localhost
??20:?
??21: ::1 localhost6.localdomain6 localhost6
??22:?
??23: [root@ Test ~]# more /proc/sys/kernel/hostname
??24:?
??25: Test
??26:?
??27: [root@ Test ~]#
??28:
但是如果重啟系統后hostname會變為DB-Server,Google了一些英文文檔資料才知道,hostname是Linux系統下的一個內核參數,它保存在/proc/sys/kernel/hostname下,但是它的值是Linux啟動時從rc.sysinit讀取的。
hostname is a kernel parameter which stores hostname of the system. Its location is”/proc/sys/kernel/hostname”
The value for this parameter is loaded to kernel by rc.sysinit file during the boot process.
而/etc/rc.d/rc.sysinit中HOSTNAME的取值來自與/etc/sysconfig/network下的HOSTNAME,代碼如下所示,至此,我們可以徹底明白了。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
HOSTNAME=`/bin/hostname`
?
HOSTTYPE=`uname -m`
?
unamer=`uname -r`
?
set -m
?
if [ -f /etc/sysconfig/network ]; then
?
. /etc/sysconfig/network
?
fi
?
if [ -z "$HOSTNAME" -o "$HOSTNAME" = "(none)" ]; then
?
????HOSTNAME=localhost
?
fi
結論:/etc/sysconfig/network 確實是hostname的配置文件,hostname的值跟該配置文件中的HOSTNAME有一定的關聯關系,但是沒有必然關系,hostname的值來自內核參數/proc/sys/kernel/hostname,如果我通過命令sysctl kernel.hostname=Test修改了內核參數,那么hostname就變為了Test了。
問題2: 修改了hostname后,如何使其立即生效而不用重啟操作系統。
方法1:修改了/etc/sysconfig/network下的HOSTNAME后,然后使用echo servername > /proc/sys/kernel/hostname。
[root@DB-Server ~]# echo Test >/proc/sys/kernel/hostname
注意當前會話還是不會變化,但是后續新建會話則會生效。
方法2:修改了/etc/sysconfig/network下的HOSTNAME后,然后使用sysctl kernel.hostname命令使其立即生效
[root@DB-Server ~]# sysctl kernel.hostname=Test2
kernel.hostname = Test2
注意當前會話還是不會變化,但是后續新建會話會生效。
方法3:修改了/etc/sysconfig/network下的HOSTNAME后,然后使用hostname命令使其生效
[root@ Test ~]# hostname DB-Server
注意當前會話還是不會變化,但是后續新建會話會生效。
其實呢,這幾種方式只是結合永久性修改和臨時性修改hostname,使其不必重啟Linux服務器,哈哈,不知道你明白沒。
?
問題3: 修改hostname有幾種方式?
1: hostname DB-Server –運行后立即生效(新會話生效),但是在系統重啟后會丟失所做的修改
2: echo DB-Server > /proc/sys/kernel/hostname –運行后立即生效(新會話生效),但是在系統重啟后會丟失所做的修改
3: sysctl kernel.hostname=DB-Server –運行后立即生效(新會話生效),但是在系統重啟后會丟失所做的修改
4: 修改/etc/sysconfig/network下的HOSTNAME變量 –需要重啟生效,永久性修改。
問題4: hostname跟/etc/hosts 下配置有關系嗎?
如果從我上面的實驗來看,其實hostname跟/etc/hosts下的配置是沒有關系的。hostname的修改、變更完全不依賴hosts文件。 其實hosts文件的作用相當如DNS,提供IP地址到hostname的對應。早期的互聯網計算機數量少,單機hosts文件里足夠存放所有聯網計算機。不過隨著互聯網的發展,這就遠遠不夠了。于是就出現了分布式的DNS系統。由DNS服務器來提供類似的IP地址到域名的對應。具體可以man hosts查看相關信息。
Linux系統在向DNS服務器發出域名解析請求之前會查詢/etc/hosts文件,如果里面有相應的記錄,就會使用hosts里面的記錄。/etc/hosts文件通常里面包含這一條記錄
127.0.0.1 localhost.localdomain localhost
hosts文件格式是一行一條記錄,分別是IP地址 、hostname、 aliases,三者用空白字符分隔,aliases可選。
127.0.0.1到localhost這一條建議不要修改,因為很多應用程序會用到這個,比如sendmail,修改之后這些程序可能就無法正常運行。
但是呢,其實hostname也不是說跟/etc/hosts一點關系都沒有。在/etc/rc.d/rc.sysinit中,有如下邏輯判斷,當hostname為localhost后localhost.localdomain時,將會使用接口IP地址對應的hostname來重新設置系統的hostname。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
# In theory there should be no more than one network interface active
?
?????# this early in the boot process -- the one we're booting from.
?
?????# Use the network address to set the hostname of the client. This
?
?????# must be done even if we have local storage.
?
?????ipaddr=
?
?????if [ "$HOSTNAME" = "localhost" -o "$HOSTNAME" = "localhost.localdomain" ]; then
?
?????????????ipaddr=$(ip addr show to 0/0 scope global | awk '/[[:space:]]inet / { print gensub("/.*","","g",$2) }')
?
?????????????if [ -n "$ipaddr" ]; then
?
?????????????????????eval $(ipcalc -h $ipaddr 2>/dev/null)
?
?????????????????????hostname ${HOSTNAME}
?
?????????????fi
?
?????fi
我們來實驗一下吧,修改hosts、network文件,修改后的值如下所示:
1
2
3
4
5
6
7
8
9
10
[root@ localhost~]# more /etc/hosts
# Do not remove the following line, or various programs
# that require network functionality will fail.
::1 localhost.localdomain localhost
127.0.0.1 localhost.localdomain localhost
192.168.244.128 DB-Server.localdomain DB-Server
[root@ localhost? ~]# more /etc/sysconfig/network
NETWORKING=yes
NETWORKING_IPV6=yes
HOSTNAME=localhost.localdomain
重啟系統后,我們再截圖看看情況:
所以這也是有時候人們以為hostname的值跟hosts文件有關系的緣故。
問題5: 如何查看hostname的值,以那個為準?
1
2
3
4
5
6
7
8
[root@DB-Server ~]# hostname
DB-Server
[root@DB-Server ~]# more /proc/sys/kernel/hostname
DB-Server
[root@DB-Server ~]# more /etc/sysconfig/network
NETWORKING=yes
NETWORKING_IPV6=yes
HOSTNAME=localhost.localdomain
以那個為準呢,如果你理解了前面4個問題,那么理解這個問題就很簡單了。
評論
查看更多