AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / unix / 问题 / 782337
Accepted
Gary U.U. Unixuser
Gary U.U. Unixuser
Asked: 2024-08-22 13:27:46 +0800 CST2024-08-22 13:27:46 +0800 CST 2024-08-22 13:27:46 +0800 CST

如何使用 awk 和 if 获取子字符串或文本值?

  • 772

我不知道如何编写简单的 ping 日志。我只需要 ms 值或“无连接”字样。

我有。我只需要“ ”和“ ”myping=$(ping -c 1 10.0.10.1)之间的值,例如“ ”中的3.151。time= mstime=3.151 ms

我可以使用 awk,但是我会得到很多空行,而且我不知道如何摆脱它们:

$ awk -F 'time=| ms' '{print $2}' <<< "$myping"

3.151

然后,当 ping 结果不包含 ms 值时,我该如何插入“无连接”?

多谢!

bash
  • 4 4 个回答
  • 73 Views

4 个回答

  • Voted
  1. ilkkachu
    2024-08-22T15:08:39+08:002024-08-22T15:08:39+08:00

    假设您使用的是 Linux,并且您的 ping 输出如下所示:

    $ cat > ping-output.txt
    PING 1.1.1.1 (1.1.1.1) 56(84) bytes of data.
    64 bytes from 1.1.1.1: icmp_seq=1 ttl=59 time=1.30 ms
    64 bytes from 1.1.1.1: icmp_seq=2 ttl=59 time=0.850 ms
    64 bytes from 1.1.1.1: icmp_seq=3 ttl=59 time=1.25 ms
    
    --- 1.1.1.1 ping statistics ---
    3 packets transmitted, 3 received, 0% packet loss, time 2027ms
    rtt min/avg/max/mdev = 0.850/1.134/1.304/0.202 ms
    

    您可以用=符号和空格分隔字段,然后选择正确的字段。最后检查是否获得了值。

    在单个 ping 线路上,这将是第 10 个字段。

    % cat ping-output.txt | 
      awk -F"[= ]" '/time=/ { rtt=$10 } 
                    END { if (rtt) print rtt; else print "no connection" }'
    1.25
    

    或者,您可以读取最后一行的摘要数字,例如选择最小 rtt。

    使用/或空格作为字段分隔符,最小 rtt 是最后一行的第 7 个字段。

    $ cat ping-output.txt |
      awk -F"[/ ]" '/^rtt/ { rtt=$7 }
                    END { if (rtt) print rtt; else print "no connection" }'
    0.850
    

    (ping我的 Mac 上的输出略有不同,主要区别在于最后一行是round-trip。rtt如果您的不同,则必须相应地调整模式和字段编号。)


    但是,与其在输出中使用特殊字符串来标记远程未应答,不如使用命令的退出状态。(这很ping可能也是它本身的作用。)

    因此,exit 1在脚本中的适当位置添加一个,然后您可以执行以下操作:

    remote=1.1.1.1
    if rtt=$(ping -c1 "$remote" | awk -F"[/ ]" '/^rtt/ { rtt=$7 }
                    END { if (rtt) print rtt; else exit 1 }'; then
        echo "'$remote' is alive, rtt=$rtt"
    else
        echo "'$remote' did not answer"
    fi
    
    • 0
  2. kos
    2024-08-22T15:24:56+08:002024-08-22T15:24:56+08:00

    如果使用pingfrom iputils/Linux,其ping -c1 localhost输出如下内容:

    PING localhost (127.0.0.1) 56(84) bytes of data.
    64 bytes from localhost (127.0.0.1): icmp_seq=1 ttl=64 time=0.026 ms
    
    --- localhost ping statistics ---
    1 packets transmitted, 1 received, 0% packet loss, time 0ms
    rtt min/avg/max/mdev = 0.026/0.026/0.026/0.000 ms
    

    ms只有当目的地可以到达时,第二行才会在末尾包含。

    因此,可能不是最优雅的解决方案,但使用sed:

    sed -En '2{s/.*=([0-9.]+) ms$/\1/p; t; s/.*/no connection/p}'
    
    • -E:开启 ERE 支持
    • -n:并且不自动打印模式空间;
    • 2{[...]}:如果我们正在处理第 2 行
    • s/.*=([0-9.]+) ms$/\1/p并且该行与该模式匹配,用第一个捕获的组替换整行并打印该行;
    • t:如果进行了替换,则开始新循环,
    • s/.*/no connection/p}:否则用 替换整行no connection。
    % ping -c1 localhost | sed -En '2{s/.*=([0-9.]+) ms$/\1/p; t; s/.*/no connection/p}'  
    0.022
    % ping -c1 192.168.1.127 | sed -En '2{s/.*=([0-9.]+) ms$/\1/p; t; s/.*/no connection/p}'
    no connection
    
    • 0
  3. Olivier Dulac
    2024-08-22T19:51:05+08:002024-08-22T19:51:05+08:00

    @terdon 的(正确)答案有一点变体:允许您进行超过 1 次 ping 来取平均值并得出更准确的结果:

    $ PS1="$ "; PS2=""
    $ cat <<EOF | awk -F 'time=| ms' '($2){k++; sum+= $2}END{ if(!k){ 
    print "no connection"} else { print ( sum * 1.0 / k ) }}'
    ...
    ... time=30.5 ms
    ...
    ... time=40 ms
    ...
    ... time=20 ms
    ...
    EOF
    30.1667
    # and of course: replace the : 
    #  cat <<EOF
    # with:
    #  ping -c 3 the_ip
    

    或者更简单:使用最后一行统计数据(min/avg/max/mdev):

    ping -c 3 ip | awk -F'/| = ' '/^rtt/{avg=$3} END{print (!avg) ? "KO" : avg }'
    
    • 0
  4. Best Answer
    terdon
    2024-08-22T16:34:40+08:002024-08-22T16:34:40+08:00

    在我的 Arch 系统上,ping可以访问的主机和无法访问的主机的结果如下所示:

    $ ping -c1 8.8.8.8
    PING 8.8.8.8 (8.8.8.8) 56(84) bytes of data.
    64 bytes from 8.8.8.8: icmp_seq=1 ttl=118 time=39.8 ms
    
    --- 8.8.8.8 ping statistics ---
    1 packets transmitted, 1 received, 0% packet loss, time 0ms
    rtt min/avg/max/mdev = 39.846/39.846/39.846/0.000 ms
    
    

    和

    $ ping -c 1 10.0.10.1
    PING 10.0.10.1 (10.0.10.1) 56(84) bytes of data.
    
    --- 10.0.10.1 ping statistics ---
    1 packets transmitted, 0 received, 100% packet loss, time 0ms
    

    假设您的输出相同,并且假设失败的连接的输出保存在$badPing成功的输出中$goodPing,您可以执行以下操作:

    $ awk -F= '/time=/{k=sub(/ms/,""); l=$NF}END{print k ? l : "no connection" }' <<< "$badPing"
    no connection
    $ awk -F= '/time=/{k=sub(/ms/,""); l=$NF}END{print k ? l : "no connection" }' <<< "$goodPing"
    39.8 
    

    上述命令使用=作为字段分隔符。然后,在包含字符串 的每一行上time=,它会替换该行上第一次出现的字符串ms(请注意,这假设相关行上只出现一次ms;如果不是这种情况,请使用sub(/ms/,"",$NF)),这只留下数值作为最后一个字段,并将k和中进行的替换次数保存为l。

    然后,在我们处理完整个输入之后,如果k设置了,那么如果发生了替换,我们就打印出的值l,如果没有,我们就打印出no connection。

    或者,您可以在开始时设置值l,然后打印它:

    $ awk -F= -v l="no connection" '/time=/ && sub(/ms/,""){l=$NF}END{print l}' <<< "$badPing"
    no connection
    $ awk -F= -v l="no connection" '/time=/ && sub(/ms/,""){l=$NF}END{print l}' <<< "$goodPing"
    39.8 
    

    要使原始方法有效,只需添加第二个字段作为条件并根据该条件设置变量。然后,当第二个字段存在时,您可以打印它,no connection如果变量未设置,则在最后打印。像这样:

    $ awk -F 'time=| ms' '($2){print $2;k=1}END{ if(!k) print "no connection"}' <<< "$goodPing"
    39.8
    
    $ awk -F 'time=| ms' '($2){print $2;k=1}END{ if(!k) print "no connection"}' <<< "$badPing"
    no connection
    
    • -1

相关问题

  • 通过命令的标准输出以编程方式导出环境变量[重复]

  • 从文本文件传递变量的奇怪问题

  • 虽然行读取保持转义空间?

  • `tee` 和 `bash` 进程替换顺序

  • 运行一个非常慢的脚本直到它成功

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    模块 i915 可能缺少固件 /lib/firmware/i915/*

    • 3 个回答
  • Marko Smith

    无法获取 jessie backports 存储库

    • 4 个回答
  • Marko Smith

    如何将 GPG 私钥和公钥导出到文件

    • 4 个回答
  • Marko Smith

    我们如何运行存储在变量中的命令?

    • 5 个回答
  • Marko Smith

    如何配置 systemd-resolved 和 systemd-networkd 以使用本地 DNS 服务器来解析本地域和远程 DNS 服务器来解析远程域?

    • 3 个回答
  • Marko Smith

    dist-upgrade 后 Kali Linux 中的 apt-get update 错误 [重复]

    • 2 个回答
  • Marko Smith

    如何从 systemctl 服务日志中查看最新的 x 行

    • 5 个回答
  • Marko Smith

    Nano - 跳转到文件末尾

    • 8 个回答
  • Marko Smith

    grub 错误:你需要先加载内核

    • 4 个回答
  • Marko Smith

    如何下载软件包而不是使用 apt-get 命令安装它?

    • 7 个回答
  • Martin Hope
    user12345 无法获取 jessie backports 存储库 2019-03-27 04:39:28 +0800 CST
  • Martin Hope
    Carl 为什么大多数 systemd 示例都包含 WantedBy=multi-user.target? 2019-03-15 11:49:25 +0800 CST
  • Martin Hope
    rocky 如何将 GPG 私钥和公钥导出到文件 2018-11-16 05:36:15 +0800 CST
  • Martin Hope
    Evan Carroll systemctl 状态显示:“状态:降级” 2018-06-03 18:48:17 +0800 CST
  • Martin Hope
    Tim 我们如何运行存储在变量中的命令? 2018-05-21 04:46:29 +0800 CST
  • Martin Hope
    Ankur S 为什么 /dev/null 是一个文件?为什么它的功能不作为一个简单的程序来实现? 2018-04-17 07:28:04 +0800 CST
  • Martin Hope
    user3191334 如何从 systemctl 服务日志中查看最新的 x 行 2018-02-07 00:14:16 +0800 CST
  • Martin Hope
    Marko Pacak Nano - 跳转到文件末尾 2018-02-01 01:53:03 +0800 CST
  • Martin Hope
    Kidburla 为什么真假这么大? 2018-01-26 12:14:47 +0800 CST
  • Martin Hope
    Christos Baziotis 在一个巨大的(70GB)、一行、文本文件中替换字符串 2017-12-30 06:58:33 +0800 CST

热门标签

linux bash debian shell-script text-processing ubuntu centos shell awk ssh

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve