iOS开发-由浅至深学习block

关于block

在iOS4.0以后,block横空出世,它自己封装了一如既往段代码并拿即时段代码当做变量,通过block()的道进行回调。这难免让咱想到以C函数中,我们得以定义一个对函数的指针并且调用:

bool executeSomeTask(void) {
    //do something and return if success or not
}
bool (*taskPoint)(void);
taskPoint = executeSomeTask;

方的函数指针可以直接通过(*taskPoint)()的方式调用executeSomeTask本条函数,这样对待block与似乎C语言的函数指针是平的,但是双方仍存在以下分别:

  • block的代码是内联的,效率超过函数调用
  • block对于外部变量默认是不过读属性
  • block被Objective-C看成是目标处理

于block的脚实现以网上曾发广大资料了,其源码更是可以在opensource.apple.com上下载,因此,本文更着重为对于block的行使

今年底看计划是圈24遵照以上,不闹意料又超额完成,并且差一遵循翻倍。与上年对待,多15照。

block特性

  • 认识block
    先由一个简约的要求来说:传入两独数,并且计算这片个数的及,为者创建了这么一个block:

    int (^sumOfNumbers)(int a, int b) = ^(int a, int b) {
        return a + b;
    };
    

立段代码等号左侧声明一个叫做也sumOfNumbers的代码块,名称前用^标志代表后面的字符串是block的名目。最左边的int意味着是block的归来值,括号中表示这个block的参数列表,这里接受两只int品种的参数。
而于当号右侧表示是block的概念,其中返回值是好大概的,编译器会根据上下文自动补充返回值类型。使用^记衔接着一个参数列表,使用括号包起来,告诉编译器这是一个block,然后下大括如泣如诉以block的代码封装起来。

block代码结构

  • 抓获外界变量
    block还好拜外界的片段变量,在自家之于UIView动画说从遭逢有如此一段落代码,其中block内部以及了标的组成部分变量:

    CGPoint center = cell.center;
    CGPoint startCenter = center;
    startCenter.y += LXD_SCREEN_HEIGHT;
    cell.center = startCenter;
    
    [UIView animateWithDuration: 0.5 delay: 0.35 * indexPath.item usingSpringWithDamping: 0.6 initialSpringVelocity: 0 options: UIViewAnimationOptionCurveLinear animations: ^{
        cell.center = center;
    } completion: ^(BOOL finished) {
        NSLog("animation %@ finished", finished? @"is": @"isn't");
    }];
    

    及时其中就是因此到了void(^animations)(void)void(^completion)(BOOL finished)鲜独block,系统会以动画开始跟动画结束的下分别调用者两只block。在贯彻动画的block内部,代码访问了高达文中的center属性——在动画开始之早晚这动画函数的生命周期早已结束,而block会捕获代码外之部分变量,当然就不过局限为才念操作。如果我们于block中修改外部变量,编译器将会报错:

    block中改外有变量

    对想于block中改的外界有对象,我们得为这些变量加上__block重在字修饰,这样就能于block中修改这些变量。在破获变量特性中,还有一个诙谐之微机制,我们把上面的代码改化这么:

    CGPoint center = CGPointZero;
    CGPoint (^pointAddHandler)(CGPoint addPoint) = ^(CGPoint addPoint) {
        return CGPointMake(center.x + addPoint.x, center.y + addPoint.y);
    }
    center = CGPointMake(100, 100);
    NSLog(@"%@", pointAddHandler(CGPointMake(10, 10)));    //输出{10,10}
    

    block在破获变量的时刻偏偏见面保留变量被抓走时之状态(对象变量除外),之后便变量再次转移,block中之价为未见面时有发生变更。所以地方的代码在盘算新的坐标值时center的价依旧等CGPointZero

  • 循环引用
    发端说了,block以iOS开发被给作为是目标,因此该生命周期会一直相当及持有者的生命周期结束了才会收。另一方面,由于block捕获变量的编制,使得所有block的目标也说不定被block持有,从而形成巡回引用,导致二者都不能够让保释:

    @implementation LXDObject
    {
       void (^_cycleReferenceBlock)(void);
    }
    
    - (void)viewDidLoad
    {
        [super viewDidLoad];
        _cycleReferenceBlock = ^{ 
            NSLog(@"%@", self);   //引发循环引用
        };
    }
    
    @end
    

逢这种代码编译器只会告诉你在警告,很多早晚咱们都是忽视警告的,这最后见面造成内存泄露,两者都没法儿自由。跟一般变量是__block着重字一样的,系统提供于咱__weak的重要字用来修饰对象变量,声明这是一个闭眼引用的目标,从而化解了巡回引用的问题:

  __weak typeof(*&self) weakSelf = self;
  _cycleReferenceBlock = ^{ 
      NSLog(@"%@", weakSelf);   //弱指针引用,不会造成循环引用
  };

对于block这种有趣之风味,在唐巧的谈Objective-C
block的实现发详实介绍block的平底实现代码,我以此间就不多说了

扣押开时集中在上半年,下半年忙实习、编写软件以及到毕业前的打,阅读量减少。

使用block

每当block出现之前,开发者实现回调基本都是通过代理的章程进行的。比如当网络要的原来生类NSURLConnection类,通过多个商量方式实现请求被的事件处理。而在时的条件下,使用的NSURLSession既采取block的方法处理任务要了。各种第三着网络要框架为还以用block进行回调处理。这种变动很老一些缘由在于block使用简易,逻辑清晰,灵活等因。接下来我会完成同样不行网络要,然后通过block进行回调处理。这些回调包括要完成、下充斥进度

按照returnValue(^blockName)(parameters)的道进行block的宣示不休麻烦了来,我们可以通过主要字typedef来啊block起类型名称,然后直接通过项目名展开block的创导:

@interface LXDDownloadManager: NSObject< NSURLSessionDownloadDelegate >

//block重命名
typedef void(^LXDDownloadHandler)(NSData * receiveData, NSError * error);
typedef void(^LXDDownloadProgressHandler)(CGFloat progress);

- (void)downloadWithURL: (NSString *)URL parameters: (NSDictionary *)parameters handler: (LXDDownloadHandler)handler progress: (LXDDownloadProgressHandler)progress;

@end

@implementation LXDDownloadManager
{
    LXDDownloadProgressHandler _progress;
}

- (void)downloadWithURL: (NSString *)URL parameters: (NSDictionary *)parameters handler: (LXDDownloadHandler)handler progress: (LXDDownloadProgressHandler)progress
{
    //创建请求对象
    NSURLRequest * request = [self postRequestWithURL: URL params: parameters]; 
    NSURLSession * session = [NSURLSession sharedSession];

    //执行请求任务
    NSURLSessionDataTask * task = [session dataTaskWithRequest: request completionHandler: ^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
        if (handler) {
            dispatch_async(dispatch_get_main_queue(), ^{
                handler(data, error);
            }); 
        }
    }];
    [task resume];
}

//进度协议方法
- (void)URLSession:(NSURLSession *)session
     downloadTask:(NSURLSessionDownloadTask *)downloadTask 
    didWriteData:(int64_t)bytesWritten // 每次写入的data字节数  
   totalBytesWritten:(int64_t)totalBytesWritten // 当前一共写入的data字节数  
  totalBytesExpectedToWrite:(int64_t)totalBytesExpectedToWrite // 期望收到的所有data字节数  
{   
    double downloadProgress = totalBytesWritten / (double)totalBytesExpectedToWrite;  
    if (_progress) { _progress(downloadProgress); }
}  

@end

点通过包NSURLSession的恳求,传入一个处理要结果的block对象,就见面自动将呼吁任务放到工作线程中推行落实,我们于网络要逻辑的代码中调用如下:

#define QQMUSICURL @"https://www.baidu.com/link?url=UTiLwaXdh_-UZG31tkXPU62Jtsg2mSbZgSPSR3ME3YwOBSe97Hw6U6DNceQ2Ln1vXnb2krx0ezIuziBIuL4fWNi3dZ02t2NdN6946XwN0-a&wd=&eqid=ce6864b50004af120000000656fe235f"
[[LXDDownloadManager alloc] downloadWithURL: QQMUSICURL parameters: nil handler ^(NSData * receiveData, NSError * error) {
    if (error) { NSLog(@"下载失败:%@", error) }
    else {
        //处理下载数据
    }
} progress: ^(CGFloat progress) {
    NSLog(@"下载进度%lu%%", progress*100);
}];

在押的最多的凡日本作家村及春树的题,图书馆能找到的都已拘留了。

仿swift高阶函数

用过swift的开发者都知swift的函数调用很好的体现了链式编程的思考,即将多个操作通过.连接起来,使得可读性更胜,比如ocString.stringByAppendingFormat("abc").stringByAppendingFormat("edf")即连调整用了多字符串的章程。这种编程方式的条件之一是每次函数调用必须发返值。虽然当用Objective-C开发的进程遭到,方法的调用是经[target action]的不二法门就的,但是block本身的调用方式吗是通过blockName(parameters)的法子实施之,与这种链式函数有异曲同工之精。

以swift中提供了连mapfilterreduce等非常简短优秀的高阶函数供我们对数组数据开展操作,同样状况下,遍历一个数组并求和以用oc(不采取kvc)和swift的环境下的代码是这么的:

#pragma mark - OC code
NSArray numbers = @[@10, @15, @99, @66, @25];
NSInteger totalNumber = 0;
for (NSNumber number in numbers) {
    totalNumber += number.integerValue;
}

#pragma mark - swift code
let numbers = [10, 15, 99, 66, 25];
let totalNumber = numbers.reduce(0, { $0+$1 })

无论是代码量还是简洁性,此时底oc都小swift。那么连下去就要通过神奇之block来啊oc添加这些高阶函数的贯彻。为者我们得新建一个NSArray的分类扩展,命名吧NSArray+LXDExtension

#import <UIkit/UIKit.h>

/// 数组元素转换
typedef id(^LXDItemMap)(id item);
typedef NSArray *(^LXDArrayMap)(LXDItemMap itemMap);

/// 数组元素筛选
typedef BOOL(^LXDItemFilter)(id item);
typedef NSArray *(^LXDArrayFilter)(LXDItemFilter itemFilter);

/**
 *  扩展数组高级方法仿swift调用
 */
@interface NSArray (LXDExtension)

@property (nonatomic, copy, readonly) LXDArrayMap map;
@property (nonatomic, copy, readonly) LXDArrayFilter filter;

@end

面前说了为实现链式编程,函数调用的前提是具有返回对象。因此自用了typedef宣示了几只例外类别的block。虽然本质上LXDArrayMapLXDArrayFilter些微单block是一样的,但是以区别它们的效力,还是建议这样做。其落实文件如下:

typedef void(^LXDEnumerateHandler)(id item);

@implementation NSArray (LXDTopMethod)

- (LXDArrayMap)map
{
    LXDArrayMap map = ^id(LXDItemMap itemMap) {
        NSMutableArray * items = @[].mutableCopy;
        for (id item in self) {
            [items addObject: itemMap(item)];
        }
        return items;
    };
    return map;
}    

- (LXDArrayFilter)filter
{
    LXDArrayFilter filter = ^BOOL(LXDItemFilter itemFilter) {
        NSMutableArray * items = @[].mutableCopy;
        for (id item in self) {
            if (itemFilter(item)) { [items addObject: item]; }
        }
        return items;
    };
    return filter;
}

- (void)setFilter:(LXDArrayFilter)filter {}
- (void)setMap:(LXDArrayMap)map {}

@end

我们通过再写setter方法保证block不会见于外表修改实现,并且以getter中遍历数组的素并调用传入的执行代码来促成mapfilter相当于效果。对于当下片独功能的贯彻啊很简短,下面举出点儿单调用高阶函数的例证:

#pragma mark - 筛选数组中大于20的数值并转换成字符串
NSArray<NSNumber *> * numbers = @[@10, @15, @99, @66, @25, @28.1, @7.5, @11.2, @66.2];
NSArray * result = numbers.filter(^BOOL(NSNumber * item) {
    return item.doubleValue > 20
}).map(^id(NSNumber * item) {
    return [NSString stringWithFormat: @"string %g", item.doubleValue];
});

#pragma mark - 将数组中的字典转换成对应的数据模型
NSArray<NSDictionary *> * jsons = @[@{ ... }, @{ ... }, @{ ... }];
NSArray<LXDModel *> * models = jsons.map(^id(id item) {
    return [[LXDModel alloc] initWithJSON: item];
})

由于语法上的限,虽然如此的调用跟swift原生的调用对比起来或复杂了,但透过block让oc实现了函数链式调用的代码看起也舒畅了无数

以下,为同一年来拘禁罢的书目(按看开时先后顺序排列):

总结

block捕获变量、代码传递、代码内联等特性与了它们多于代理体制的效能及灵活性,尽管它也设有循环引用、不易调试追溯等毛病,但正确它的长处于码农们的慈。如何更灵活的下block需要我们针对她不止的采取、探究了解才能够不辱使命
文集:iOS开发

转载请注明原文地址和作者

1、《槽边往事:和菜头自选集》和菜头
老爸来学看自己,吃了饭逛新华书店为我请的。

2、《身边的凡》野夫
良有古诗的字。

3、《我之专职是小说家》村达到春树
干货满满啊。村达到之新书中文译本明年要上市了咔嚓?

4、《火星救援》安迪·威尔
同名电影呢尴尬。

5、《人类群星闪耀时》斯蒂芬·茨维格
无独有偶于达标初中的表弟看之开。颠覆了我对新高中必读名著的记忆。五星推荐!

6、《Objective-C 程序设计:第4版》斯蒂芬·G·科昌
老三单月才看了。前12章还算是简单,后几乎节就云里雾里了。

7、《费正清中国回忆录》费正清
历史书及之大事件,在笔者亲历过后的回忆里,也不怕是那回事了。没有特别酷之惊,一切都是那么自然而然。部分章节涉及具体的中国的总人口与物可细读,部分章节可有些读。

8、《粉墨春秋:盖叫天舞台艺术经验》盖叫天
无非拘留了「艺术人生」这等同章,看时脑子里闪转在电影《霸王别姬》里的画面。吃得辛苦着苦,方为人上人。台上一分钟,台下十年功夫。

9、 《台北人》白先勇

10、《图解HTTP》上野宣

11、《地下》村达到春树

12、《在约定的场所:地下2》村达到春树
切切实实包含了合不如意;理想化的组织现实,说不定现实会反噬你。

13、《图书馆奇谈》村达到春树
离奇之故事。

14、《村达到朝日堂的卷土重来》村达到春树
有趣之随笔。

15、《旋转木马鏖战记》村及春树

16、《30龙自制操作系统》川合秀实
Mac不能够就做,挑了汇编语言及c语言指针部分的情节看。作者写的有趣幽默,不错,推荐。

17、《编码:隐匿在计算机软硬件背后的言语》Charles Petzold
离散数学、电子技术与数字电路、计算机组成原理、数据结构等,计算机有关的事物面面俱到啊,真该早放贷来探视的。5星推荐。

18、《神之男女都跳跳舞》村达到春树

19、《列克星敦的鬼魂》村及春树

20、《悉尼!》村及春树
村达到春树亲临2000年悉尼奥运会的日志合集,很有趣。

21、《图解TCP/IP:第5本》竹下隆史等
纵深不够,当科普书看。

22、《浪潮的巅》吴军
念了及时按照开后,看IT新闻还能够领悟那道理了。强烈推荐!!!另:这应该算是吴军博士太好的同等本书了,最近几年之书还有硌炒冷饭之头痛。

23、《如何阅读一本书》莫提默·J等

24、《树上的男》卡尔维诺
平生活于树上是种啊感受?看这本开便明白了。

25、《七堂极简物理课》卡洛·罗韦利
真正很简短。。。

26、《神祗、陵墓和专家:考古学传奇》C.W.策拉姆
好看呐!

27、《月亮与六便士》毛姆

28、《人工智能狂潮:机器人会跨人类呢?》松尾丰
同一按照小小的科普书。现在处于第三次等人工智能浪潮,「深度上」开创新时代啊。

29、《全程软件测试》朱少民
良详细的同样本书,面面俱到。

30、《Linux系统命令及Shell脚本实践指南》王军
修Linux必备。很好老好老好~

31、《大话移动测试:Android与iOS应用测试指南》陈晔
圈了晚,加了笔者微信,进了作者曾经工作过之公司。哈哈哈~

32、《阿城精选集》阿城
马拉松闻大名的藏的作,在上下班的地铁达到吹吹拍拍在kindle看了了。

33、《精进:如何变成一个雅厉害的总人口》采铜

34、《悉达多:一篇印度的诗文》赫尔曼·黑塞

35、《第一履代码:Android(第2版)》郭霖
安卓开发的入门好写,强推。

36、《硅谷之谜:《浪潮的奇峰》续集》吴军
关押罢《浪潮的巅》的,建议并非看了。

37、《成为乔布斯》布伦特·施兰德等
很好看。

38、《Android群英传》徐宜生
1、2、8、9、10、11、12,这几节有点多余,什么还写,什么都浅。3、4、5、6、7,这几乎回可多扣几乎所有。最后一章实例,未看。

39、《黑旗:ISIS的崛起》乔比·沃里克

40、《质数的一身》保罗·乔尔达诺
书籍的页码按照质数排序的。

41、《你一定爱读之极简欧洲史:为什么欧洲对现代文明的震慑这么好》约翰·赫斯特

42、《思维的意趣》王小波
理智最得意。

43、《Android开发方式探索》任玉刚
经过源码讲解常用知识点,写的好哎!

44、《疯狂Java讲义:第3版》李刚
大而全,值得一看。

45、《玫瑰的名字》翁贝托·埃科

46、《大数量时代:生活、工作及思考的酷变革》维克托·迈尔·舍恩伯格
多少、技术、思维。分析数据的艺会如编程一样越普遍,而想是特种的。当然,最要的要么多少我。

47、《程序员的数学》

发表评论

电子邮件地址不会被公开。 必填项已用*标注

网站地图xml地图