《精通Objective-C》 块 阅读笔记

前言

《精通Objective-C》第15章块的知识点略概况性的,基本没有深入讲解块的实现原理。想深入了解块还需要去看Objective-C高级编程的第2章 Blcoks。

Demo

Pro_Objective-C_Block_Demo
提取码: z5q6

第15章.块

问题1 块类型变量

Snip20160214_1.png
  • 返回值类型是指当块被调用时返回值的类型。
  • 块变量名带有一个脱字符(^)前缀并且被封装在括号中。
  • 参数类型列表被封装在括号中并且位于块名称的后面,其中的参数类型以逗号分隔。
  • 如果块变量声明中没有参数,那么应该将该变量的参数类型列表设置为void(封装在括号中)。
  • 没有返回值的块变量的声明中将其返回值类型设置为void。
  • 块变量可以被用作函数和方法的参数。在这类情况下,通过创建类型定义(通过typedef语句)为块类型提供别名,从而起到简化块类型变量名称的作用。

*** 方法声明中的块变量 ***

typedef int (^AdderBlock)(int);
@interface Calulator : NSObject
- (int)process:(int)count withBlock:(AdderBlock)adder;

问题2 块类型常量

Snip20160214_2.png
  • 使用块常量表达式可以定义对块的引用。该表达式以^开头,后跟返回值类型、参数列表和封装在花括号中的语句集合(块的主体)。

*** 块常量表达式 ***

^int (int addend)
{
  return addend + 1;
}

*** 没有设置返回值类型的块常量表达式 ***

^(int addend)
{
  return addend + 1;
}

*** 不带参数的块常量定义 ***

^{
  NSLog(@"Hello, World!");
}

问题3 块的声明和定义

可以将块常量赋予相应的块变量定义。
*** 块的声明和定义 ***

int (^oneParamBlock)(int);
oneParamBlock = ^(int param1)
{
  return param1 * param1;
}
// 定义了一个名为oneParamBlock的块变量,并将一个块常量赋予了它。

*** 将块声明和定义组合在一起 ***

int (^incBlock) (int) = ^(int addend)
{
  return addend + 1;
}
// 块常量表达式的参数类型和返回值类型必须符合相应的块变量定义。

问题4 块语法元素的比较

块语法元素 块变量声明 块常量表达式
^ 标识一个块变量声明的开始,位于变量名称之前。 标识一个块常量表达式的开始。
名称 块变量的名称是必选项 块常量表达式没有名称
返回值类型 在块变量声明中返回值类型是必选项。没有返回值的块变量会将返回值类型设置为void。 在块常量表达式中返回值类型是可选项。
参数 在块变量声明中,参数类型列表是必选项。如果块变量没有参数,必须将参数类型列表声明为void。 在块常量表达式中,参数列表是可选项。

问题5 块常量表达式的定义和调用

可以使用一条语句定义并调用块表达式。注意,块表达式是匿名的,因此调用操作符会接收带有符合“块表达式的参数列表”的匿名(即不带名称的)函数。

*** 定义和调用块表达式 ***

^(NSString *user)
{
  NSLog(@"Greetings, %@!", user);
}(@"Earthling");

问题6 将块常量表达式用作调用方法的参数

还可以像处理匿名函数一样,以内嵌方式定义块表达式,从而将其用作函数或方法的参数。
*** 将块常量表达式用作调用方法的参数 ***

Calculator calc = [Calculator new];
int value = [calc process:2 withBlcok:^(int addend){
                                        return addend + 1;
                                      }];

问题7 块是闭包

块是一个闭包,一个允许访问在其常规范围外部声明的局部变量的函数。块参数通过以下特性提高了变量的可见性:

  • 对词汇范围的支持。
  • __block关键字可以应用于块外部但仍处于同一词汇范围的变量。通过该特性可以修改块内部的变量。
  • 访问实例变量。在对象的方法实现代码中定义的块可以访问该对象中的实例变量。
  • 导入参数。通过头文件导入的常数变量在块常量表达式中是可见的。

词汇范围

块常量表达式可以访问在同一词汇范围内声明的变量。
*** 使用块通过词汇范围访问局部变量 ***

{
  int myVar = 10;
  void (^logValueBlock)(void) = ^{
    NSLog(@"Variable value = %d", myVar);
  };
  logValueBlock();
}
// 局部变量myVar是在定义logValueBlock块的范围内被声明的,而且其声明位于块常量表达式之前,因而可以在这个块常量表达式中使用。

*** 因局部变量的声明在块常量表达式之后,块对局部变量的访问是非法的 ***

{
  void (^logValueBlock)(void) = ^{
    // 错误
    NSLog(@"Variable value = %d", myVar);
  };
  int myVar = 10;
  logValueBlock();
}
// 局部变量myVar是在块常量表达式之后声明和初始化的,因此无法编译。

*** 尝试修改在块常量表达式中通过词汇范围访问的原始数据类型 ***

{
  int myVar = 10;
  void (^logValueBlock)(void) = ^{
    // 错误,词汇范围变量无法重现赋值
    myVar = 5;
    NSLog(@"Variable value = %d", myVar);
  };
  logValueBlock();
}

*** 块可以捕捉多重(嵌套)范围内的局部变量 ***

for(int ii = 0; ii < 2; ii++)
{
  for(int jj = 0; jj < 3; jj++)
  {
    void (^logValueBlock)(void) = ^{
      NSLog(@"Varible values = %d, %d", ii, jj);
    };
    logValueBlock();
  }
}

可修改的__block变量

  • 使用存储类型修改法__block可以将块常量表达式使用的局部变量切换为读写模式。
  • 除了C语言的变长数组和含有变长数组的C语言结构之外,可以对Objective-C支持的所有数据类型使用__block修改符。
  • __block修改符不能与局部存储修改符auto、register和static组合使用。

*** 正确使用__block存储类型修改符 ***

{
  __block int myVar = 10;
  void (^intBlock)(int) = ^(int amout){
    myVar += amout;
    NSLog(@"New value = %d", myVar);
  };
  intBlock(5);
}
// 当引用变量的块被复制到堆存储区域时,使用__block修改符的变量也会被复制到堆存储区域。

块的内存管理

在运行程序时,块常量表达式会获得栈内存,因而会拥有与局部变量相同的生命周期。因此,它们必须被复制到永久存储区域(堆)中,才能在定义它们的范围之外使用。

*** 在块常量的词汇范围之外使用它 ***

void (^greetingBlcok)(void);
{ // 范围的起点,将块变量(局部变量)压入栈中
  greetingBlcok = ^{
    NSLog(@"Hello, World!");
  };
} // 范围的终点,从栈中弹出块变量
greetingBlcok(); // 调用这个块可能会使程序崩溃!

*** 使用块的复制和释放方法 ***

void (^greetingBlcok)(void);
{ 
  greetingBlcok = [^{
    NSLog(@"Hello, World!");
  } copy]; // 将块常量表达式复制到堆中
} 
greetingBlcok(); // 块调用语句起了作用(使用堆内存)
[greetingBlcok release]; // 释放块,防止内存泄露

*** 对使用id类型参数的块调用Block_copy()和Blcok_release()函数 ***

void (^greetingBlcok)(id salutation);
{ 
  greetingBlcok = Block_copy(^(id salutation){
                              NSLog(@"%@, world!", salutation);
                             });
} 
greetingBlock(@"Hello"); 
Block_release; 

注意

  • 使用Blcok_copy()命令可以将块常量复制到堆中(这就像实现了一个将块常量引用作为输入参数并返回相同类型块常量的函数)。
  • 使用Block_release()命令可以释放堆中的块常量(这就像实现了一个将块常量引用作为输入参数的函数。只有当堆中的块常量引用与块常量对应时,才能从堆中释放块常量引用;否则,该函数调用不会有效果)。
  • Blcok_copy()命令必须与Block_release()命令搭配使用,以防内存泄露。
  • Foundation框架提供了处理块的copy和release方法。
  • 在ARC内存管理方式时,只要块没有返回id类型值或将id类型值用作参数,编译器就会自动执行块的复制和释放操作。否则,就必须手动执行复制和释放操作。
  • 在使用MRR时,如果在块常量中使用__block变量,那么这些__block变量就不会被保留。 在使用ARR时,如果在块常量中使用__block变量,那么这些__block变量就会被保留。这意味着如果在使用ARC时,不想保留__block变量(如避免循环引用),还应对变量使用__weak存储类型修改符。

本章要点

介绍了块的语法、块的内存管理、使用块编写程序的方式和如何使用现有API中的块,要点有:

  • 块与函数和方法类似,但块除了是可执行代码,还含有与栈内存和堆内存绑定的变量。块就是一个实现的闭包,一个允许访问其常规范围之外变量的函数。
  • 可以将块声明为块类型的变量,在可以使用普通变量的地方就可以使用块变量,如作为函数/方法的参数等。
  • 块常量表达式含有调用块时执行的代码。可以通过内嵌方式定义块常量表达式(像定义匿名函数一样),然后将块用作函数/方法调用的参数。
  • 与C语言函数相比,通过词汇范围、__block变量、实例变量访问和常数导入,块参数具有更好的可见性。
  • 在运行程序时,块常量表达式会获得栈内存,因而会拥有与局部变量一样的生命周期。因此,必须将之复制到永久存储区域(即堆内存),才能在定义它们的范围之外使用它们。
  • 在使用MRR时,使用Block_copy()和Blcok_release()命令可以复制和释放堆内存中的块。
  • 在使用ARC时,如果块没有返回id类型的值或者使用id类型的参数,编译器就能自动执行复制和释放块的操作。
  • 通常,块用于实现小型的、用于封装任务单元的独立代码段。它们通常以并行方式执行集合中的多个项目,或者被用作完成操作后的回调函数。因为可以通过内嵌方式定义块,所以不用为上下文关联的代码(如异步完成处理程序)创建完全独立的类和方法。
  • 块还可以使有关联的代码被放在一起,而不会将这些代码分割存储到不同文件中。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,657评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,662评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,143评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,732评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,837评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,036评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,126评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,868评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,315评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,641评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,773评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,859评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,584评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,676评论 2 351

推荐阅读更多精彩内容

  • 前言 把《C++ Primer》[https://book.douban.com/subject/25708312...
    尤汐Yogy阅读 9,513评论 1 51
  • 2.1 Blcoks概要 2.1.1 什么是Blocks Blocks是C语言的扩充功能——“带有自动变量(即局部...
    SkyMing一C阅读 2,326评论 6 18
  • 禅与 Objective-C 编程艺术 (Zen and the Art of the Objective-C C...
    GrayLand阅读 1,608评论 1 10
  • 个人战略规划研讨会总结 (一) 有目标和没目标有很大的区别。 1. 带着明确的目标,想要解决什么问题。 2. 学会...
    游魔阅读 2,607评论 1 7
  • 一个学期结束了,学生们的学习生活也告了一个段落,作为家长,帮孩子一起回望,哦,收获丰富啊!数学,语文,英语,...
    柿意阅读 264评论 0 1