iOS中ping-SimplePing的使用

前言

  手机网络连接状态的检测对于iOS 开发来说是一个非常基础的需求。为了更好的用户体验,我们会在无网络时展现本地或者缓存的内容,并对用户进行合适的提示。对于网络状态的检测,苹果提供了Reachability,由此也衍生出各种 Reachability 框架,比较著名的有Github上的 tonymillion/Reachability 以及 AFNetworking 中的 AFNetworkReachabilityManager 模块,它们的实现原理基本上是完全相同的。
  但事实上,此方案能获取的只是设备的本地连接状态,有时它很难为我们检测真正的网络连接状态,如以下场景:

  • 现在很多的公共场所的 WiFi,需要网页登录授权,授权之前无法上网,但本地连接已经建立;
  • 存在了本地网络连接,但信号很差,实际无法连接到服务器;
  • iOS 连接的路由设备本身没有连接外网等。

Ping

  ping 是 WindowsUnixLinuxmacOS 等系统下一个常用的命令,利用 ping 命令可以用来测试数据包 (ICMP) 能否通过IP 协议到达特定主机,并收到主机的应答,以检查网络是否连通和网络连接速度,帮助我们分析和判定网络故障。
  SimplePing是苹果封装好的ping的功能,它利用resolve hostcreate socket(send&recv data),解析ICMP 包验证 checksum 等实现了 ping功能。并且支持iPv4iPv6

SimplePing类结构

iOS中ping-SimplePing的使用_第1张图片
SimplePing.h

SimplePing的使用很简单:

   // 1. 利用 HostName 创建 SimplePing
    self.simplePing = [[SimplePing alloc] initWithHostName:@"www.apple.com"];
    // 2. 指定 IP 地址类型,默认是都支持
    if (isIpv4 && !isIpv6) {
        self.simplePing.addressStyle = SimplePingAddressStyleICMPv4;
    }else if (isIpv6 && !isIpv4) {
        self.simplePing.addressStyle = SimplePingAddressStyleICMPv6;
    }
    // 3. 设置 delegate,用于接收回调信息
    self.simplePing.delegate = self;
    // 4. 开始 ping
    [self.simplePing start];

然后在delegate中进行回调,代理方法中提现了ping的过程:

// [self.simplePing start] 成功之后,解析 HostName 拿到 ip 地址之后,发送封包
- (void)simplePing:(SimplePing *)pinger didStartWithAddress:(NSData *)address
{
    //这里面可以创建一个时间戳NSTimer定时发送数据监测
    [self.simplePing sendPingWithData:nil];
}

// [self.simplePing start] ping 功能启动失败
- (void)simplePing:(SimplePing *)pinger didFailWithError:(NSError *)error
{
    //停止所以的请求,如果有时间戳也停止
    [self.simplePing stop];
}

// sendPingWithData 发送数据成功 
- (void)simplePing:(SimplePing *)pinger didSendPacket:(NSData *)packet sequenceNumber:(uint16_t)sequenceNumber
{
    NSLog(@"#%u sent", sequenceNumber);
}

 // sendPingWithData 发送数据失败,并返回错误信息
- (void)simplePing:(SimplePing *)pinger didFailToSendPacket:(NSData *)packet sequenceNumber:(uint16_t)sequenceNumber error:(NSError *)error
{
     NSLog(@"#%u send failed: %@", sequenceNumber,error);
}

// ping 发送封包之后收到响应
- (void)simplePing:(SimplePing *)pinger didReceivePingResponsePacket:(NSData *)packet sequenceNumber:(uint16_t)sequenceNumber
{
    NSLog(@"#%u received, size=%zu", sequenceNumber, packet.length);
}

// ping 接收响应封包发生异常
- (void)simplePing:(SimplePing *)pinger didReceiveUnexpectedPacket:(NSData *)packet
{
    NSLog(@"unexpected packet, size=%zu", packet.length);
}

  梳理了一下使用SimplePing类进行 ping 操作的流程如下图所示:

iOS中ping-SimplePing的使用_第2张图片
SimplePing流程

注意SimplePing对象实例化一定要使用一直持有对象的属性方法。
  因此,在Reachability的方法优化后采用了ping更加精确的获取网络的连接状态,网上有的不错的第三方库比如RealReachability就是采用了这种方法。

Demo下载

参考文档

https://blog.csdn.net/mnbvcxz111123/article/details/54286050
Traceroute(路由追踪)的原理及实现
iOS--网络测试小工具(ping tarceroute)
网易网络诊断

你可能感兴趣的:(iOS中ping-SimplePing的使用)