美文网首页
AFNetWorking涉及的知识点

AFNetWorking涉及的知识点

作者: woniu | 来源:发表于2018-02-24 18:27 被阅读14次

    这里我们只是向大家简单介绍下AFNetWorking涉及到的知识点,我们在之前以及之后的文章会详细分析这些知识点。

    一、Block的写法。

    详细地址+Demo:https://www.jianshu.com/p/73e8796942ec

    二、FOUNDATION_EXPORT定义与#define定义的区别

    #define kMyConstantString @"Hello"
    
    //.h
    FOUNDATION_EXPORT NSString * const AFNetworkingReachabilityDidChangeNotification;
    FOUNDATION_EXPORT NSString * const AFNetworkingReachabilityNotificationStatusItem;
    
    //.m
    NSString * const AFNetworkingReachabilityDidChangeNotification = @"com.alamofire.networking.reachability.change";
    NSString * const AFNetworkingReachabilityNotificationStatusItem = @"AFNetworkingReachabilityNotificationStatusItem";
    
    • FOUNDATION_EXPORT 与 #define 都可以用来定义常量
    • FOUNDATION_EXPORT定义的字符串直接比较的是指针地址
    • (#define)定义的字符串则是一一比较字符串的每一个字符是否相等

    FOUNDATION系列函数
    对比资料视图传送门

    重点:

    1、FOUNDATION_EXPORT NSString * const定义的多个变量,不管它们的值相同或者不同,它们指针指向的内存地址都相同。
    2、#define定义的多个变量,如果变量值相同,那么内存地址相同,值不同,内存地址则不同。

    为什么内存地址会相同呢?

    等找到原因了再补充下。

    三、关于使用C语言写私有方法

    我个人的意见是一个类中的私有方法写成static void funcName() 这样的c函数比较好。

    3.1. 在文件的最前方,比较容易查找

    3.2. 可以适当的使用内联函数,提高效率。

    四、AFSecurityPolicy

    4.1. AFSSLPinningModeNone 代表无条件信任服务器的证书

    4.2. AFSSLPinningModePublicKey 代表会对服务器返回的证书中的PublicKey进行验证,通过则通过,否则不通过

    4.3. AFSSLPinningModeCertificate 代表会对服务器返回的证书同本地证书全部进行校验,通过则通过,否则不通过

    五、私有方法

    开发中,使用私有化方法来协助我们达到某种目的或获取某个数据,且方法内部没有使用self,类似于这样的方法,跟我们的业务并没有太大的关系,此时我们可以使用static void这样的私有化方法.

    static void AFPostReachabilityStatusChange(SCNetworkReachabilityFlags flags, AFNetworkReachabilityStatusBlock block) {
        AFNetworkReachabilityStatus status = AFNetworkReachabilityStatusForFlags(flags);
        dispatch_async(dispatch_get_main_queue(), ^{
            if (block) {
                block(status);
            }
            NSNotificationCenter *notificationCenter = [NSNotificationCenter defaultCenter];
            NSDictionary *userInfo = @{ AFNetworkingReachabilityNotificationStatusItem: @(status) };
            [notificationCenter postNotificationName:AFNetworkingReachabilityDidChangeNotification object:nil userInfo:userInfo];
        });
    }
    

    六、代码思维碰撞

    当遇到比较复杂功能的时候,能不能写出一个思路图,先不管思路的实现如何,先一一列出来,最后再一一实现,一一拼接起来。 没有全局观是不行的。

    七、NSIndexSet

    定义:NSIndexSet是一个有序的,唯一的,无符号整数的集合。

    有序的输出数据,从小到大。

    应用场景:删除表的区到就会用到indexset 来作为索引,才知道删除第几个区。

    https://www.jianshu.com/p/84a1d5296844

    八、锁

    AFNetworking中我们总共见过三种锁。

    • synchronized 美 ['sɪŋkrənaɪzd]
    - (BOOL)isNetworkActivityOccurring {
        @synchronized(self) {
            return self.activityCount > 0;
        }
    }
    
    • dispatch_semaphore_wait 美 ['sɛməfɔr] 信号量
    - (NSArray *)tasksForKeyPath:(NSString *)keyPath {
        __block NSArray *tasks = nil;
        dispatch_semaphore_t semaphore = dispatch_semaphore_create(0);
        [self.session getTasksWithCompletionHandler:^(NSArray *dataTasks, NSArray *uploadTasks, NSArray *downloadTasks) {
            if ([keyPath isEqualToString:NSStringFromSelector(@selector(dataTasks))]) {
                tasks = dataTasks;
            } else if ([keyPath isEqualToString:NSStringFromSelector(@selector(uploadTasks))]) {
                tasks = uploadTasks;
            } else if ([keyPath isEqualToString:NSStringFromSelector(@selector(downloadTasks))]) {
                tasks = downloadTasks;
            } else if ([keyPath isEqualToString:NSStringFromSelector(@selector(tasks))]) {
                tasks = [@[dataTasks, uploadTasks, downloadTasks] valueForKeyPath:@"@unionOfArrays.self"];
            }
    
            dispatch_semaphore_signal(semaphore);
        }];
    
        dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
    
        return tasks;
    }
    
    • NSLock
    + (UIImage *)af_safeImageWithData:(NSData *)data {
        UIImage* image = nil;
        static dispatch_once_t onceToken;
        dispatch_once(&onceToken, ^{
            imageLock = [[NSLock alloc] init];
        });
        
        [imageLock lock];
        image = [UIImage imageWithData:data];
        [imageLock unlock];
        return image;
    }
    

    九、NSURLCache

    NSURLCache为应用的URL提供了内摧毁总以及磁盘上的综合缓存机制。网络缓存减少了向服务器发送请求的次数,通知也提升了离线或在低速网路中使用应用的体验。当一个请求完成了来自服务器的回应,回应将会在本地保存。下一次同一个请求再发起时,本地保存的回一个就会马上返回,不需要链接服务器。

    - (BOOL)application:(UIApplication *)application
        didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
    {
      NSURLCache *URLCache = [[NSURLCache alloc] initWithMemoryCapacity:4 * 1024 * 1024
                                                           diskCapacity:20 * 1024 * 1024
                                                               diskPath:nil];
      [NSURLCache setSharedURLCache:URLCache];
    }
    

    URLRequest 有个 cachePolicy 属性,我们平时最常用的有四个属性:

    • NSNSURLRequestUseProtocolCachePolicy: 对特定的 URL 请求使用网络协议中实现的缓存逻辑。这是默认的策略
    • NSURLRequestReloadIgnoringLocalCacheData:数据需要从原始地址加载。不使用现有缓存。
    • NSURLRequestReturnCacheDataElseLoad:无论缓存是否过期,先使用本地缓存数据。如果缓存中没有请求所对应的数据,那么从原始地址加载数据
    • NSURLRequestReturnCacheDataDontLoad:无论缓存是否过期,先使用本地缓存数据。如果缓存中没有请求所对应的数据,那么放弃从原始地址加载数据,请求视为失败(即:“离线”模式)。

    十、内联函数(inline)

    我们在AFNetWorking中经常见到inline(内联函数),心里多多少少会犯嘀咕,这是什么函数?有什么用?用了有什么好处?什么时候用?

    static inline NSString * AFContentTypeForPathExtension(NSString *extension) {
    #ifdef __UTTYPE__
        NSString *UTI = (__bridge_transfer NSString *)UTTypeCreatePreferredIdentifierForTag(kUTTagClassFilenameExtension, (__bridge CFStringRef)extension, NULL);
        NSString *contentType = (__bridge_transfer NSString *)UTTypeCopyPreferredTagWithClass((__bridge CFStringRef)UTI, kUTTagClassMIMEType);
        if (!contentType) {
            return @"application/octet-stream";
        } else {
            return contentType;
        }
    #else
        return @"application/octet-stream";
    #endif
    }
    
    • 什么是内联函数(inline)
      定义:内联函数是指用inline关键字修饰的函数。在类内定义的函数被默认成内联函数。 当编译器发现某段代码在调用一个内联函数时,它不是去调用该函数,而是将该函数的代码整段插入到当前位置。这样做的好处是省去了调用的过程,加快了程序运行速度。(函数的调用过程,由于有前面所说的参数入栈等操作,所以总要多占用一些时间)
      坏处:由于每当代码调用到内联函数,就需要在调用处直接插入一段该函数的代码,所以程序的体积将增大。
      内联函数的本质是:节省时间但是消耗空间。
      这么说吧,就是我们早期抗战的原则:拿空间换时间。

    • 内联函数调用效率
      1、函数之间的调用是内存地址之间的调用,当函数调用完毕之后,还会返回原来函数执行的地址。函数调用有时间开销,内联函数就是为了解决这个问题。
      2、不用inline修饰的函数,汇编时会调用call指令:
      . call指令是计算机转移到调用的子程序,两步操作:
      (1)将程序下一条指令的位置的IP压入堆栈中;
      (2)转移到调用的子程序

    • 内联函数使用规则
      1、一个函数可以自己调用自已,称为递归调用(后面讲到),含有递归调用的函数不能设置为inline;
      2、使用了复杂流程控制语句:循环语句和switch语句,无法设置为inline;(这一点存疑,AFNetWorking里面有使用switch的内联函数)
    static inline NSString * AFKeyPathFromOperationState(AFOperationState state) {
        switch (state) {
            case AFOperationReadyState:
                return @"isReady";
            case AFOperationExecutingState:
                return @"isExecuting";
            case AFOperationFinishedState:
                return @"isFinished";
            case AFOperationPausedState:
                return @"isPaused";
            default:
                return @"state";
        }
    }
    

    3、由于inline增加体积的特性,所以建议inline函数内的代码应该很短小。
    4、inline仅作为一种“请求”,特定情况下编译器将不会理会inline关键字,而强制让函数称为普通函数,这样编译器会输出警告信息。
    5、在盗用一个内联函数之前,一定要在函数之前定义或声明inline,如果在前面声明为普通函数,而在调用代码后面才定义为一个inline函数,程序可以通过编译,但该函数没有实现inline。


    • 内联函数相较于函数的优点
      1、inline函数避免了普通函数的,在汇编时必须调用call的缺点:取消了函数的参数压栈,减少了调用的开销,提高效率.所以执行速度确比一般函数的执行速度要快.
      2、集成了宏的优点,使用时直接用代码替换(像宏一样);

    • 内联函数相较于宏的优点
      1、避免的宏的缺点,宏需要编译,inline不需要编译,内联函数也是函数,不需要编译。
      2、编译器在调用一个内联函数时,会首先检查它参数的类型,保证调用的正确。然后进行一系列的相关检查,就像对待一个真正的函数一样。这就消除的了隐患和局限性。
      3、可以使用所在类的保护成员以及私有成员。

    内联函数参考资料1
    内联函数参考资料2

    参考资料:http://www.cnblogs.com/machao/p/5768253.html

    相关文章

      网友评论

          本文标题:AFNetWorking涉及的知识点

          本文链接:https://www.haomeiwen.com/subject/czlcxftx.html