1

这是我的情况的简短版本。我当前的 ISP 无法正常工作,因此,我想记录 ping 成功和失败以证明它无法正常工作。我会在 Linux 上,但 Windows 也可以。我一直在阅读,据我所知,shell 脚本或 Python 代码效果最好。

所以,更详细一点:我想每 5 秒在一个网站上执行一次 ping 操作。如果 ping 成功,它应该将 ping 的正常结果输出到文件中。但是 Ping 失败有很长的超时时间,所以我想制作程序,如果 ping 在 4 秒左右没有成功,它会停止 ping 尝试,但仍会输出带有“1 个数据包”的完整 ping 数据发送,0 接收”等,就像它已经用完时间一样。

这是我到目前为止所拥有的:

while true
do
    ping -c1 -w4 www.example.com >> log.txt
    sleep 5
done

不幸的是,有两个问题。首先,“-w4”并没有像我想象的那样在 4 秒后超时。其次,“睡眠”会增加已经花费的时间。因此,如果 ping 需要 3 秒,则循环将需要 8 秒。因为我将编写另一个程序来计算成功、失败和输出停机时间百分比,所以让循环在失败时花费大量额外时间是不可接受的。

那么,鉴于我的需要,最好的方法是什么?我可以改进什么来满足需求吗?感谢您的任何帮助!

PS:我对编写 shell 脚本几乎一无所知(尽管我能胜任终端),而我对 Python 的了解甚至更少(尽管我知道 C++)。提前道歉!

4

2 回答 2

1

如果你真的想让整个循环不超过 5 秒,你必须知道 ping 用了多长时间,然后从 5 中减去它,然后睡眠差值。

您可以通过 bash 记录 ping 和减法之前和之后的时间,或者使用time -p命令运行ping(第一个非空白行的第二列将挂钟时间作为浮点数,如0.01)来执行此操作。

但我认为这从 Python 会更容易:

import datetime
import subprocess
import time

while True:
    start = datetime.datetime.now()
    with open('log.txt', 'a+b') as logfile:
        subprocess.call(['ping', '-c1', '-w4', 'www.example.com'], stdout=logfile)
    delta = datetime.datetime.now() - start
    time.sleep(delta.total_seconds())

当然,如果ping最多 5 秒内没有完成,这将不起作用,因为你不能在负时间睡觉。您可以通过使用显式Popen并调用wait(5)它来解决这个问题,然后如果它失败则杀死......</p>

但是总是给它 5 秒要简单得多,不管它是否需要那么多:

while True:
    with open('log.txt', 'a+b') as logfile:
        ping = subprocess.Popen(['ping', '-c1', '-w4', 'www.example.com'],
                                stdout=logfile)
        time.sleep(5)
        # Not strictly necessary, but it avoids signaling for no reason
        if ping.poll() is None:
            ping.kill()
        ping.wait()
于 2013-09-17T21:44:56.257 回答
0

您可以将 ping 置于后台并使用另一个子进程等待并终止它:

while true
do
    ping -c1 www.example.com >> log.txt &
    p=$!
    ( sleep 4s; kill -s SIGABRT "$p"; ) &>/dev/null &
    wait "$p"
    sleep 5
done

如果需要,调整超时。

于 2013-09-17T21:44:07.260 回答