Blocks读书笔记

2017-04-20  本文已影响35人  动物园园长熊熊酱

——Objective-C高级编程读书笔记 p79-p137

Blocks定义

带有自动变量(局部变量)的匿名函数。在计算机科学中,此概念也称作闭包(Closure)、lambda计算等。

Blocks语法

Block语法的BN范式 : ^ 返回值类型 参数列表 表达式
^int (int count){return count +1;}

1.省略返回值类型时,如果表达式中有return语句就使用该返回值类型,如果表达式中没有return语句就使用void类型。
2.如果不使用参数,参数列表也可以省略。
最简式:^ 表达式
^{printf("Blocks\n");}

Block类型变量

在定义C语言函数时,可以将定义函数的地址赋值给函数指针的类型变量中。同样地,在Block语法下,可将Block语法赋值给声明为Block类型的变量中。即源代码中一旦使用了Block语法就相当于生成了可赋值给Block类型变量的“值”。
声明Block类型变量示例:
int (^blk) (int);

自动变量值截获只能保存执行Block语法瞬间的值。保存后就不能改写该值。使用__block说明符的自动变量可以在Block中赋值,该变量被称为__block变量。

定义Block

这里表示定义一个people类型的Block,无返回值,接收NSMutableArray类型的数组以及布尔型的isAlive。
#typedef void (^people) (NSMutableArray *array,Bool isAlive);
即使不使用以上语句,也可以直接定义为成员变量:
@property (nonatomic, strong) void (^people) (NSMutableArray *array,Bool isAlive);
注意,在MRC下,strong应该改为copy或retain。

Block本身是像对象一样可以retain,和release。但是,block在创建的时候,它的内存是分配在栈(stack)上,而不是在堆(heap)上。他本身的作于域是属于创建时候的作用域,一旦在创建时候的作用域外面调用block将导致程序崩溃。
block要用copy修饰,还是用strong

Block的实现

Block的实质为Objecitve-C的对象。

Block循环引用

来看下面这个代码片段,blk为Block类型成员变量。

- (id)init
{
    self = [super init];
    blk = ^{NSLog(@"self = %@",self);};
    return self;
}

在这段代码中,本类对象的成员变量blk持有赋值为Block的强引用,即本类对象持有Block。而init方法中,执行的Block语法使用附有__strong类修饰符的id类型变量self,并且Block语法赋值给了blk,Block语法生成的Block又栈赋值到堆,并持有所使用的self。self持有Block,Block持有self,这就造成了循环引用。

解决方案一:
声明附有__weak修饰符的变量,并将self赋值使用。

- (id)init
{
    self = [super init];
    id __weak tmp = self;
    blk = ^{NSLog(@"self = %@",tmp);};
    return self;
}

值得注意的是,在面向iOS4或以下的版本,这里可以用__unsafe_unreained修饰符代替__weak。但是他们是有区别的。

对于__weak,当释放指针指向的对象时,该对象的指针将转换为nil,这是比较安全的行为。而__unsafe_unretain,正如其名称隐藏的含义,尽管释放指针指向的对象时,该指针将继续指向原来的内存。这将会导致应用crash,所以是unsafe。
iOS 高级内存管理:比较__unsafe_unretain、__strong、__weak、__autoreleasing

解决方法二:

    __weak typeof(self) weakSelf = self;

Block使用

AController中:

    BController *bVc = [BController new];
    bVc.block = ^(NSMutableDictionary *dict) {
        NSLog(@"Block %@",dict);
    };
    [self.navigationController pushViewController:allStarVc animated:YES];

BController中:

//BController.h
#import <UIKit/UIKit.h>

typedef void(^Block)(NSMutableDictionary *dict);

@interface BController : UIViewController

@property (nonatomic,copy)Block block;

@end

//BController.m
- (void)pop
{
    if (_allstarBlock) {
        allstarBlock(self.dict);
    }
    [self.navigationController popViewControllerAnimated:YES];
}


上一篇 下一篇

猜你喜欢

热点阅读