iOS可达性SCNetworkReachabilitySetCallback在切换窗体wifi到anthor wifi时不回叫

时间:2018-05-14 04:08:57

标签: ios objective-c xcode networking reachability

我希望在网络更换iPhone时接听电话。

我使用Apple指南的Reachability.m:

xxx.m

struct sockaddr_in zeroAddress;
bzero(&zeroAddress, sizeof(zeroAddress));
zeroAddress.sin_len = sizeof(zeroAddress);
zeroAddress.sin_family = AF_INET;
Reachability = [self reachabilityWithAddress: (const struct sockaddr *) &zeroAddress]

Reachability.m

BOOL returnValue = NO;
SCNetworkReachabilityContext context = {0, (__bridge void *)(self), NULL, NULL, NULL};
if (SCNetworkReachabilitySetCallback(_reachabilityRef, ReachabilityCallback, &context))
{
    if (SCNetworkReachabilityScheduleWithRunLoop(_reachabilityRef, CFRunLoopGetMain(), kCFRunLoopDefaultMode))
    {
        returnValue = YES;
    }
}
return returnValue;

xxx.m

ReachabilityCallback {
    //do something when network change
}

我测试将移动设备(4G)更改为wifi1

收听像

这样的电话
Status4G
StatusNone
StatusWifi1

从wifi1更改为wifi2时

StatusWifi1
StatusNone
StatusWifi2

然而,有时候,当wifi变化非常快并且没有变为StatusNone时,当我将wifi1更改为wifi2或反过来时,我没有得到回调。

我想要的是

StatusWifi1
StatusWifi2

-------------------------UPDATE----------------------------

谢谢@Hitesh Surani,现在我的问题有时在某些设备上,我没有收到disconnect state,我尝试使用

[self reachabilityWithAddress: @"www.google.com"]

替换

[self reachabilityWithAddress: (const struct sockaddr *) &zeroAddress]

现在,即使没有进入disconnect state,我也可以收到回拨的电话,但我仍然不知道为什么,这是状态变化:

//wifi1:
1 kSCNetworkReachabilityFlagsReachable

//wifi1 -> wifi2
2 kSCNetworkReachabilityFlagsReachable | kSCNetworkReachabilityFlagsTransientConnection
3 kSCNetworkReachabilityFlagsReachable

当改变wifi时,有tmp状态kSCNetworkReachabilityFlagsReachable | kSCNetworkReachabilityFlagsTransientConnection,这就是我可以检测到wifi变化的原因,但kSCNetworkReachabilityFlagsTransientConnection是什么意思?我读过Apple的文档,仍然感到困惑。

1 个答案:

答案 0 :(得分:0)

感谢您提出了很好的问题。

主要Reachability类用于跟踪互联网连接更改。没有检测到wifi改变通知的证据。所以在一个单词Reachability中没有提供这种类型的功能。如果您想达到您的要求,则需要进行以下定制。

您可以通过以下方式获得wifi更改通知。

  • 当您的wifi连接发生变化或切换到任何其他网络时,您已经收到了断开状态的小数秒。因此调用reachabilityChanged方法。

  • 通过使用以下代码,您可以获得诸如BSSID,SSID(名称)和SSID数据之类的wifi信息。只需将wifi信息存储到本地。

  • 当网络发生变化时,您可以检查新的wifi信息是否与previos匹配。如果没有,则用户连接新的wifi。

将以下代码放入reachabilityChanged方法。

import SystemConfiguration.CaptiveNetwork

 func printCurrentWifiInfo() {
            if let interface = CNCopySupportedInterfaces() {
                for i in 0..<CFArrayGetCount(interface) {
                    let interfaceName: UnsafeRawPointer = CFArrayGetValueAtIndex(interface, i)
                    let rec = unsafeBitCast(interfaceName, to: AnyObject.self)
                    if let unsafeInterfaceData = CNCopyCurrentNetworkInfo("\(rec)" as CFString), let interfaceData = unsafeInterfaceData as? [String : AnyObject] {
                        // connected wifi
                        print("BSSID: \(String(describing: interfaceData["BSSID"])), SSID: \(String(describing: interfaceData["SSID"])), SSIDDATA: \(String(describing: interfaceData["SSIDDATA"]))")
                    } else {
                        //Wifi is not connected
                    }
                }
            }
        }

<强>更新

Apple提供com.apple.system.config.network_change来收听wifi更改通知。基本上它是Core Foundation框架的一部分。我相信它会对你有用。

请添加以下代码来收听wifi更改。

 let notificationName = "com.apple.system.config.network_change"


    func onNetworkChange(_ name : String) {
        if (name == notificationName) {
            // Do your stuff
            print("Network was changed")
        }
    }

    func registerObserver() {
        let observer = UnsafeRawPointer(Unmanaged.passUnretained(self).toOpaque())
        CFNotificationCenterAddObserver(CFNotificationCenterGetDarwinNotifyCenter(), observer,
                                        { (nc, observer, name, _, _) -> Swift.Void in
                                            if let observer = observer, let name = name {
                                                let instance = Unmanaged<Reachability>.fromOpaque(observer).takeUnretainedValue()
                                                instance.onNetworkChange(name.rawValue as String)
                                            } },
                                        notificationName as CFString, nil, .deliverImmediately)
    }


    func removeObserver() {
        let observer = UnsafeRawPointer(Unmanaged.passUnretained(self).toOpaque())
        CFNotificationCenterRemoveObserver(CFNotificationCenterGetDarwinNotifyCenter(), observer, nil, nil)
    }

在init上注册观察者并在deinit上删除。 查找Here的参考资料。