iOS开发-block

*block的代码是内联的,效率高于函数调用

block对于外部变量默认是只读属性

block被Objective-C看成是对象处理


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中提供了包括map、filter、reduce等十分简洁优秀的高阶函数供我们对数组数据进行操作,同样情况下,遍历一个数组并求和在使用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

/// 数组元素转换

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。虽然本质上LXDArrayMap和LXDArrayFilter两个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中遍历数组的元素并调用传入的执行代码来实现map和filter等功能。对于这两个功能的实现也很简单,下面举出两个调用高阶函数的例子:

#pragma mark - 筛选数组中大于20的数值并转换成字符串

NSArray * 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 * jsons = @[@{ ... }, @{ ... }, @{ ... }];

NSArray * models = jsons.map(^id(id item) {

return [[LXDModel alloc] initWithJSON: item];

})

由于语法上的限制,虽然这样的调用跟swift原生的调用对比起来还是复杂了,但通过block让oc实现了函数链式调用的代码看起来也清爽了很多

总结

block捕获变量、代码传递、代码内联等特性赋予了它多于代理机制的功能和灵活性,尽管它也存在循环引用、不易调试追溯等缺陷,但无可置疑它的优点深受码农们的喜爱。如何更加灵活的使用block需要我们对它不断的使用、探究了解才能完成

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

推荐阅读更多精彩内容

  • 本文分两部分介绍Block: 第一部分:Block基础知识介绍第二部分:Block经常使用的三种情况(方法回调,C...
    小小土豆dev阅读 801评论 2 9
  • 博客地址 Blocks概要 什么是Blocks Blocks是C语言的扩充功能。可以用一句话来表示Blocks的扩...
    换个名字再说阅读 389评论 0 2
  • 一、基本概念: Block是一种C语言的数据类型,指向结构体的指针,平常我们将Block当作一个代码段使用,相当于...
    常绿箩阅读 509评论 0 2
  • 1、block跟swift中的闭包(closure)基本一样,都常用于值的回调,特别是在多线程的网络请求回调中,使...
    kkj1996阅读 285评论 0 0
  • 首先来了解下什么是Block (1)Block是OC中的一种数据类型,在iOS开发中被广泛使用(2)^是Block...
    爱吃鱼的小灰阅读 2,342评论 2 7