iOS

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

2016-02-14  本文已影响116人  dibadalu

前言

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

Demo

Pro_Objective-C_Block_Demo
提取码: z5q6

第15章.块

问题1 块类型变量

Snip20160214_1.png

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

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 块是闭包

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

词汇范围

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

{
  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存储类型修改符 ***

{
  __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; 

注意

本章要点

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

上一篇 下一篇

猜你喜欢

热点阅读