AFNetworking源码学习(一)- AFNetworkReachabilityManager

AFNetworkReachabilityManager是监控网络环境变化的类

AFNetworkReachabilityManager.h

See Apple's Reachability Sample Code ( https://developer.apple.com/library/ios/samplecode/reachability/ )

从这部分注释可知,AFNetworkReachabilityManager这个类参考自苹果官方Reachability样例代码。

#import <SystemConfiguration/SystemConfiguration.h>

typedef NS_ENUM(NSInteger, AFNetworkReachabilityStatus) {
    AFNetworkReachabilityStatusUnknown          = -1,
    AFNetworkReachabilityStatusNotReachable     = 0,
    AFNetworkReachabilityStatusReachableViaWWAN = 1,
    AFNetworkReachabilityStatusReachableViaWiFi = 2,
};

从这部分可以看出,网络监控的实现依赖SystemConfiguration这个api;至于NS_ENUM定义了AFNetworkReachabilityStatus网络状态的枚举。

NS_ASSUME_NONNULL_BEGIN
NS_ASSUME_NONNULL_END

这个是为了swift的可选类型配置添加的,其中间的参数默认都是nonnull的。

接下来,声明了以下几个属性:

  • networkReachabilityStatus:网络状态
  • reachable:是否是可达的
  • reachableViaWWAN:是否是WWAN
  • reachableViaWiFi:是否是WiFi

注意BOOL属性一般是要写getter方法的:

/**
 Whether or not the network is currently reachable.
 */
@property (readonly, nonatomic, assign, getter = isReachable) BOOL reachable;

然后,就是初始化方法:

+ (instancetype)sharedManager;
+ (instancetype)manager;
+ (instancetype)managerForDomain:(NSString *)domain;
+ (instancetype)managerForAddress:(const void *)address;
- (instancetype)initWithReachability:(SCNetworkReachabilityRef)reachability NS_DESIGNATED_INITIALIZER; 

5个初始化方法,满足了大部分需求,同样的,参考苹果官方的类,以NSString为例:

+ string
- init
- initWithBytes:length:encoding:
- initWithString:
- initWithFormat:
.....

所以,封装一个类的时候,应多考虑不同的场景、不同的需求

// 开始监控
- (void)startMonitoring;
// 停止监控   
- (void)stopMonitoring; 
// 获取网络状态的本地语言的字符串   
- (NSString *)localizedNetworkReachabilityStatusString;
// 设置网络状态改变的block回调
- (void)setReachabilityStatusChangeBlock:(nullable void (^)(AFNetworkReachabilityStatus status))block;

监听网络改变的回调有两种方式:

  • 使用setReachabilityStatusChangeBlock:方法
  • 监听AFNetworkingReachabilityDidChangeNotification通知

在.h文件最后使用了FOUNDATION_EXPORT声明了两个通知key的常量和一个函数:

FOUNDATION_EXPORT NSString * const AFNetworkingReachabilityDidChangeNotification;
FOUNDATION_EXPORT NSString * const AFNetworkingReachabilityNotificationStatusItem;
// 根据网络状态获取相应字符串
FOUNDATION_EXPORT NSString * AFStringFromNetworkReachabilityStatus(AFNetworkReachabilityStatus status);

苹果官方样例代码使用的是extern,那这里为什么使用FOUNDATION_EXPORT呢?
查看FOUNDATION_EXPORT的定义:

#if defined(__cplusplus)
#define FOUNDATION_EXTERN extern "C"
#else
#define FOUNDATION_EXTERN extern
#endif

#if TARGET_OS_WIN32

    #if defined(NSBUILDINGFOUNDATION)
        #define FOUNDATION_EXPORT FOUNDATION_EXTERN __declspec(dllexport)
    #else
        #define FOUNDATION_EXPORT FOUNDATION_EXTERN __declspec(dllimport)
    #endif

    #define FOUNDATION_IMPORT FOUNDATION_EXTERN __declspec(dllimport)

#else
    #define FOUNDATION_EXPORT  FOUNDATION_EXTERN
    #define FOUNDATION_IMPORT FOUNDATION_EXTERN
#endif

#if !defined(NS_INLINE)
    #if defined(__GNUC__)
        #define NS_INLINE static __inline__ __attribute__((always_inline))
    #elif defined(__MWERKS__) || defined(__cplusplus)
        #define NS_INLINE static inline
    #elif defined(_MSC_VER)
        #define NS_INLINE static __inline
    #elif TARGET_OS_WIN32
        #define NS_INLINE static __inline__
    #endif
#endif

可以知道,FOUNDATION_EXPORT相当于C语言中的extern、C++中的extern "C",还有Win32上的类似特性,所以可以说FOUNDATION_EXPORT是通用的,而且它能够隐藏定义细节,这应该就是使用FOUNDATION_EXPORT的原因吧。

AFNetworkReachabilityManager.m

首先就是两个通知key值常量的具体定义:

NSString * const AFNetworkingReachabilityDidChangeNotification = @"com.alamofire.networking.reachability.change";
NSString * const AFNetworkingReachabilityNotificationStatusItem = @"AFNetworkingReachabilityNotificationStatusItem";

但是,我觉得这块处理的不好,且风格不统一,我们可参考苹果官方的:

NSString * const kReachabilityChangedNotification = @"kNetworkReachabilityChangedNotification";

使用k做前缀代表key值常量,可读性好,且易于查找。

typedef void (^AFNetworkReachabilityStatusBlock)(AFNetworkReachabilityStatus status);

定义了一个回传status的block。

接下来就是.h文件声明的AFStringFromNetworkReachabilityStatus函数的实现,主要是根据status获取本地化的字符串,值得一提的是NSLocalizedStringFromTable,它与NSLocalizedString的区别在于,存放本地化字符串的文件是自定义的,这里就是AFNetworking.strings,而不是Localizable.strings,第三方框架使用自定义文件名是必要的,易于区分,且可避免冲突。

再之后就是几个私有方法,后面三个是辅助方法就不说了,重要的是前面两个:

  • 第一个方法,是根据SCNetworkReachabilityFlags这个标记转换成自定义的AFNetworkReachabilityStatus枚举类型;
  • 第二个方法,是当AFNetworkReachabilityStatus改变时block回调,并发送AFNetworkingReachabilityDidChangeNotification通知。

值得注意的是,私有方法的写法:static void function(),平时我们可能是这样写:- (void)function;查看了苹果官方样例代码,也是它这样写的,以及一些著名的开发框架都是这样写的,那为什么要这样写呢?

使用static用于函数定义时,对函数的连接方式产生影响,使得函数只在本文件内部有效,对其他文件是不可见的。这样的函数又叫作静态函数,这种方式对函数本身是一种保护机制。而用在这里,在文件最前方,易于查找;且可适当使用内联函数,提高效率。

而初始化方法中,前面四个用的都是类方法,第一个是创建单例,调用了+manager,该方法使用预编译条件编译,用于兼容iOS9之后推出的IPv6,而最后调用了+managerForAddress:这个方法,该方法是根据地址初始化,而+managerForDomain:是根据域名初始化。具体实现就不多说了,都是根据官方样例代码而来的。而最后-initWithReachability:根据SCNetworkReachabilityRef初始化,并赋了初值:

_networkReachability = CFRetain(reachability);   //此处用了CFRetain,释放的时候注意要CFRelease
self.networkReachabilityStatus = AFNetworkReachabilityStatusUnknown;//初值为未知状态

继续往下,就是dealloc方法:

- (void)dealloc {
    [self stopMonitoring];//停止监控
    
    if (_networkReachability != NULL) {
        CFRelease(_networkReachability);//CF类型的,且CFRetain赋值过,一定要CFRelease处理
    }
}

然后就是,isReachableisReachableViaWWANisReachableViaWiFi三个getter方法,我们可以看到,都是键值关联的,且都围绕着self.networkReachabilityStatus这个值。在文件最后看到KVO的实现:

+ (NSSet *)keyPathsForValuesAffectingValueForKey:(NSString *)key {
    if ([key isEqualToString:@"reachable"] || [key isEqualToString:@"reachableViaWWAN"] || [key isEqualToString:@"reachableViaWiFi"]) {
        return [NSSet setWithObject:@"networkReachabilityStatus"];
    }

    return [super keyPathsForValuesAffectingValueForKey:key];
}

更加证实这一点,它们是键值依赖的,self.networkReachabilityStatus这个值变化,其他三个BOOL值也相应的变化。注册键值依赖这样的做法,值得我们学习,这可以让代码看起来更加精简、优雅。

至于-localizedNetworkReachabilityStatusString-setReachabilityStatusChangeBlock:这两个方法就不讲了,getter、setter方法,很简单。

接下来,就是这个类的两个核心方法:

  • startMonitoring:开始监控
  • stopMonitoring:停止监控

startMonitoring方法中,其核心是:

SCNetworkReachabilityContext context = {0, (__bridge void *)callback, AFNetworkReachabilityRetainCallback, AFNetworkReachabilityReleaseCallback, NULL};
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);
        }
    });

SCNetworkReachabilityContext是一个结构体,其定义是:

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;

第一个参数,看CFIndex的定义:

typedef signed long CFIndex;

可知,这是一个signed long类型的参数;
第二个参数,是一个void * 类型的值,即可以指向任何类型的参数;
第三个参数,是一个函数,目的是对info做retain操作;
第四个参数,是一个函数,目的是对info做release操作;
第五个参数,是一个函数,根据info获取Description字符串。

那么,携带的info即为callback,也就是下面这个block:

__weak __typeof(self)weakSelf = self;
    AFNetworkReachabilityStatusBlock callback = ^(AFNetworkReachabilityStatus status) {
        __strong __typeof(weakSelf)strongSelf = weakSelf;

        strongSelf.networkReachabilityStatus = status;
        if (strongSelf.networkReachabilityStatusBlock) {
            strongSelf.networkReachabilityStatusBlock(status);
        }

    };

查看SCNetworkReachabilityContext的文档:

Structure containing user-specified data and callbacks used with SCNetworkReachabilitySetCallback.

使用SCNetworkReachabilitySetCallback设置其回调:

SCNetworkReachabilitySetCallback(self.networkReachability, AFNetworkReachabilityCallback, &context);

文档说明:which receives callbacks when the reachability of the target changes.(当目标的可达性变化时接收回调)。

SCNetworkReachabilityScheduleWithRunLoop(self.networkReachability, CFRunLoopGetMain(), kCFRunLoopCommonModes);

SCNetworkReachabilityScheduleWithRunLoop的文档中说明:Schedules the specified network target with the specified run loop and mode.
安排指定的网络目标到指定的run loop和mode中。

接下来,开一个异步线程,用于发送网络状态:

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_BACKGROUND, 0),^{
        SCNetworkReachabilityFlags flags;
        if (SCNetworkReachabilityGetFlags(self.networkReachability, &flags)) {
            AFPostReachabilityStatusChange(flags, callback);
        }
    });

stopMonitoring中刚好相反:

SCNetworkReachabilityUnscheduleFromRunLoop(self.networkReachability, CFRunLoopGetMain(), kCFRunLoopCommonModes);

不安排到run loop和mode中,即停止监控。

总结

虽然这是一个并不复杂的类,且很多实现都来源于官方样例代码,但其中很多细节和技巧是值得我们学习的:

  • NS_ENUM枚举的恰当使用
  • NS_ASSUME_NONNULL_BEGIN、NS_ASSUME_NONNULL_END
  • BOOL属性一般要写getter方法,使用键值依赖能事半功倍
  • 考虑不同场景、不同需求,提供相应的初始化方法
  • FOUNDATION_EXPORT声明常量以及函数是通用的,且需注意key值常量的格式统一
  • 注意字符串的本地化
  • 回调使用block、notification,便于使用者选择
  • 静态函数用于私有方法,放在文件最前面,易于查找管理
  • CF类型,注意CFRelease
  • 适当场景,使用异步线程

参考资料:
AFNetworking 3.0 源码解读(一)之 AFNetworkReachabilityManager
AFNetworkReachabilityManager Class Reference
Reachability

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,386评论 6 479
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,939评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,851评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,953评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,971评论 5 369
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,784评论 1 283
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,126评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,765评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,148评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,744评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,858评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,479评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,080评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,053评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,278评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,245评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,590评论 2 343

推荐阅读更多精彩内容