转载

block使用

iOS中block比较常用,但是又和OC的语法显得有点格格不入,难于理解。

以下是我个人初步的理解,供查阅。

1.block的声明

//声明一个block  typedef NSString *(^WXYTestBlock)(NSString *name, int age);

以上声明了一个名字叫做WXYTestBlock的block,参数为一个字符串类型的name和一个int类型的age,返回值为NSString。

当然,你也可以声明成这样:

typedef void *(^WXYTestBlock)(void);

无参数,无返回值。

当然也可以有参数无返回值,或者有返回值无参数,不一一列举。

2.block的使用

首先是独立block

    //独立block     WXYTestBlock myBlock = ^ (NSString *name, int age){         return [NSString stringWithFormat:@"%@的年龄是%d",name,age];     };     NSLog(@"独立block--->%@", myBlock(@"小宇", 16));

独立block可以直接定义和使用,运行输出如下

2015-06-03 23:32:32.532 WXYBlock[3537:237755] 独立block--->小宇的年龄是16

然后是内联block

//使用内联block的方法 - (void)printWithName:(NSString *)name age:(int)age block:(WXYTestBlock)block{     NSLog(@"内联block--->%@",block(name, age)); }

内联block需要将定义的block作为参数传入相应的方法中,然后在方法中使用block。

//内联block     [self printWithName:@"王兴宇" age:26 block:^(NSString *str, int age){         return [NSString stringWithFormat:@"%@的年龄是%d", str, age];     }];

内联block可以在调用方法的时候写入代码块,运行结果如下

2015-06-03 23:32:32.532 WXYBlock[3537:237755] 内联block--->王兴宇的年龄是26

3.block使用外部变量

//变量的使用 int myAge = 100; //独立block myBlock = ^ (NSString *name, int age){  return [NSString stringWithFormat:@"使用变量--->%@的年龄是%d", name, myAge]; }; NSLog(@"独立block--->%@", myBlock(@"小宇", 16)); //内联block [self printWithName:@"王兴宇" age:26 block:^(NSString *str, int age){  return [NSString stringWithFormat:@"使用变量--->%@的年龄是%d", str, myAge]; }]; 

block内部可以直接使用外部定义的变量,运行结果如下

#注意:此处为了方便,直接用myAge代替了原来的age,所以参数16传进去根本没有使用。

2015-06-03 23:32:32.533 WXYBlock[3537:237755] 独立block--->使用变量--->小宇的年龄是100 2015-06-03 23:32:32.533 WXYBlock[3537:237755] 内联block--->使用变量--->王兴宇的年龄是100

一个有趣的现象:

现在你定义了一个独立block,并且这个block使用了外部的变量。

然后这个变量被改变了,然后你调用了这个block。

注意,是这样的顺序:

定义独立block并且使用外部变量---->外部变量改变---->调用block

//外部改变变量     myAge = 50;     NSLog(@"独立block--->变量在外部被改变--->%@", myBlock(@"小宇", 16));          [self printWithName:@"王兴宇" age:26 block:^(NSString *str, int age){         return [NSString stringWithFormat:@"变量在外部被改变--->使用变量--->%@的年龄是%d", str, myAge];     }];

这时候的输出是

2015-06-03 23:32:32.533 WXYBlock[3537:237755] 独立block--->变量在外部被改变--->使用变量--->小宇的年龄是100 2015-06-03 23:32:32.533 WXYBlock[3537:237755] 内联block--->变量在外部被改变--->使用变量--->王兴宇的年龄是50

这是为什么呢?根据查阅,我总结的原因是这样的:

block中如果使用了外部变量,他会拷贝一份这个变量,并且这个变量是只读的。

所以外部变量改变并不影响block内部拷贝的那一份变量。

代码中的内联block是在变量改变后才使用这个变量的,所以并不影响。

如果不想让block拷贝变量,想让内部使用的变量和外部使用的变量指向同一地址的话,

需要在变量前面加上__block关键字。

像这样:

__block int myAge = 100;

输出就变成了:

2015-06-03 23:32:32.533 WXYBlock[3537:237755] 独立block--->变量在外部被改变--->使用变量--->小宇的年龄是50 2015-06-03 23:32:32.533 WXYBlock[3537:237755] 内联block--->变量在外部被改变--->使用变量--->王兴宇的年龄是50

另外值得一提的是,加上__block关键字之后,外部变量不再是只读的,在block内部也可以改变它的值。

//改变变量加__block关键字 __block int otherAge = 100; myBlock = ^ (NSString *name, int age){  otherAge = 99;  return [NSString stringWithFormat:@"改变变量加__block--->%@的年龄是%d", name, otherAge]; }; NSLog(@"独立block--->%@", myBlock(@"小宇", 16)); [self printWithName:@"王兴宇" age:26 block:^(NSString *str, int age){  otherAge = 98;  return [NSString stringWithFormat:@"改变变量加__block--->%@的年龄是%d", str, otherAge]; }]; 

输出如下:

2015-06-03 23:32:32.534 WXYBlock[3537:237755] 独立block--->改变变量加__block--->小宇的年龄是99 2015-06-03 23:32:32.534 WXYBlock[3537:237755] 内联block--->改变变量加__block--->王兴宇的年龄是98

#注意:如果不加__block关键字,在block内部改变外部变量的值的话,编译会报错!

4.block循环引用的问题

这部分我的理解可能不太深入,下面只说一下我自己简单的理解。

首先在self类中声明一个NSString的属性

#import <UIKit/UIKit.h>  @interface ViewController : UIViewController  @property (strong, nonatomic) NSString *myStr;   @end

初始化这个属性

self.myStr = @"myStr";

使用内联block的另一个方法,为了方便,还用之前声明的block,只是参数用不到了

- (void)printWithblock:(WXYTestBlock)block{     block(@" ", 0); }

现在,你想要在block中使用self,或者使用self.myStr

如果,self的类中包含block,block中又引用了self

这样就会造成循环引用。

解决的方法如下

//使用self和self的属性 //加__weak避免循环引用 __weak ViewController *weakSelf = self; //独立 myBlock = ^ (NSString *name, int age){  NSLog(@"独立Block使用self--->%@", weakSelf);  NSLog(@"独立Block使用self的属性--->%@", weakSelf.myStr);  return @" "; }; myBlock(@"", 0); //内联 [self printWithblock:^(NSString *name, int age){  NSLog(@"内联Block使用self--->%@", weakSelf);  NSLog(@"内联Block使用self的属性--->%@", weakSelf.myStr);  return @" "; }]; 

将self转化成为一个用__weak修饰的weakSelf,就可以避免循环引用。

#注意:只有self中包含block的引用,并且block内使用了self才会循环引用。不过为了保险起见,所有block内用到self的还是加上__weak为好。

以上是我个人对block至今为止全部的理解,希望对初学者有一定的帮助。

有不足和错误之处,欢迎指正。

正文到此结束
Loading...