关于Block

关键字

block一般使用copy关键字进行修饰,block使用copy是从MRC遗留下来的“传统”,在MRC中,方法内容的block是在栈区的,使用copy可以把它放到堆区。但在ARC中写不写都行:编译器自动对block进行了copy操作。

捕获自动变量

int val = 10;  
void (^blk)(void) = ^{printf("val=%d\n",val);};  
val = 2;  
blk();    

上面这段代码,输出值是:val = 10.而不是2.
block 在实现时就会对它引用到的它所在方法中定义的栈变量进行一次只读拷贝,然后在 block 块内使用该只读拷贝;换句话说block截获自动变量的瞬时值;或者block捕获的是自动变量的副本。(我更喜欢这么理解:block捕获变量的机制更像是函数按值传递而非按地址传递)

block函数中不能随意变量值,但是可以修改静态变量、外部变量以及用__block关键字修饰的普通变量。同时,对于这几种情况时,在block函数后修改静态变量、外部变量以及用__block关键字修饰的普通变量时,也会影响block函数内这些变量的值。

NSInteger CounterGlobal = 0; //不使用extern为消除警告。写在函数外默认存在extern。
static NSInteger CounterStatic = 0;

- (void)block4
{
        NSInteger localCounter = 42;
        __block char localCharacter;
        
        void (^aBlock)(void) = ^(void) {
            ++CounterGlobal;
            ++CounterStatic;
            CounterGlobal = localCounter; // localCounter fixed at block creation
            localCharacter = 'a'; // sets localCharacter in enclosing scope
            
            //result:CounterGlobal(42),CounterStatic(5),localCharacter(‘a’)
        };
        CounterStatic = 4;
        ++localCounter; // unseen by the block
        localCharacter = 'b';
        
        aBlock(); // execute the block
    
}

block循环引用的原理

@interface KSViewController ()  
{  
    id _observer;  
}  
  
@end  
  
@implementation KSViewController  
  
- (void)viewDidLoad  
{  
    [super viewDidLoad];  
    // Do any additional setup after loading the view, typically from a nib.  
      
    KSTester * tester = [[KSTester alloc] init];  
    [tester run];  
      
    _observer = [[NSNotificationCenter defaultCenter]  
                 addObserverForName:@"TestNotificationKey"  
                 object:nil queue:nil usingBlock:^(NSNotification *n) {  
                     NSLog(@"%@", self);  
                 }];  
}  
  
- (void)dealloc  
{  
    if (_observer) {  
        [[NSNotificationCenter defaultCenter] removeObserver:_observer];  
    }  
} 

self持有observer,observer持有block,block中持有self。因为block始终强引用self导致self无法释放,进而dealloc方法不会执行,observer不会释放,内存泄漏。

如下图:


循环引用原理.png

使用weak,strong解决

__weak KSViewController * wself = self;  
    _observer = [[NSNotificationCenter defaultCenter]  
                 addObserverForName:@"TestNotificationKey"  
                 object:nil queue:nil usingBlock:^(NSNotification *n) {  
                     KSViewController * sself = wself;  
                     if (sself) {  
                         NSLog(@"%@", sself);  
                     }  
                     else {  
                         NSLog(@"<self> dealloc before we could run this code.");  
                     }  
                 }];  

在 block 之前定义对 self 的一个弱引用 wself,因为是弱引用,所以当 self 被释放时 wself 会变为 nil;然后在 block 中将该弱引用置为强引用,便于Block内后续方法调用。

如下图:

避免循环引用的原理.png

当外部引用中断,self释放,dealloc方法执行,observer释放,进而全部释放。

弱引用与强引用的区别

当还有一个strong(绳子)指向对象(狗)的时候,对象(狗)无法释放(跑掉)。
当没有strong(绳子)指向对象(狗)的时候,无论有多少weak(看着)对象(狗),对象(狗)都会释放(跑掉)。

Block的使用

在函数式编程中,把函数当参数来回传递,而这个,说成术语,我们把他叫做高阶函数。而在oc中,blocks是被广泛使用的参数传递,它实际上是匿名函数
一个简单的带有成功失败的网络请求,简单block作为参数。

- (void)unBindDevice:(NSString *)mac success:(dispatch_block_t)success failure:(HWBusinessFailureBlock)failure
{
    HWAPIV2UnBindDevice *client = [[HWAPIV2UnBindDevice alloc] initWithdeviceId:mac];

    [client startWithCompletionBlockWithSuccess:^(__kindof YTKBaseRequest *_Nonnull request) {
      HWBusinessException *enception = [[HWBusinessException alloc] initWithRequest:request];
      HWLogDeBug(enception.retInfo);
      if (enception.success) {
          [self removeDeviceFromUserDeviceListWithMac:mac];
          if (success) {
              success();
          }
      }
      else if (failure) {

          failure(enception);
      }
    }
        failure:^(__kindof YTKBaseRequest *_Nonnull request) {
          if (failure) {
              failure([HWBusinessException netException]);
          }
        }];
}

rac使用block作为参数的一个经典代码事例。

RACSignal *filteredSsid = [ssidSourceSignal filter:^BOOL(NSString *value) {
    NSString *text = value;
    return text.length > 1;
}];

其中block:^BOOL(NSString *value)作为参数传入。 value是filter函数内部返回,我们用于Block体内使用实现。而bool值作为block的返回值由block体内返回给filter函数内部。而filter内部使用该bool值对数据进行处理返回带筛选结果的RACSignal类型变量。

同样,block也可以在方法中作为闭包进行代码抽象。

typedef void (^successBlock)(NSDictionary * dict);
successBlock block = ^(NSDictionary * dict){
  //统一处理
};

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

推荐阅读更多精彩内容