iOS开发之runtime的运用-获取当前网络状态

时间:2021-07-02 06:34:14

之前写过runtime的一些东西,这次通过runtime获取一些苹果官方不想让你拿到的东西,比如,状态栏内部的控件属性。本文将通过runtime带你一步步拿到状态栏中显示网络状态的控件,然后通过监测该控件的属性来获取当前精确网络状态,比如2G/3G/4G/WIFI。

首先,我们需要拿到状态栏,然后通过runtime去探讨状态栏内部的组成结构。

1、导入运行时头文件

#import <objc/message.h>

2、编写运行时代码,获取到当前应用程序的所有成员变量

 #import "ViewController.h"
#import <objc/message.h> @interface ViewController () @end @implementation ViewController - (void)viewDidAppear:(BOOL)animated
{
// 状态栏是由当前app控制的,首先获取当前app
UIApplication *app = [UIApplication sharedApplication]; // 遍历当前app的所有属性,找到关于状态栏的
unsigned int outCount = ; Ivar *ivars = class_copyIvarList(app.class, &outCount); for (int i = ; i < outCount; i++) {
Ivar ivar = ivars[i];
printf("|%s", ivar_getName(ivar));
}
} @end

直接运行,可以看到打印结果为:iOS开发之runtime的运用-获取当前网络状态

3、可以看app里确实有个关于状态栏的成员变量,我们通过KVC取出它

 - (void)viewDidAppear:(BOOL)animated
{
// 状态栏是由当前app控制的,首先获取当前app
UIApplication *app = [UIApplication sharedApplication]; id statusBar = [app valueForKeyPath:@"statusBar"]; // 遍历状态栏的所有成员
unsigned int outCount = ;
Ivar *ivars = class_copyIvarList([statusBar class], &outCount); for (int i = ; i < outCount; i++) {
Ivar ivar = ivars[i];
printf("|%s", ivar_getName(ivar));
}
}

运行后可以看到打印结果为iOS开发之runtime的运用-获取当前网络状态

4、状态栏里有foregroundView这个成员,应该代表着所有当前显示的视图,通过KVC取出它里面的所有子视图

 // 状态栏是由当前app控制的,首先获取当前app
UIApplication *app = [UIApplication sharedApplication]; NSArray *children = [[[app valueForKeyPath:@"statusBar"] valueForKeyPath:@"foregroundView"] subviews]; for (id child in children) {
NSLog(@"--%@", [child class]);
}

打印结果为iOS开发之runtime的运用-获取当前网络状态

5、遍历数组,取出用于显示网络状态的视图,并遍历其内部的所有成员变量

 // 状态栏是由当前app控制的,首先获取当前app
UIApplication *app = [UIApplication sharedApplication]; NSArray *children = [[[app valueForKeyPath:@"statusBar"] valueForKeyPath:@"foregroundView"] subviews]; for (id child in children) {
if ([child isKindOfClass:NSClassFromString(@"UIStatusBarDataNetworkItemView")]) {
// 遍历当前状态栏的所有属性,找到关于状态栏的
unsigned int outCount = ;
Ivar *ivars = class_copyIvarList([child class], &outCount); for (int i = ; i < outCount; i++) {
Ivar ivar = ivars[i];
printf("|%s", ivar_getName(ivar));
}
}
}

打印结果为iOS开发之runtime的运用-获取当前网络状态

6、下面通过KVC,取出dataNetworkType

 if ([child isKindOfClass:NSClassFromString(@"UIStatusBarDataNetworkItemView")]) {
id type = [child valueForKeyPath:@"dataNetworkType"];
NSLog(@"_dataNetworkType class is %@, value is %@", [type class], type);
}

打印结果为: iOS开发之runtime的运用-获取当前网络状态

可见,dataNetworkType类型是NSNumber,值是5。【以上均为模拟器测试】

经过测试,发现,可能的值为 1,2,3,5 分别对应的网络状态是2G、3G、4G及WIFI。 当没有网络时,隐藏UIStatusBarDataNetworkItemView,无法获取dataNetworkType值

总结:

以下是完整的代码,并经过真机测试:

 - (void)viewDidAppear:(BOOL)animated
{
// 状态栏是由当前app控制的,首先获取当前app
UIApplication *app = [UIApplication sharedApplication]; NSArray *children = [[[app valueForKeyPath:@"statusBar"] valueForKeyPath:@"foregroundView"] subviews]; int type = ;
for (id child in children) {
if ([child isKindOfClass:NSClassFromString(@"UIStatusBarDataNetworkItemView")]) {
type = [[child valueForKeyPath:@"dataNetworkType"] intValue];
}
}
NSLog(@"----%d", type);
}

打印出的type数字对应的网络状态依次是: - 无网络; - 2G; - 3G; - 4G; - WIFI

建议: 将获取的UIStatusBarDataNetworkItemView保存起来,定时去取它的dataNetworkType,这样就可以实时监控网络状态啦(KVO在这里是行不通的哟)

当然,此方法存在一定的局限性,比如当状态栏被隐藏的时候,无法使用此方法。