深刻理解Linux修改hostname

當我以爲對Linux系統下修改hostname已經很是熟悉的時候,今天碰到了幾個個問題,這幾個問題給我好好上了一課,不少知識點,當你以爲你已經掌握的時候,其實你瞭解的還只是皮毛。技術活,切勿淺嘗則止!html

 

實驗環境:Red Hat Enterprise Linux Server release 5.7 (Tikanga) ,其它版本Linux可能有所不一樣。請以實際環境爲準。linux

其實我屢次修改過hostname,通常只須要修改 /etc/hosts 和 /etc/sysconfig/network 兩個文件下相關配置便可。可是,今天我遇到了兩個問題:服務器

 

問題1: 爲何/etc/sysconfig/network配置文件中HOSTNAME爲localhost.localdomain,可是顯示的hostname爲po132345806-a,那到底hostname的配置值放在哪裏?網絡

[root@po132345806-a ~]# hostname
po132345806-a.gfg1.esquel.com
[root@po132345806-a ~]# more /etc/hosts
# Do not remove the following line, or various programs 
# that require network functionality will fail. 
127.0.0.1 localhost.localdomain localhost 
::1 localhost6.localdomain6 localhost6 
[root@po132345806-a ~]# more /etc/sysconfig/network 
NETWORKING=yes 
NETWORKING_IPV6=yes 
HOSTNAME=localhost.localdomain

 

有圖有真相,省得你們不相信這個現象,當我第一次碰到這種特殊狀況時,我也很是納悶。Google了一些資料加上本身的實踐才弄明白app

clipboard

    問題2: 修改了hostname後,如何使其當即生效而不用重啓操做系統。dom

    問題3: 修改hostname有幾種方式?分佈式

    問題4: hostname跟/etc/hosts 下配置有關係嗎?oop

    問題5: 如何查看hostname的值,以那個爲準?ui

 

問題1解答:我一直覺得hostname的值配置在/etc/sysconfig/network中,這個文件裏面HOSTNAME配置爲啥,hostname值就是啥。可是爲何出現上面那種狀況呢?難道/etc/sysconfig/network不是hostname的配置文件,難道還另有其它配置文件?因而我當時實驗了一下修改了/etc/sysconfig/network文件中HOSTNAME爲DB-Server,發現hostname的值依然沒有變化,因而重啓了計算機this

[root@po132345806-a ~]# hostname
po132345806-a.gfg1.esquel.com
[root@po132345806-a ~]# more /proc/sys/kernel/hostname
po132345806-a.gfg1.esquel.com
[root@po132345806-a ~]# sysctl kernel.hostname
kernel.hostname = po132345806-a.gfg1.esquel.com
[root@po132345806-a ~]#
[root@po132345806-a ~]# reboot

clipboard[1]

 

    重啓事後發現竟然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 沒有什麼用。只是當時實驗時覺得必須重啓網絡服務。

 

clipboard[2]

 

在SecureCRT新建克隆一個會話發現hostanme已經從DB-Server變爲Test了,可是/etc/sysconfig/network的值仍是DB-Server.localdomain,並無變爲Test。

   1: [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:  

clipboard[3]

 

可是若是重啓系統後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,代碼以下所示,至此,咱們能夠完全明白了。

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。

        # 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文件,修改後的值以下所示:

[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

 

重啓系統後,咱們再截圖看看狀況:

 

clipboard[4]

    因此這也是有時候人們覺得hostname的值跟hosts文件有關係的緣故。

 

問題5: 如何查看hostname的值,以那個爲準?

[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個問題,那麼理解這個問題就很簡單了。

 

 

參考資料:

http://jblevins.org/log/hostname

http://www.ducea.com/2006/08/07/how-to-change-the-hostname-of-a-linux-system/

https://www.kernel.org/doc/Documentation/sysctl/kernel.txt

http://soft.chinabyte.com/os/281/11563281.shtml

相關文章
相關標籤/搜索