iOS中ping-SimplePing的使用

作者: 小盟城主 | 来源:发表于2019-02-19 19:15 被阅读6次

    前言

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

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

    Ping

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

    SimplePing类结构

    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 操作的流程如下图所示:

    SimplePing流程

    注意SimplePing对象实例化一定要使用全局对象

    Demo下载

    参考文档

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

    相关文章

      网友评论

        本文标题:iOS中ping-SimplePing的使用

        本文链接:https://www.haomeiwen.com/subject/jhoaeqtx.html