如何在C#中实现PsPing TCP ping

时间:2014-09-26 19:55:33

标签: c# tcp ping

我正在尝试在C#中实现Mark Russinovich的SysInternals PsPing tool来测量使用TCP ping的延迟。

我不确定它是如何进行ping调用的(显然不使用原始Window套接字,因为它不需要管理员权限来运行)。我知道hping通过TCP发送SYN数据包并测量响应时间。

准确测量服务器延迟的实现技术是什么,而TCP不会测量页面加载时间,而只测量程序包确认的网络延迟?这个库有吗?

C:\>psping stackoverflow.com:80

PsPing v2.01 - PsPing - ping, latency, bandwidth measurement utility
Copyright (C) 2012-2014 Mark Russinovich
Sysinternals - www.sysinternals.com

Pinging 198.252.206.16 with 32 bytes of data:
5 iterations (warmup 1) ping test:
Reply from 198.252.206.16: 81.57ms
Reply from 198.252.206.16: 80.81ms
Reply from 198.252.206.16: 80.68ms
Reply from 198.252.206.16: 80.52ms
Reply from 198.252.206.16: 80.71ms

Ping statistics for 198.252.206.16:
  Sent = 4, Received = 4, Lost = 0 (0% loss),
  Minimum = 80.52ms, Maximum = 80.81ms, Average = 80.68ms

更新:请不要回答“你为什么不使用Ping类,它已经做了”,因为我在询问ping TCP而不是ICMP。

3 个答案:

答案 0 :(得分:18)

我尝试了几种方法,首先想到我必须使用原始套接字或者至少使用本机调用,但是简单的TCP连接和关闭似乎创建与psping实用程序完全相同的结果:

var times = new List<double>();
for (int i = 0; i < 4; i++)
{
    var sock = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
    sock.Blocking = true;

    var stopwatch = new Stopwatch();

    // Measure the Connect call only
    stopwatch.Start();
    sock.Connect(endPoint);
    stopwatch.Stop();

    double t = stopwatch.Elapsed.TotalMilliseconds;
    Console.WriteLine("{0:0.00}ms", t);
    times.Add(t);

    sock.Close();

    Thread.Sleep(1000);
}
Console.WriteLine("{0:0.00} {1:0.00} {2:0.00}", times.Min(), times.Max(), times.Average());

使用Wireshark检查流量,我可以确认psping和上面的代码段正在创建完全相同的数据包序列。

-> [SYN]
<- [SYN,ACK]
-> [ACK]
-> [FIN,ACK]
<- [FIN,ACK]
-> [ACK]

psping输出,没有预热并使用TCP ping:

C:\>psping -w 0 stackoverflow.com:80

PsPing v2.01 - PsPing - ping, latency, bandwidth measurement utility
Copyright (C) 2012-2014 Mark Russinovich
Sysinternals - www.sysinternals.com

TCP connect to 198.252.206.16:80:
4 iterations (warmup 0) connecting test:
Connecting to 198.252.206.16:80: 92.30ms
Connecting to 198.252.206.16:80: 83.16ms
Connecting to 198.252.206.16:80: 83.29ms
Connecting to 198.252.206.16:80: 82.98ms

TCP connect statistics for 198.252.206.16:80:
  Sent = 4, Received = 4, Lost = 0 (0% loss),
  Minimum = 82.98ms, Maximum = 92.30ms, Average = 85.43ms

上述程序的输出:

C:\>TcpPing.exe stackoverflow.com 80
88.60ms
83.65ms
84.05ms
84.05ms
83.65 88.60 85.09

至于测量,我必须说,有时在不同的跑步中会有不同的结果,但总的来说它们看起来非常接近:证明测量Connect()的呼叫足够好了。我在思考,取几百个结果的中位数可能会更有信心。

答案 1 :(得分:2)

Winsock肯定会让你轻松做到这一点。

您是否看过像http://www.elifulkerson.com/projects/tcping.php这样的节目来源?

非常简单的程序(控制台),它完全符合您的要求(AFAIK),并提供了看起来非常清晰,简洁易读的源代码(即使对于非c ++程序员,我也没有做过)从一段时间开始练习C ++,尽管如此,我发现阅读非常好。

您可以使用VS构建它并对其进行调试,以快速找到您想要的内容。应该很容易实现TCP Ping中涉及的少量Win32 API调用。 这样,您当然可以轻松地将有趣的部分转换为C#或将其嵌入到托管DLL中。

如果它完全符合您的要求,请先测试一下。

来源链接:http://www.elifulkerson.com/projects/downloads/tcping-0.23/tcping-src.zip

答案 2 :(得分:0)

我已将以上tcpping.exePsPing.exe进行了比较。 PsPing.exe的输出结果非常接近窗口的ping.exetcpping.exe的输出变化很大。