zoukankan      html  css  js  c++  java
  • 关注block

    代码块本质上是和其他变量类似。不同的是,代码块存储的数据是一个函数体。使用代码块是,你可以像调用其他标准函数一样,传入参数数,并得到返回值。

    脱字符(^)是块的语法标记。按照我们熟悉的参数语法规约所定义的返回值以及块的主体(也就是可以执行的代码)。下图是如何把块变量赋值给一个变量的语法讲解:

    按照调用函数的方式调用块对象变量就可以了:
    int result = myBlock(4); // result是 28

    1、参数是NSString*的代码块

    [cpp] view plain copy
     
    1. void (^printBlock)(NSString *x);  
    2. printBlock = ^(NSString* str)  
    3. {  
    4.     NSLog(@"print:%@", str);  
    5. };  
    6. printBlock(@"hello world!");  

    运行结果是:print:hello world!

    2、代码用在字符串数组排序

    [cpp] view plain copy
     
    1. NSArray *stringArray = [NSArray arrayWithObjects:@"abc 1", @"abc 21", @"abc 12",@"abc 13",@"abc 05",nil];  
    2. NSComparator sortBlock = ^(id string1, id string2)  
    3. {  
    4.     return [string1 compare:string2];  
    5. };  
    6. NSArray *sortArray = [stringArray sortedArrayUsingComparator:sortBlock];  
    7. NSLog(@"sortArray:%@", sortArray);  

    运行结果:sortArray:(

        "abc 05",

        "abc 1",

        "abc 12",

        "abc 13",

        "abc 21"

    )

    3、代码块的递归调用

    代码块想要递归调用,代码块变量必须是全局变量或者是静态变量,这样在程序启动的时候代码块变量就初始化了,可以递归调用

    [cpp] view plain copy
     
    1. static void (^ const blocks)(int) = ^(int i)  
    2. {  
    3.     if (i > 0) {  
    4.         NSLog(@"num:%d", i);  
    5.         blocks(i - 1);  
    6.     }  
    7. };  
    8. blocks(3);  

    运行打印结果:

    num:3

    num:2

    num:1

     4、在代码块中使用局部变量和全局变量

    在代码块中可以使用和改变全局变量

    [cpp] view plain copy
     
    1. int global = 1000;  
    2. int main(int argc, const char * argv[])  
    3. {  
    4.     @autoreleasepool {  
    5.         void(^block)(void) = ^(void)  
    6.         {  
    7.             global++;  
    8.             NSLog(@"global:%d", global);  
    9.         };  
    10.         block();  
    11.         NSLog(@"global:%d", global);  
    12.     }  
    13.     return 0;  
    14. }  

    运行打印结果:

    global:1001

    global:1001

    而局部变量可以使用,但是不能改变。

    [cpp] view plain copy
     
    1. int local = 500;  
    2. void(^block)(void) = ^(void)  
    3. {  
    4.       local++;  
    5.     NSLog(@"local:%d", local);  
    6. };  
    7. block();  
    8. NSLog(@"local:%d", local);  

    在代码块中改变局部变量编译不通过。怎么在代码块中改变局部变量呢?在局部变量前面加上关键字:__block

    [cpp] view plain copy
     
    1. __block int local = 500;  
    2. void(^block)(void) = ^(void)  
    3. {  
    4.     local++;  
    5.     NSLog(@"local:%d", local);  
    6. };  
    7. block();  
    8. NSLog(@"local:%d", local);  

    运行结果:local:501

                        local:501

     

     

    // 无参无返的block
        // float a = 0
       void (^noTouchBlock)()  = ^void () {
           // 功能语句
            NSLog(@"别动我电脑");
       };
        
       // block调用
        noTouchBlock();
        
        
       // 带参的block
       __block int number = 10;
        
       void (^aBlock)(int) = ^void (int a) {
            
            NSLog(@"a = %d", a);
           
           number = 20;
           
       };
        
        NSLog(@"调用前 number = %d", number);
        aBlock(5);
        NSLog(@"调用后number = %d", number);
        
        
        // 打印block的地址
        // 没有使用局部变量的block存储在全局区
        NSLog(@"%p", noTouchBlock);
        
        // 使用局部变量的block存储在栈区
        NSLog(@"%p", aBlock);

    一.block在OC中的用法可以分为大概一下几种.

      1>用于成员属性,保存一段代码,可以替代代理传值.

        比如说,创建一个ViewController控制器,点击屏幕就跳转到ModalViewController控制器里的时候,不用代理用block实现一些功能:

    复制代码
     
     // 在ModalViewController.h文件里声明:
    
    @property (nonatomic, strong) void(^valueBlock)(NSString *data);
    
      //在ModalViewController.m文件里:
    - (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
    
    {
    
        if (_valueBlock) {
    
            _valueBlock(@"dddd");
    
        }
    
    }
    
    //在ViewController.m里:
    - (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{  
        ModalViewController *modalVc = [[ModalViewController alloc] init];
        
        modalVc.valueBlock = ^(NSString *data){
            NSLog(@"%@",data);
        };
        
        [self presentViewController:modalVc animated:YES completion:nil];
     
    }
    复制代码

      2>用于参数传递

    复制代码
    //自定义一个类,用于计算,在CalculatorManage.h文件里提供一个接口
    - (double)calculator:(int(^)(int result))block;
    
    //在CalculatorManage.m里
    - (double)calculator:(int (^)(int result))block
    {
       _reslut = block(_reslut);
        
        return _reslut;
    }
    
    
    //当在外界调用的时候,
    - (void)viewDidLoad {
        [super viewDidLoad];
        
        CalculatorManage *mgr = [[CalculatorManage alloc] init];
        
        [mgr calculator:^(int result) {
            result += 5;
            result *= 2;
            
            return result;
        }];
        
        NSLog(@"%d",mgr.reslut);
         
    }
    复制代码

      3>用于返回值

    复制代码
    //在CalculatorManage.h文件里提供接口
    - (void(^)(int value))add;
    
    //在CalculatorManage.m里
    - (void (^)(int value))add
    {
        return ^(int value){
            _reslut += value;
        };
    }
    
    //在外界调用的时候可以直接用:
    - (void)viewDidLoad {
        [super viewDidLoad];
    
        CalculatorManage *mgr = [[CalculatorManage alloc] init];
        
        mgr.add(5);
    }
    复制代码

      4>注意block的循环引用(难点) 

        1)简单的循环引用

           解决方法:__weak typeof(self) weakSelf = self;

    复制代码
    //场景1:当点击控制器的view的时候,modal出来一个控制器,在modal出来的控制器的view再点击一下,dismiss当前的控制器,以下是核心部分的代码 
    @property (nonatomic, strong) void(^block)();

      //block会把外界的强指针强引用

    - (void)viewDidLoad {

        [super viewDidLoad];

      
        __weak typeof(self) weakSelf = self;
       
        self.block = ^(){
            
            typeof(self) strongSelf = weakSelf;
                
                NSLog(@"%d",strongSelf.age);
            
        };
        
        self.block();
        
    }
    - (void)dealloc
    {
      NSLog("控制器销毁");
    }
    复制代码

     我画了个图来方便理解,

         <1>modal出来之前最开始由modaVC强引用modalVC对象

      <2>modal出来之后由self.presented强引用modaVC对象

      <3>dismiss之后,没有强指针强引用modalVC对象

      <4>但是block对象会对访问的外部的强指针强引用,所有把self变成弱指针,就可以解决循环引用

     

       2)复杂的循环引用

          

    复制代码
    //场景2:当点击控制器的view的时候,modal出来一个控制器,在modal出来的控制器的view再点击一下,dismiss当前的控制器,在控制器释放之前,需要在block块里面完成一些其他的业务逻辑,以下是核心部分的代码 
     @property (nonatomic, strong) void(^block)();
    
    - (void)viewDidLoad {
    
        [super viewDidLoad];
    
       _age = 1;
        __weak typeof(self) weakSelf = self;
       
        self.block = ^(){
            //把weakSelf变成强指针
            __strong typeof(weakself) strongSelf = weakSelf;
            
            dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(3 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
                
                NSLog(@"%d",strongSelf.age);
    
            });
            
        };
        
        self.block();
        
    }
    - (void)dealloc
    {
      NSLog("控制器销毁");
    }
     
    复制代码

    为了方便理解,我画了个图:

    前面几步和上面的一样,从block块里的第一句代码开始

    1>有个strongSelf是强指针指向modalVC对象

    2>在dispatch  block对象里面,也有个strongSelf指向modalVC对象

    3>整个block块一过,strongSelf指针销毁

    4>但是dispatch block对象是延迟执行的,所以在modalVCdismiss之后,并不会马上销毁,在这延迟的3秒里,还是有dispatch block对象强指针指向modalVC的

    5>过了3秒之后,dispatch block块执行完毕,系统不再对dispatch进行强引用,所以dispatch block销毁,因此它里面的strongSelf指向modalVC的强指针也会销毁

    6>modalVC对象完全释放.

      5>block值传递

       首先,注意,全局变量,静态变量,__block都是指针传递,局部变量是值传递

    复制代码
    #import <Foundation/Foundation.h>
    

    void test(int a){ } int a = 10; int main(int argc, const char * argv[]) { @autoreleasepool { void (^block)() = ^(){ NSLog(@"%d",a); }; a = 20; block(); } return 0; }
    复制代码

    打印结果各位朋友觉得是什么?

    打印结果是20.因为先是执行a = 20,此时a已经变成了20,然后再执行block块,全局变量是指针传递,所以block块一执行完,打印得到a结果就是20了!

    6>block内存管理

       1)在MRC中

      (1)block没有访问外部变量,是在全局区

      (2)如果访问了外部变量,默认是在栈中

      (3)用了copy,才会保存在堆中

       2)ARC中

       如果访问了外部变量,默认在堆中

      

     

     

     
  • 相关阅读:
    BZOJ3277: 串(广义后缀自动机)
    BZOJ3998: [TJOI2015]弦论(后缀自动机)
    SPOJ7258 SUBLEX
    洛谷P3808 【模板】AC自动机(简单版)
    SPOJ8222 NSUBSTR
    BZOJ2946 [Poi2000]公共串(后缀自动机)
    Codeforces Round #490 (Div. 3)
    SPOJ1811 LCS
    洛谷P3804 【模板】后缀自动机
    list排序成员函数对string对象与char*对象排序的差别
  • 原文地址:https://www.cnblogs.com/mingjieLove00/p/5537909.html
Copyright © 2011-2022 走看看