gpt4 book ai didi

linux - 优化基于MAC地址匹配提取子串的bash脚本

转载 作者:塔克拉玛干 更新时间:2023-11-03 00:10:09 24 4
gpt4 key购买 nike

客户端要求50个CentOS 7主机名根据网卡MAC地址命名:

目前我结合了下面的 bash 脚本,但似乎可以做很多优化,该脚本只会在图像首次启动时启动,它必须稳定和紧凑:

for i in `ip a |awk '/ether/ {print toupper($2) "_"}' | sed -r 's/:/-/g'`; do grep -i $i* MAC_HOST | awk -F _ '{print $2}'; done

MAC_HOST 内容:

22-8A-07-4F-16-A8_BT001
22-8A-07-5C-0F-58_BT002
22-8A-07-5C-0F-5B_BT003
22-8A-07-5D-D1-3A_BT004
22-8A-07-5D-D1-3D_BT005
22-8A-07-5D-D1-4C_BT006
22-8A-07-5D-D1-52_BT007
22-8A-07-5D-D1-58_BT008
22-8A-07-5D-D1-5B_BT009
22-8A-07-5D-D1-5E_BT010
22-8A-07-5D-D1-61_BT011
22-8A-07-5D-D1-67_BT012
22-8A-07-5D-D1-7C_BT013
22-8A-07-5D-D1-82_BT014
22-8A-07-5D-D1-8E_BT015
22-8A-07-5D-D1-91_BT016
22-8A-07-5E-2F-4D_BT017
22-8A-07-5E-2F-5C_BT018
22-8A-07-5E-2F-65_BT019
22-8A-07-5E-2F-7D_BT020
22-8A-07-5F-36-65_BT021
22-8A-07-5F-36-9B_BT022
22-8A-07-5F-36-A1_BT023
22-8A-07-5F-36-A4_BT024
22-8A-07-5F-36-A7_BT025
22-8A-07-5F-36-AA_BT026
22-8A-07-5F-36-AD_BT027
22-8A-07-5F-36-B0_BT028
22-8A-07-5F-36-B3_BT029
22-8A-07-5F-36-B6_BT030
22-8A-07-5F-36-BC_BT031
22-8A-07-5F-D8-0B_BT032
22-8A-07-5F-D8-0E_BT033
22-8A-07-5F-D8-11_BT034
22-8A-07-5F-D8-14_BT035
22-8A-07-5F-D8-17_BT036
22-8A-07-5F-D8-1A_BT037
22-8A-07-5F-D8-1D_BT038
22-8A-07-5F-D8-20_BT039
22-8A-07-5F-D8-26_BT040
22-8A-07-5F-D8-29_BT041
22-8A-07-5F-D8-2F_BT042
22-8A-07-5F-D8-32_BT043
22-8A-07-5F-D8-35_BT044
22-8A-07-5F-D8-38_BT045
22-8A-07-5F-D8-3B_BT046
22-8A-07-5F-D8-3E_BT047
22-8A-07-5F-D8-41_BT048
22-8A-07-5F-D8-44_BT049
22-8A-07-5F-D8-47_BT050

注意:在“before ;done”下方添加找到的主机名。我删除了它,这样就不会有人错误地覆盖他/她的主机名。

| cat >> /etc/hostname

最佳答案

我认为这可以满足您的需求,它消除了所有 shell 循环和大量进程:

ip a | awk 'FNR==NR{a[$1]=$2;next} /ether/{gsub(/:/, "-", $2); print a[toupper($2)]}' FS=_ MAC_HOST FS=" "  -

工作原理

  • FNR==NR{a[$1]=$2;next}

    这会读取第一个文件 MAC_HOST 的内容,并将其存储在关联数组 a 中。

    请注意,无论 ip a 返回多少个接口(interface),文件 MAC_HOST 都只被读取一次。此外,关联数组具有快速查找功能。

  • /ether/{gsub(/:/, "-", $2);打印[toupper($2)]}

    这将选择包含 ether 的行并处理第二个字段

  • FS=_ MAC_HOST FS=""-

    这会在读取 MAC_HOST 时将字段分隔符设置为 _,然后在读取 stdin(来自 ip a)时将其设置为空格。

    请注意,在 awk 中,FS=""a special case : 这意味着任何一个或多个空白字符的序列都被视为字段分隔符。

多行版本

相同的代码,当分布在多行时,是:

ip a | awk '

FNR==NR{
a[$1]=$2
next
}

/ether/{
gsub(/:/, "-", $2)
print a[toupper($2)]
}
' FS=_ MAC_HOST FS=" " -

变化:忽略不匹配的 MAC 地址

ip a 返回的 MAC 地址不在 MAC_HOST 中时,我们该怎么办?上面的版本打印一个空行。当缺少 MAC_HOST 条目时,以下版本不打印任何内容:

ip a | awk 'FNR==NR{a[$1]=$2;next} {b=""} /ether/{gsub(/:/, "-", $2); b=a[toupper($2)]} b {print b}' FS=_ MAC_HOST FS=" "  -

例如,这里的假输入提供了两个 MAC 地址,一个在 MAC_HOST 中,一个不在:

$ printf 'ether 22:8a:07:5f:36:a4\n\nether 22:8a:07:5f:36:a5\n\n' | awk 'FNR==NR{a[$1]=$2;next} {b=""} /ether/{gsub(/:/, "-", $2); b=a[toupper($2)]} b {print b}' FS=_ MAC_HOST FS=" "  -
BT024

要使用上面的覆盖/etc/hostname:

ip a | awk 'FNR==NR{a[$1]=$2;next} {b=""} /ether/{gsub(/:/, "-", $2); b=a[toupper($2)]} b {print b}' FS=_ MAC_HOST FS=" "  - >/etc/hostname

关于linux - 优化基于MAC地址匹配提取子串的bash脚本,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38583202/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com