浅析 iOS 中 Block 的用法

Block 代码块

闭包 = 一个函数「或指向函数的指针」+ 该函数执行的外部的 上下文变量「也就是自由变量」
Block 是 Objective-C 对于闭包的实现

简介

  • 可以嵌套定义,定义 Block 方法和定义函数方法相似
  • Block 可以定义在方法内部或外部
  • 只有调用 Block 时候,才会执行其{}体内的代码
  • 本质是对象,使代码高聚合

使用 clang 将 OC 代码转换为 C++ 文件查看 block 的方法

  • 在命令行输入以下代码 clang -rewrite-objc 需要编译的OC文件.m
  • 这时查看当前的文件夹里 多了一个相同的名称的 .cpp 文件,在命令行输入 open main.cpp 查看文件

I. 定义格式

  • 立刻执行的 block
    ^{ /*执行的代码*/ }();

  • 默认格式
    返回值类型 (^Block变量名)(形参列表) = ^返回值类型 (形参列表){ 内容 }
    返回值类型通常省略,根据 内容中的 return 返回值的类型来决定最终类型,如果没有 return 语句,则返回值类型默认为 void

  • 有参
    返回值类型 (^Block变量名)(形参列表) = ^(形参列表){ 内容 };

  • 无参
    返回值类型 (^Block变量名)(形参列表) = ^(){ 内容 };
    返回值类型 (^Block变量名)(形参列表) = ^{ 内容 };

II. 使用方法

  • 作为变量的方法「Xcode快捷键:inlineBlock
int (^sum) (int, int); // 定义一个 Block 变量 sum
// 给 Block 变量赋值
// 一般 返回值省略:sum = ^(int a,int b)…
sum = ^int (int a,int b){  
    return a+b;
}; // 赋值语句最后有 分号
int a = sum(10,20); // 调用 Block 变量
  • 作为属性的方法「Xcode 快捷键:typedefBlock
// 1. 给  Calculate 类型 sum变量 赋值「下定义」
typedef int (^Calculate)(int, int); // calculate就是类型名
Calculate sum = ^(int a,int b){ 
    return a+b;
};
int a = sum(10,20); // 调用 sum变量

// 2. 作为对象的属性声明,copy 后 block 会转移到堆中和对象一起
@property (nonatomic, copy) Calculate sum;    // 使用   typedef
@property (nonatomic, copy) int (^sum)(int, int); // 不使用 typedef

// 声明,类外
self.sum = ^(int a,int b){
    return a+b;
};
// 调用,类内
int a = self.sum(10,20);
  • 作为函数参数
    例:请求网络数据(延迟)先把展示到控件的代码保存到block中,等请求到数据的时候直接调用block
// 实现,不使用 typedef
// void (^iblock)()作为函数参数类型,iblock函数形参名
void iprint( void (^iblock)() ){ 
    iblock(); // 调用 block参数}

// 实现,使用 typedef
typedef void (^IBlock)();
void iprint(IBlock iblock){
    iblock(); // 调用 block参数

}

// 调用「不管使用不使用 typedef 调用方式一致」
iprint(^{
    NSLog(@"传入的实参代码块区域");
});
  • 作为 OC 中的方法参数
// ---- 无参数传递的 Block ---------------------------
// 实现
- (CGFloat)testTimeConsume:(void(^)())middleBlock {
    // 执行前记录下当前的时间
    CFTimeInterval startTime = CACurrentMediaTime();
    middleBlock();
    // 执行后记录下当前的时间
    CFTimeInterval endTime = CACurrentMediaTime();
    return endTime - startTime;

}

// 调用
[self testTimeConsume:^{
       // 放入 block 中的代码 

}];

// ---- 有参数传递的 Block ---------------------------
// 实现
- (CGFloat)testTimeConsume:(void(^)(NSString * name))middleBlock {
    // 执行前记录下当前的时间
    CFTimeInterval startTime = CACurrentMediaTime();
    NSString *name = @"有参数";
    middleBlock(name);
    // 执行后记录下当前的时间
    CFTimeInterval endTime = CACurrentMediaTime();
    return endTime - startTime;
}

// 调用
[self testTimeConsume:^(NSString *name) {
   // 放入 block 中的代码,可以使用参数 name
   // 参数 name 是实现代码中传入的,在调用时只能使用,不能传值    

}];

III. 访问外界变量

  • 不访问外界变量,block 既不在 又不在 中,在代码段中「ARC 和 MRC 下都是如此」
void (^myblock)(void) = ^(){
    printf("不访问 block 作用域范围外的外界变量\n");
};
myblock();  // 在代码段中和 C 函数一样
  • 默认情况下,可以访问,不能修改 外界变量的值
    MRC 环境下访问外界变量的 block 默认存储在
    ARC 环境下访问外界变量的 block 默认存储在 中,自动释放
int a = 4;
char text[] = "fuckBlock";

char *p = text;
NSMutableArray *arr = [NSMutableArray array];

void (^myblock)(void) = ^(){
    [arr addObject:@(a)];    // 能修改对象 arr 内部的值
    NSLog(@"a = %d",a);      // 能访问变量 a   的值
    NSLog(@"arr = %@",arr);  // 能访问对象 arr 的值
    NSLog(@"%c",p[2] = 'F'); // 能访问指着 指向的值,并修改内容
    NSLog(@"%c",text[2]);    // 不能访问 C 语言数组及其内容,这里编译器会报错
    a = 5;                   // 不能修改变量 a   的值,这里编译器会报错
    arr = [NSMutable array]; // 不能修改对象 arr 的值,这里编译器会报错
};

myblock();    // 在调用 block 时,深复制变量 a 和对象 arr 的值到 block 数据结构中
printf("after block a = %d\n",a); // 两次 a 输出都是 4
  • 使用 __block 修改外界变量
    ARC 下:访问外部对象用 __block 修饰:访问对象进行 1 次 retain 操作
    MRC 下:访问外部对象用 __block 修饰:访问对象不会进行 retain 操作
__block int a = 4;
void (^myblock)(void) = ^(){
    // 此时,该 block 对象会持有 __block 的值 a
    // 多个 block 对象会持有 同一个 __block 的值 a
    a = 5;    // 能修改
    printf("%d\n",a); // 复制 a 的引用地址,能访问
};
myblock();    // block 在栈中
printf("%d\n",a); // 两次输出都是 5
  • block 从 复制到 编译器自动复制的情况
    block 调用 copy 方法
    block 作为函数返回值返回
    block 赋值给 __strong 修饰的 id 类型 或 block 类型的成员变量 时
    方法名含有 usingBlock 的 Cocoa 框架方法 或 GCD 的 API 接口传入的 block 时

  • block 被 copy 时,block 被复制到
    block 在 中,copy,从栈复制到
    block 在 代码段 中,copy,什么也不做
    block 在 中,copy,引用计数+1

// 1. 简单示例

[blockName copy];      // block 的 copy 操作
Block_copy(blockName); // block 的 copy 操作「宏定义方法,在 ARC 模式下会出错」

// 2. 编译器在部分情况下会自动将 block 从 栈复制到堆中,但有些情况需要手动复制
- (NSArray *)getBlockArray {
    int val = 10;
    // block 就是对象,所以可以直接加入到数组中,并且有 copy 方法
    return [[NSArray alloc] initWithObjects:
            [^{ NSLog(@"block_0:%d",val); } copy],
            [^{ NSLog(@"block_1:%d",val); } copy], nil];

}

IV. 防止 Block 循环引用的方法

  • Block 循环引用的情况
    某个类将 block 作为自己的属性变量,然后该类在 block 的方法体里面又使用了该类本身
    self.someBlock = ^(Type var){[self dosomething];};
  • 解决方法「ARC 下:使用 __weak」
__weak typeof(self) weakSelf = self;
self.someBlock = ^(Type var){
   [weakSelf dosomething];
};
  • 解决方法「MRC 下:使用 __block」
    优点:可控制对象的持有期间
    缺点:为了避免循环引用,必须执行 block
__block typeof(self) blockSelf = self;
self.someBlock = ^(Type var){
   [blockSelf dosomething];
};
  • 使用 __block 带来的循环引用「ARC」
// 循环引用 self -> _attributBlock -> tmp -> self
typedef void (^Block)();
@interface TestObj : NSObject
{
    Block _attributBlock;
}
@end

@implementation TestObj
- (id)init {
    self = [super init];
    __block id tmp = self;
    self.attributBlock = ^{
        NSLog(@"Self = %@",tmp);
        tmp = nil;
   };
}

- (void)execBlock {
    self.attributBlock();
}
@end

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

推荐阅读更多精彩内容