获取iOS设备WiFi Name和WiFiMac地址及iOS12以上系统兼容

苹果提供了NetworkExtension框架让开发者实现VPN以及WiFi热点相应的功能,虽然iOS9系统出来之后,这个框架的很多功能被苹果屏蔽了,需要申请权限才能用,但是本篇简单的获取WiFi名字和WiFi Mac地址信息的方法还是可以用的。

  • iPhone获取WiFi名字和WiFi Mac地址信息

获取WiFi相关信息的时候需要引入以下头文件

#import 
#import 

要拿到手机的WiFi名字和WiFi的mac地址我们只需要用到CNCopySupportedInterfacesCNCopyCurrentNetworkInfo两个类,实现方法如下:
1、获取SSIDService Set Identifier),服务集标识,也就是WiFi网络所取的名字。

+ (NSString *)wifiName
{
    NSArray *ifs = CFBridgingRelease(CNCopySupportedInterfaces());
    id info = nil;
    for (NSString *ifname in ifs) {
        info = (__bridge_transfer id)CNCopyCurrentNetworkInfo((CFStringRef) ifname);
        if (info && [info count]) {
            break;
        }
    }
    NSDictionary *dic = (NSDictionary *)info;
    NSString *ssid = [[dic objectForKey:@"SSID"] lowercaseString];

    return ssid;
}

2、获取BSSID即mac地址(注:这里的Mac地址是WiFi的Mac地址,不是硬件的Mac地址)。

+ (NSString *)wifiMac
{
    NSArray *ifs = CFBridgingRelease(CNCopySupportedInterfaces());
    id info = nil;
    for (NSString *ifname in ifs) {
        info = (__bridge_transfer id)CNCopyCurrentNetworkInfo((CFStringRef) ifname);
        if (info && [info count]) {
            break;
        }
    }
    NSDictionary *dic = (NSDictionary *)info;
    NSString *bssid = [dic objectForKey:@"BSSID"];

    return bssid;
}

获取到的结果如下:

SSID = mywifiname
BSSID = e3:fc:40:38:ac:e6

  • iOS 12系统获取WiFi名字和WiFiMac地址失败问题

这个方法在iOS12以下的系统完全可以胜任我们的任务,但是iOS12+的系统无法正常获取到WiFi信息。苹果对方法的使用提升了要求。

Important
To use this function in iOS 12 and later, enable the Access WiFi Information capability for your app in Xcode. When you enable this capability, Xcode automatically adds the Access WiFi Information entitlement to your entitlements file and App ID.
重要描述
在iOS 12+中使用此方法需要在Xcode中为应用授权获取WiFi信息的能力。授权后,Xcode会自动在App ID和应用的权限列表中增加获取WiFi信息的权限。

通过上面我们可以了解到,要在iOS12以上的系统中继续使用方法,就需要获取授权。如果你使用的是自动签名,授权之后Xcode会自动在App ID和应用的权限列表中增加WiFi的权限。如果你使用的是手动签名,可能还需要去App ID中配置一下权限,并生成新的profile文件。
具体的操作如下
Xcode11以下版本配置WiFi权限:设置Capabilities步骤:Target -> Capabilities -> Access WiFi Information -> ON

Xcode11以下开启Access WiFi Information

Xcode11以上版本配置WiFi权限:设置Capabilities步骤:Target -> Signing & Capabilities -> + Capability -> 选择Access WiFi Information双击添加。

Xcode11以上开启Access WiFi Information

如果项目使用的是手动签名,上面打开Access WiFi Information之后,可能Add the Access WiFi Information feature to your App ID这一项会报错,需要手动去App ID账号中设置。
打开Access WiFi Information之后,工程会在.entitlements 文件中添加Access WiFi Information信息,如果没有.entitlements文件会同时创建文件。

.entitlements文件中添加值

配置好上面的信息,在iOS12+的系统中获取WiFi名字和mac信息的方法就可以正常获取信息了。
如果采用的是手动签名,App ID中配置Access WiFi Information的方法如下,编辑App ID,勾选相应的选项。
App ID配置

然后重新生成项目的Provisioning Profiles文件,包括Distribution的和Development的,下载之后双击打开,项目就可以正常签名,并获取到WiFi信息了。


  • iOS 13系统获取WiFi名字和WiFiMac地址失败问题

系统升级到iOS 13之后,这个函数又获取不到我们要的信息了,根据苹果官方的说法:为了进一步保护用户隐私,并防止未经授权的位置跟踪(毕竟WiFi信息也可以暴露你的位置嘛),所以从iOS 13开始,CNCopyCurrentNetworkInfo API将不再返回有效的Wi-Fi SSIDBSSID信息,关于API CFDictionaryRef CNCopyCurrentNetworkInfo(CFStringRef interfaceName)官方说明如下:

The requesting app must meet one of the following requirements:
1)The app uses Core Location, and has the user’s authorization to use location information.
2)The app uses the NEHotspotConfiguration API to configure the current Wi-Fi network.
3)The app has active VPN configurations installed.
应用想要使用这个方法获取到WiFi信息必须满足下面三个条件之一:
1)应用程序使用Core Location,并具有用户使用位置信息的授权。
2)应用程序使用NEHotspotConfiguration API配置当前Wi-Fi网络。
3)应用程序已安装活动VPN配置。

An app that fails to meet any of the above requirements receives the following return value:
1)An app linked against iOS 12 or earlier receives a dictionary with pseudo-values. In this case, the SSID is Wi-Fi (or WLAN in the China region), and the BSSID is 00:00:00:00:00:00.
2)An app linked against iOS 13 or later receives NULL.
如果应用程序不满足上面的任何一个条件,那么此方法将返回下面结果:
iOS 12或更早的系统,将会返回一个带有伪值的字典。这种情况下
SSID(即WiFi名字)的值为Wi-Fi(中国地区的值为WLAN
BSSID(WiFi Mac地址)值为00:00:00:00:00:00
iOS 13或更高的系统,应用程序获取到的返回值为NULL。

Important
To use this function, an app linked against iOS 12 or later must enable the Access WiFi Information capability in Xcode. Calling this function without the entitlement always returns NULL when linked against iOS 12 or later.
This function returns NULL for iPad apps running in macOS.
重要描述
和iOS 12 系统出来的时候描述的一样,在iOS 12+系统使用此方法需要在Xcode中为应用授权Access WiFi Information,否则方法返回值为NULL。
同时在MacOS系统运行iPad应用的时候这个方法返回值也为NULL。

显然,苹果不再鼓励获取用户的WiFi信息。但是也没有把路堵死,符合上面提到的条件之一的应用程序,仍然可以获取到WiFi信息。
所以最简单的解决方案是,获取用户的地理位置权限。
取得用户的地理位置权限之后,上面的方法将可以成功获取到WiFi名字和WiFi Mac地址信息,代码不需要做任何修改。
另一种方法是使用NEHotSpotConfiguration API配置当前Wi-Fi网络
但是这个不现实,使用NEHotSpotConfiguration API配置Wi-Fi网络,需要用到WiFi的名字和密码,WiFi的连接信息只能用户手动输入。我们的应用程序大多是直接使用网络,不需要在使用WiFi的时候专门去配置,或者配置专用的WiFi网络,所以这个方法不可行。
建议采用第一种方法,开启用户位置权限。

你可能感兴趣的:(获取iOS设备WiFi Name和WiFiMac地址及iOS12以上系统兼容)