美文网首页iOS奋斗iOS相关带我飞2
iOS-Block的使用你看我啊

iOS-Block的使用你看我啊

作者: StrongX | 来源:发表于2015-10-02 03:37 被阅读39578次

为什么题目是“Block的使用你看我啊”,而不是牛逼哄哄的“Block你看我就够了”,原因是本文并不会讲解Block在C++中的实现部分,而是停留在OC语言中。主要讲诉一些语法和使用。
(如果本文中有讲述不对或者不准确的地方欢迎大家提出来)

1、Block是什么?
 - 匿名函数
 - 截获自动变量
2、Block语法。
3、Block类型变量。
4、Block的用途。
 - 作为函数参数
 - 反向传值
 - 循环引用
(delegate差不多作用,但是显得更加简洁)

首先就是Block是什么?用一句话来概括就是带有自动变量的匿名函数。
那么我们解释清楚了什么是“匿名函数”,什么是“自动变量”,那么相信大家大概就对Block有了一个大概的认识。

  • 匿名函数
    匿名函数顾名思义就是不带名字的函数,在C语言中不允许这样的方法存在,而在OC中的Block则可以用指针来直接调用一个函数,但虽说如此我们还是需要知道指针的名称。(关于这点,额~~我们还是不要纠结的比较好。~!~~)
  • 自动变量
    自动变量在Block中的具体表现就是截获自动变量,来看下面这一段代码:
 int b = 0;
    void (^blo)() = ^{
        NSLog(@"Input:b=%d",b);
    };
    b = 3;
    blo();
    /**
     *    Input:b=0
     */

虽然我们在调用blo之前改变了b的值,但是输出的还是Block编译时候b的值,所以截获瞬间自动变量就是:在Block中会保存变量的值,而不会随变量的值的改变而改变。

我们再来看一段代码

int b = 0;
    void (^blo)() = ^{
        b = 3;
    };

这段代码编译出错,编译器提示的大概就是不能在Block中改变变量的值。因为在Block中截获了变量的瞬间值以后就不能再改变变量的值,如果想要在Block中改变变量的值,那么我们只需要在变量声明的时候加上__Block修饰符,像这样:

__block int b = 0;
    void (^blo)() = ^{
        b = 3;
    };

然而这样的情况又是允许的:

 NSMutableArray *array = [[NSMutableArray alloc]init];
    void (^blo)() = ^{
        [array addObject:@"Obj"];
    };

为什么呢,因为我们只是对截获的变量进行了操作,而没有进行赋值,所以对于截获变量,可以进行操作而不可以进行赋值。

还有一点需要注意,在Block中不可以对C语言数组进行操作,原因是:~~~不支持。。。。

结合匿名函数和截获自动变量的特性,Block可以做很多事情,我们下面在看。


我们来具体看一下Block语法的书写,我们首先来看一个完整的Block:

 ^ NSString *(NSString *a,NSString *b){
        return a;
    };

我们来分别解释下每一个部分都是什么东西:

  • “^”这个符号表示这是一个Block;
  • NSString *表示返回值。
  • (NSString *a,NSString *b)这个括号中是Block的参数,语法和C语言类似。

其实我们可以省略Block的返回值,像这样写:

^ (NSString *a,NSString *b){
        return a;
    };

这样写和上面那种写法是一模一样的,其实如果没有参数列表我们甚至可以省略参数列表,像这样:

^ {
        NSLog(@"我没有参数列表");
    };

如果把这段代码写完整,那么就是这样的:

^void(void) {
        NSLog(@"我没有参数列表");
    };

为什么需要Block变量?我们可以这样理解,我们通过这个Block变量来获取Block的指针,然后通过这个指针就可以来使用Block函数。我们先来看一下如何声明一个Block变量

      int (^Blo)(NSString *s1,NSString *s2);

对照前面的Block函数,我们可以比较容易的理解各个部分的含义:

他们分别是:

  • 返回值
  • 变量名
  • 参数列表

好的,然后我们用上面讲到的Block语法来对这个Block变量进行赋值:

    int (^Blo)(NSString *s1,NSString *s2);
    
    Blo = ^(NSString *s1,NSString *s2){
        return 1;
    };

然后我们就可以将这个Block变量当作C语言函数来使用了。


那么Block到底怎么用呢?

Block能够当作函数参数,首先我们声明一个Block类型变量 ,并加上typedef修饰符,像这样:

typedef void(^Blo)(NSString *s1,UIColor *c);

这样我们就可以使用Blo来表示这个Block,然后我就可以将Blo加入到函数参数中,我们来声明一个函数:

-(void)func:(Blo)BlockPra{
    
    BlockPra(@"Str",[UIColor redColor]);

}

然后我们可以这样使用这个函数:

[self func:^(NSString *s1, UIColor *c) {
        NSLog(@"%@",s1);
        self.view.backgroundColor = c;
    }];

是不是觉得十分眼熟,平时使用的许多回调当中大多都是这样的形式,可能其中其较多的就是网络回调了,我们只需要调用方法,然后在回调当中就可以对结果进行操作,很多苹果自己写的API都是使用了这样的方法,这样做的好处就是形式上十份简洁,当然像这种地方你使用delegate肯定也是可以的,但是表现上就没有Block那么简洁,使用起来也没有Block那么方便。

除此之外,Block还可以用来作为控制器之间的一个通信。
前面我们已经知道Blcok是一个匿名函数,同时也是一个指针,那么使用Block就可以弥补在iOS中函数传递的功能。通常是这么用的:

页面B的.h文件中定义了这样一个Block执政,然后声明了一个变量,像这样:

  typedef void(^Blo)(NSString *s1,UIColor *c);
  @property (nonatomic, copy) Blo block;

然后我们在页面A当中有这么一段代码:

ViewController *b = [[ViewController alloc]init];
    
    __weak  ViewController *wself = self;
    b.block = ^(NSString *s1,UIColor *c){
        NSLog(@"%@",s1);
        wself.view.backgroundColor = c;
    };
    [self.navigationController pushViewController:b animated:true];

然后在页面B的任意地方我们调用block变量,像这样:

     self.block(@"str",[UIColor redColor]);

都会在A页面中调用B页面传过来的参数,在A页面进行操作,对控制器A进行改变,这样的做法通常用做 控制器 反向传值。

在这里有一点需要注意就是Block的使用引起的循环引用。如果在Block中使用附有__strong修饰符的对象类型自动变量,那么当Block从栈复制到堆时,改对象为Block所有。这样容易引起循环引用,从而发生内存泄漏,然而我们只需要保证当前控制器也就是self在需要释放的时候正确释放就可以,所以我们再来看上面那段代码:

      __weak  ViewController *wself = self;

我们定义一个wself变量并加上__weak修饰符,在Block代码块中,所有需要self的地方都用wself来替代。这样就不会增加引用计数,所以Block持有self对象也就不会造成循环引用,从而造成内存泄漏。
不管是将Block当作函数参数,还是用来反向传值,其实都是对Block的本质,也就是“带有自动变量的匿名函数”的两个修饰,“带有自动变量”、“匿名函数”这两个特性 的应用。


总结一下Block到底是什么、用来干什么:

  • C++中的Struct(本文未提到)。
  • 用来弥补iOS中函数传递的功能。
  • 他是一段代码块的内存的指针。
  • 和delegate一样的功能,但是显的更加简洁。

差不多就这样吧,自我感觉本文是有逻辑的,但是部分讲解可能不是特别到位,觉得还不错的请关注本人。

相关文章

  • iOS-Block的使用你看我啊

    为什么题目是“Block的使用你看我啊”,而不是牛逼哄哄的“Block你看我就够了”,原因是本文并不会讲解Bloc...

  • iOS-Block的使用你看我啊

    为什么题目是“Block的使用你看我啊”,而不是牛逼哄哄的“Block你看我就够了”,原因是本文并不会讲解Bloc...

  • iOS-Block本质

    iOS-Block本质 参考篇:iOS-Block浅谈[https://www.jianshu.com/p/25a...

  • 别用你的眼睛看我

    瞧一瞧 看一看 你的鞋子没有那么油光瓦亮 你又为什么觉得我的鞋子上沾满了粪土 别以为全世界的菜肴你都品尝过 就有权...

  • 不要用你的眼光看我【180518】

    因为你不是我,我怕你不懂。 经常看到人家评论王菲,人家情商高啊,会玩,有担当,有内涵,任性!你不喜...

  • 用你的眼光来看我自己

    每当想起你 就像红艳艳的花朵掺杂着水珠 向着太阳反射出 七彩的颜色出来 令我激动和感到愉悦 每当想起你 那温柔的园...

  • Objective-C的本质(6)——Block本质

    参考:iOS-Block本质iOS底层原理总结 - 探寻block的本质(一)iOS底层原理总结 - 探寻bloc...

  • 疗愈文

    164使用与被使用 有一种维度的使用, 除了自己使用自己, 别人使不使用你呢? 单位用不用你呢? 团队用不用你呢?...

  • 你我之间·你快爱我

    你快来爱我啊 天要黑了 水要凉了 风要歇了 火要灭了 快来爱我 用你的唇 用你的手 用你的胸膛 用你的身体 紧一点...

  • 尹志鹏

    (三) 作者:大鹏展翅 “姐,你看我出汗了,不用你的物理疗法了...

网友评论

  • _Gyro:没有用__block修饰的变量b在block中不能修改,原因是block拿到的只是b的值,而非b的地址,但用__block修饰后,block拿到的就是b的地址了,所以是可以修改的
  • 遛遛食:不错
  • AdhereDamon:不错
    AdhereDamon:@KWK 嗯
  • e4fa7826e618:我想知道为什么我self.block()运行时报错了
  • 行走的栀子花:int (^Blo)(NSString *s1,NSString *s2); 中的Blo 与typedef int (^Blo)(NSString *s1,NSString *s2);中的Blo有什么区别?
    7cef2e74c910:第一个Blo可以用来声明block,第二个就是block
    例如声明一个整型变量: int a;
    第一个就相当于int,第二个就相当于a
    StrongX:@StrongX 说是函数名可能有些偏差,不加typedef的时候就是一个代码块。
    StrongX:@仰望星空的小哥 当你加了typedef Blo是一个类型名,如果你不加 Blo是一个函数名。
  • zweic:typedef void(^Blo)(NSString *s1,UIColor *c); 像这样, Blo中B换成小写行吗,为什么
    StrongX:@zweic 楼上正解,至于为什么我写的Blo,只是因为顺手。。。。。。。
    行走的栀子花:@zweic 可以的,因为这句话是给 void(^)(NSString *s1,UIColor *c) 这种函数命名别名的,别名怎么起都可以,一般都会首字母大写,就像类名一样。
  • DavidWu11:赞,讲得挺明白的
  • 4bc347552fb5:博主求教一个关于Block作为函数参数的问题~testAction是类A的一个方法,当我把类A释放的时候,作为函数参数的testBlock依然存在,请问下这是为什么?有什么方法是能够让testBlock的生命周期和类A的生命周期保持一致的吗?
    -(void)testAction:(void(^)())testBlock {
    WS(weakSelf, self);
    dispatch_time_t popTime = dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC));

    dispatch_after(popTime, dispatch_get_main_queue(), ^(void){
    if (testBlock) { //此时weakSelf=nil 但是testBlock依然存在
    testBlock(); //会执行回调
    }
    });
    }
  • 超_iOS:很详细,希望作者可以再深入讲解下
  • neobuger:在b页面怎么能用到 block的? bolck 不相当于 在a 控制器里, 在b 不就是出了他的作用域么? 还是 block本来就是无论哪里都能访问到?
    StrongX:@neobuger 在超出作用域的情况下,block将从栈拷贝到堆,所以超出作用域并没有被释放
  • Hc小龙:然后在页面B的任意地方我们调用block变量
    1: 放到b页面的 viewDidLoad 里面,当从a进来的时候,就会执行block.把值传过去。
    2: 放到b页面的 返回a页面的函数里面,点击返回的时候才会执行block。
    微漫步:我想请问一下block中weak的用法
    实在block中调用A界面的方法是吗,所以那个ViewController应该是A的试图控制器是吗?
    EXC_BAD_ACCESS:@Hc小龙 想表达什么눈_눈
    StrongX:@Hc小龙 是的,有什么问题吗。
  • 火星的蝈蝈:有空再深研究。谢谢作者了
  • 煜寒了:不明觉厉
  • tom510230:必须强调以上例子是基于ARC的,如果是MRC的话,__block反而是不增加引用计数的
  • f289b72b44e9:你那种演示好像没有循环引用,直接用self也可以,个人经验遇到block里面引用了self时,重写下dealloc测试下就可以了!
    StrongX:@NicoLin 我是这个意思呀,哪里写的有歧义?
    ANTI_JAM:@StrongX 是self和block相互持有才会造成循环引用,self持有block不一定会造成
    StrongX:@我相信 是否发生循环引用取决于 self是否持有block
  • Stony:Block 是 匿名函数指针,是指向匿名函数的指针,不是匿名函数。
    除非同步调用,不推荐在block内和之后同时引用同一个变量,更不要修改,没必要去搞这个脑子。
    Block引用的变量会被复制,标记block的变量会被跟踪变化并复制回原栈的产量,同步调用时很实用。

    用通俗的语言解释问题很好,倒是还精确用术语的地方还希望正确使用。
    StrongX:@Stony 1、匿名函数指针用来定义Block类型变量会更加合适,这点我在文章中也说了。而Block本身则是带自动变量的匿名函数。我们需要把这两个概念区分开来。
    2、关于你说的第二点。。。。我只是用来举个列子来说明下__block修饰符。
  • 一一无痕:赞👍
  • 4f1798560025:在 “然后我们在页面A当中有这么一段代码:
    ViewController *wself = self;” 中
    是否需要写成 __weak ViewController *wself = self; ?
    三线小奋青:@StrongX 这个不需要加__weak吧,因为block是页面B持有,页面A持有block啊,不会构成循环医用吧?求解,在线等……
    StrongX:@spclz 对啊,要加weak。感谢看的这么仔细。。。落下了
    StrongX:@spclz 这是为了避免循环引用的
  • 从来吃不胖:炒鸡棒!

本文标题:iOS-Block的使用你看我啊

本文链接:https://www.haomeiwen.com/subject/pjkgcttx.html