AFNetworking 框架小结 五(AFNetworkReachabilityManager)

2024-06-06 13:08

本文主要是介绍AFNetworking 框架小结 五(AFNetworkReachabilityManager),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

AFNetworkReachabilityManager

在 AFNetworking 框架中,提供了一个 AFNetworkReachabilityManager 类用来监控网络状态,其可能的状态如下:

  • AFNetworkReachabilityStatusUnknown 未知状态(-1)
  • AFNetworkReachabilityStatusNotReachable 不可达状态(0)
  • AFNetworkReachabilityStatusReachableViaWWAN 蜂窝网络连通(1)
  • AFNetworkReachabilityStatusReachableViaWiFi 无线网络连通(2)

通过 AFNetworkReachabilityManager 的属性 networkReachabilityStatus 可以获取当前网络的状态,这个属性值是上述状态之一。

还可以通过方法 isReachableisReachableViaWWANisReachableViaWiFi 判断当前网络是否连通或是蜂窝数据连通还是无线网络连通。

AFNetworkReachabilityManager 中提供了多个创建实例的方法,常用的是 sharedManager 方法,这也是 AFURLSessionManager 类中获取网络状态监视器的方法。
该方法获取一个公用的实例对象,其实际上也是调用 manager 方法,创建一个监控默认套接字地址是否可达的监视器。

当然,还可以直接调用 managerForAddress:managerForDomain: 方法来指点要监控的套接字地址或域名地址。

当获取到 AFNetworkReachabilityManager 实例对象后,根据需要可以使用下面的方法设置网络状态发生变化时的回调。

- (void)setReachabilityStatusChangeBlock:(nullable void (^)(AFNetworkReachabilityStatus status))block;

之后,调用实例方法 startMonitoring 便可以开启网络状态监控了,不需要时,可以调用方法 stopMonitoring 关闭监控。

每当网络发生变化时,除了调用设置的回调方法外,还会推送一个 AFNetworkingReachabilityDidChangeNotification 通知,该通知中的 userInfo 信息中的 AFNetworkingReachabilityNotificationStatusItem 包含网络状态信息。

SCNetworkReachabilityRef

网络监控的实现,关键是 SCNetworkReachabilityRef 变量的创建及设置。该变量及其相关的接口可以用来决定当前系统的网络状态,并且可以在网络状态发生变化时,发送通知。

在该接口里也定义了一些常量来标记指定节点域名或地址的网络状态:

  • kSCNetworkReachabilityFlagsTransientConnection 表示指定的节点或地址可以短暂连通,如 PPP
  • kSCNetworkReachabilityFlagsReachable 表示当前网络配置可以连通到指定的节点或地址
  • kSCNetworkReachabilityFlagsConnectionRequired 表示要使用当前网络配置连通到指定地址必需先建立连接,如拨号连接
  • kSCNetworkReachabilityFlagsConnectionOnTraffic 同样需要先建立连接才能传输网络数据,但是每一次传输数据都会初始化连接
  • kSCNetworkReachabilityFlagsConnectionAutomatic 等同于 kSCNetworkReachabilityFlagsConnectionOnTraffic
  • kSCNetworkReachabilityFlagsInterventionRequired 该状态表示除了要先建立连接外,还要提供诸如用户名、密码等信息的网络环境
  • kSCNetworkReachabilityFlagsConnectionOnDemand 该状态表示由 CFSocketStream 接口根据需要建立连接后可以连通到指定地址,其他接口并不会建立连接
  • kSCNetworkReachabilityFlagsIsLocalAddress 表示指定的地址是本地系统的地址
  • kSCNetworkReachabilityFlagsIsDirect 表示连通到指定地址的网络通信不会经过某个网关,而是直接通过本系统的某个接口
  • kSCNetworkReachabilityFlagsIsWWAN 表示指定的网络地址可以通过 EDGE 、GPRS 或其他蜂窝数据连接到达(只用于 iOS 系统)

除了上述的状态概念外,还需要理解一个网络可达性上下文结构,如下:

typedef struct {CFIndex     version;void *      __nullable info;const void  * __nonnull (* __nullable retain)(const void *info);void        (* __nullable release)(const void *info);CFStringRef __nonnull (* __nullable copyDescription)(const void *info);
} SCNetworkReachabilityContext;

在构造使用这个结构体时,version 设置为 0 ,info 是一个指向数据块的 C 指针,retain 是一个含有一个参数和一个返回值的函数,
该回调函数用来对 info 添加引用,与之对应,release 则是用来取消对 info 引用的回调函数,copyDescription 则可以返回对 info 的描述。

该接口里还对回调函数进行了定义:

typedef void (*SCNetworkReachabilityCallBack) (SCNetworkReachabilityRef target,            SCNetworkReachabilityFlags flags,void     *  __nullable  info
);

SCNetworkReachabilityCallBack 指向一个函数,该函数没有返回值,且包含三个参数,受监控的 target ,网络状态变化后的状态 flags ,
以及上下文中的 info 信息,这个信息就是 SCNetworkReachabilityContext 结构体中的 info 成员变量。

理解上面的定义后,现在再来看下面设置回调的函数,则十分简单了。

Boolean SCNetworkReachabilitySetCallback (SCNetworkReachabilityRef target,SCNetworkReachabilityCallBack __nullable callout,SCNetworkReachabilityContext * __nullable context
) __OSX_AVAILABLE_STARTING(__MAC_10_3,__IPHONE_2_0);

三个参数,即为 target 设置回调函数 callout ,同时 callout 关联着 context 上下文,其中可以保存 callout 要用的数据。

在设置回调函数之前,需要使用下面三个函数中的一个创建变量。

//指定一个网络地址
SCNetworkReachabilityRef __nullable SCNetworkReachabilityCreateWithAddress (CFAllocatorRef __nullable allocator,const struct sockaddr *address
) __OSX_AVAILABLE_STARTING(__MAC_10_3,__IPHONE_2_0);//指定网络连接的本地地址和远程地址
SCNetworkReachabilityRef __nullable SCNetworkReachabilityCreateWithAddressPair (CFAllocatorRef __nullable allocator,const struct sockaddr * __nullable localAddress,const struct sockaddr * __nullable remoteAddress
) __OSX_AVAILABLE_STARTING(__MAC_10_3,__IPHONE_2_0);//指定节点域名
SCNetworkReachabilityRef __nullable SCNetworkReachabilityCreateWithName (CFAllocatorRef __nullable allocator,const char *nodename
) __OSX_AVAILABLE_STARTING(__MAC_10_3,__IPHONE_2_0);

前两个方法中都使用下面的结构体保存网络地址:

struct sockaddr {__uint8_t       sa_len;     /* 总长度 */sa_family_t  sa_family; /* 协议簇 */char             sa_data[14];   /* 地址 */
};

上面这个结构体和下面的结构体可以通用

struct sockaddr_in {__uint8_t       sin_len;sa_family_t  sin_family;in_port_t        sin_port;struct  in_addr  sin_addr;char              sin_zero[8]; //为了兼容 sockaddr 结构体而保留的成员变量
};

这两个结构体都占有 16 Bytes 即 sa_lensin_len 的值都是 16 。

创建并设置好 SCNetworkReachabilityRef 变量后,还需要将变量加入到运行循环中,取消监控时,对应的要将该变量移除运行循环。

Boolean SCNetworkReachabilityScheduleWithRunLoop (SCNetworkReachabilityRef target,CFRunLoopRef            runLoop,CFStringRef         runLoopMode
) __OSX_AVAILABLE_STARTING(__MAC_10_3,__IPHONE_2_0);Boolean SCNetworkReachabilityUnscheduleFromRunLoop (SCNetworkReachabilityRef target,CFRunLoopRef            runLoop,CFStringRef         runLoopMode
) __OSX_AVAILABLE_STARTING(__MAC_10_3,__IPHONE_2_0);

另外,还可以使用下面的函数指定回调函数调用的队列:

Boolean SCNetworkReachabilitySetDispatchQueue (SCNetworkReachabilityRef target,dispatch_queue_t __nullable queue
) __OSX_AVAILABLE_STARTING(__MAC_10_6,__IPHONE_4_0);

使用下面的方法获取监控对象的网络状态:

Boolean SCNetworkReachabilityGetFlags (SCNetworkReachabilityRef    target,SCNetworkReachabilityFlags *flags
) __OSX_AVAILABLE_STARTING(__MAC_10_3,__IPHONE_2_0);

startMonitoring

在获取了 AFNetworkReachabilityManager 监视器后,需要调用 startMonitoring 方法来开启监控。

在这个开启方法中,会对创建的 SCNetworkReachabilityRef 变量进行设置,不管这个变量是由域名或地址创建的。

参见下面的源代码:

- (void)startMonitoring {[self stopMonitoring];if (!self.networkReachability) {return;}//这里 callback 与 networkReachabilityStatusBlock 是相同类型的回调代码块//此处进行了包裹,是为了避免 networkReachabilityStatusBlock 为 nil __weak __typeof(self)weakSelf = self;AFNetworkReachabilityStatusBlock callback = ^(AFNetworkReachabilityStatus status) {__strong __typeof(weakSelf)strongSelf = weakSelf;strongSelf.networkReachabilityStatus = status;if (strongSelf.networkReachabilityStatusBlock) {strongSelf.networkReachabilityStatusBlock(status);}};//构建上下文,包含回调代码块SCNetworkReachabilityContext context = {0, (__bridge void *)callback, AFNetworkReachabilityRetainCallback, AFNetworkReachabilityReleaseCallback, NULL};//设置 AFNetworkReachabilityCallback 为回调函数//这个函数中,会对上面的 callback 进行调用,并推送 AFNetworkingReachabilityDidChangeNotification 通知SCNetworkReachabilitySetCallback(self.networkReachability, AFNetworkReachabilityCallback, &context);//加入运行循环SCNetworkReachabilityScheduleWithRunLoop(self.networkReachability, CFRunLoopGetMain(), kCFRunLoopCommonModes);dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_BACKGROUND, 0),^{SCNetworkReachabilityFlags flags;//首先查询了一次网络状态if (SCNetworkReachabilityGetFlags(self.networkReachability, &flags)) {AFPostReachabilityStatusChange(flags, callback);}});
}

这篇关于AFNetworking 框架小结 五(AFNetworkReachabilityManager)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1036196

相关文章

Java中的String.valueOf()和toString()方法区别小结

《Java中的String.valueOf()和toString()方法区别小结》字符串操作是开发者日常编程任务中不可或缺的一部分,转换为字符串是一种常见需求,其中最常见的就是String.value... 目录String.valueOf()方法方法定义方法实现使用示例使用场景toString()方法方法

Java中List的contains()方法的使用小结

《Java中List的contains()方法的使用小结》List的contains()方法用于检查列表中是否包含指定的元素,借助equals()方法进行判断,下面就来介绍Java中List的c... 目录详细展开1. 方法签名2. 工作原理3. 使用示例4. 注意事项总结结论:List 的 contain

Flutter打包APK的几种方式小结

《Flutter打包APK的几种方式小结》Flutter打包不同于RN,Flutter可以在AndroidStudio里编写Flutter代码并最终打包为APK,本篇主要阐述涉及到的几种打包方式,通... 目录前言1. android原生打包APK方式2. Flutter通过原生工程打包方式3. Futte

Docker镜像pull失败两种解决办法小结

《Docker镜像pull失败两种解决办法小结》有时候我们在拉取Docker镜像的过程中会遇到一些问题,:本文主要介绍Docker镜像pull失败两种解决办法的相关资料,文中通过代码介绍的非常详细... 目录docker 镜像 pull 失败解决办法1DrQwWCocker 镜像 pull 失败解决方法2总

Python Dash框架在数据可视化仪表板中的应用与实践记录

《PythonDash框架在数据可视化仪表板中的应用与实践记录》Python的PlotlyDash库提供了一种简便且强大的方式来构建和展示互动式数据仪表板,本篇文章将深入探讨如何使用Dash设计一... 目录python Dash框架在数据可视化仪表板中的应用与实践1. 什么是Plotly Dash?1.1

基于Flask框架添加多个AI模型的API并进行交互

《基于Flask框架添加多个AI模型的API并进行交互》:本文主要介绍如何基于Flask框架开发AI模型API管理系统,允许用户添加、删除不同AI模型的API密钥,感兴趣的可以了解下... 目录1. 概述2. 后端代码说明2.1 依赖库导入2.2 应用初始化2.3 API 存储字典2.4 路由函数2.5 应

Python GUI框架中的PyQt详解

《PythonGUI框架中的PyQt详解》PyQt是Python语言中最强大且广泛应用的GUI框架之一,基于Qt库的Python绑定实现,本文将深入解析PyQt的核心模块,并通过代码示例展示其应用场... 目录一、PyQt核心模块概览二、核心模块详解与示例1. QtCore - 核心基础模块2. QtWid

Android Kotlin 高阶函数详解及其在协程中的应用小结

《AndroidKotlin高阶函数详解及其在协程中的应用小结》高阶函数是Kotlin中的一个重要特性,它能够将函数作为一等公民(First-ClassCitizen),使得代码更加简洁、灵活和可... 目录1. 引言2. 什么是高阶函数?3. 高阶函数的基础用法3.1 传递函数作为参数3.2 Lambda

Ollama Python 使用小结

《OllamaPython使用小结》Ollama提供了PythonSDK,使得开发者能够在Python环境中轻松集成和使用本地运行的模型进行自然语言处理任务,具有一定的参考价值,感兴趣的可以了解一... 目录安装 python SDK启动本地服务使用 Ollama 的 Python SDK 进行推理自定义客

java String.join()的使用小结

《javaString.join()的使用小结》String.join()是Java8引入的一个实用方法,用于将多个字符串按照指定分隔符连接成一个字符串,本文主要介绍了javaString.join... 目录1. 方法定义2. 基本用法2.1 拼接多个字符串2.2 拼接集合中的字符串3. 使用场景和示例3