• Blocks编程


    1.介绍

    Block是一个C Level的语法以及运行时的一个特性,非常像标准C中的函数(函数指针),但是其运行需要编译器和运行时支持,目前LLVM+Clang可以很好的支持Block(苹果修改过的GCC也可以)。Block和函数不同的是其语义 闭包 特性,以及可以有匿名block的存在。 你可以在LLVM的官方网站查看Block语言规范
    你可以通过{}内的.

    int multiplier = 2;   
    int (^myBlock)(int) = ^(int num){   
        return num * multiplier;   
    };   
    printf("%d",myBlock(4));   
    

    上面代码中的 2. 声明创建和调用

    声明Block变量 
    Block变量保存着指向Block的指针,声明一个Block变量就和声明一个函数指针变量类似,只是将*改成了. 其他的就和C的类型系统保持一致了。

    void (^blockReturningVoidWithVoidArgument)(void);   
    int (^blockReturningIntWithIntAndCharArguments)(int, char);   
    void (^arrayOfTenBlocksReturningVoidWithIntArgument[10])(int);
    

    另 Block还支持可变参数variadic (...) ,没有参数的话,变量列表的地方必须写上void关键字.

    声明Block类型 你可以通过typedef声明Block的类型,这样多个地方需要使用同种类型的Block的时候会比较方便,

    typedef float (^MyBlockType)(float, float);   
    MyBlockType myFirstBlock = // ... ;   
    MyBlockType mySecondBlock = // ... ;
    

    Block创建 声明了一个Block变量之后,可以为这个变量赋值

    blockReturningVoidWithVoidArgument = ^{   
          printf("%s","Block Returing Void With Void Argument.");   
    }
    

    当然你可以在声明变量的同时赋值

    void (^blockReturningVoidWithVoidArgument)(void) = ^{   
          printf("%s","Block Returing Void With Void Argument.");   
    }
    

    Block调用 Block的调用和函数的调用是非常相似的,如上面定义的blockReturningVoidWithVoidArgument,调用的时候则直接blockReturningVoidWithVoidArgument();便可.

    匿名Block 当一个Block作为函数参数的时候,一般实参都是以匿名Block的方式传过去的。

    void callVoidVoid(void (^closure)(void)) {   
       closure();   
    }   
    int main(int argc, char *argv[]) {   
        __block int i = 10;       
        callVoidVoid(^{ ++i; });   
       if (i != 11) {   
           printf("*** %s didn't update i
    ", argv[0]);   
           return 1;   
       }   
       printf("%s: success
    ", argv[0]);   
       return 0;   
    }
    

    当然你也可以直接调用匿名Block,如

    ^{ ++i; }();
    

    3.Block和变量

    一个Block的内部是可以引用自身作用域外的变量的,包括static变量,extern变量或自由变量(定义一个变量的时候,如果不加存储修饰符,默 认情况下就是自由变量auto,auto变量保存在stack中的.除了auto之外还存在register,static等存储修饰符) ,对于自由变量,在Block中是只读的。在引入block的同时,还引入了一种特殊的变量存储修饰符 代码示例 3.1

    __block int blockValue = 0;
    int autoValue = 0;
    
    void(^printValue)(void) = ^{
        printf("blockValue = %d
    ",blockValue);
        printf("autoValue = %d
    ",autoValue);
    
    };
    blockValue ++;
    autoValue ++;
    printValue();
    

    3.1中的代码,输出的值为

    blockValue = 1
    autoValue = 0
    

    可以看到自由变量尽管自增了,但是在调用printValue这个Block的时候,看到的还是其定义的时候看到的那个autoValue的值,autoValue的值在Block的内部示无法修改的,要不然编译器会报错:

    Semantic Issue: Variable is not assignable (missing __block type specifier)
    

    Block定义时内存是分配在stack上的,当其作用域结束,就会被自动释放,所以你不需要去担心它的内存情况,我们可以对一个Block进行Block_copy()之后,Block会被拷贝到heap中的内存中,且其所有的引用到的自由变量也将会被拷贝,当然你得记得通过 在Block内部如果引用到对象或者对象的成员变量,那么当Block被拷贝 代码示例 3.2

    NSObject *testObject = [[NSObject alloc] init];
    
    NSLog(@"%lu",[testObject retainCount]); //1
    NSLog(@"%lu",[self retainCount]); //1
    
    void(^testBlock)(void) = ^{
           NSLog(@"The Test String : %@", testObject);
       NSLog(@"The Window Object : %@", _window);
    };
    
    NSLog(@"%lu",[testObject retainCount]); //1
    NSLog(@"%lu",[self retainCount]); //1
      
    void(^testBlock2)(void) = Block_copy(testBlock); //testBlock会被拷贝到heap中,所以用完了要自己调用Block_release进行释放
    NSLog(@"%lu",[testObject retainCount]); //2
    NSLog(@"%lu",[self retainCount]); //2
    
    testBlock2();
    
    Block_release(testBlock2);
    
    NSLog(@"%lu",[testObject retainCount]); //1
    NSLog(@"%lu",[self retainCount]); //1
    
    [testObject release];
    

    Block的闭包特性使得Block可以脱离其定义的作用域进行运行,所以你可以在一个函数中返回一个Block,在别的线程或者当前线程的RunLoop中进行运行,而不用担心那些引用到的外部变量是否被释放掉了。

    4.Block实际应用

    那么我们一般什么时候会用到Block呢? Blocks通常是一小段自包含的代码片段.所以它经常被用于多线程运行的代码单元(如GCD),或用于处理聚合类元素单元,或者作为某个函数调用完成后的回调函数.

    Block用作回调函数比传统的回调函数有以下的优越性: (Jacky Shin:下面这个是重点,需要仔细理解使用场合)

    • 在函数调用的时候,将Block作为一个参数传给函数
    • 允许访问本地变量,这样可以避免通过结构体将本地变量封装后传递给回调函数

    应用1: Animations & Completion Handler

    [UIView animateWithDuration:2 
                   animations:^{
                        self.view.backgroundColor = [UIColor redColor];
                    }  
                    completion:^(BOOL finished){
                        if (finished){
                            self.view.backgroundColor = [UIColor blueColor];
                        } 
                    }];
    

    应用2: Enumeration 
    对数据集合类中的每一个元素进行遍历,每次传入一个对象,进行处理

    NSArray *cards = [NSArray arrayWithObjects:@"Jack", @"Queen", @"King", @"Ace", nil];
    [cards enumerateObjectsUsingBlock:^(id object, NSUInteger index, BOOL *stop) {
         NSLog(@"%@ card at index %d", object, index);
     }];
    

    应用3: Notification Handler

        
    [[NSNotificationCenter defaultCenter] 
     addObserverForName:@"TestNotification" 
     object:nil 
     queue:aNSOperationQueue 
     usingBlock:^(NSNotification *notification){
         NSLog(@"Notification: %@",notification);
     }];
    

    应用4: GCD

        
    dispatch_queue_t imageDownloadQueue = dispatch_queue_create("Image Download Queue", NULL);
    dispatch_async(imageDownloadQueue, ^{
        NSURL *imageURL = [NSURL URLWithString:@"http://xxx.xx.com/a.png"];
        NSData *imageData = [NSData dataWithContentsOfURL:imageURL];
        UIImage *image = [UIImage imageWithData:imageData];
        dispatch_async(dispatch_get_main_queue(), ^{
            [imageView setImage:image];
        });
    });
  • 相关阅读:
    如何保存PDF、Word和Excel文件到数据库中
    C#添加PDF页眉——添加文本、图片到页眉
    C#数组,List,Dictionary的相互转换
    C#向PPT文档插入图片以及导出图片
    【CTSC2018】暴力写挂(边分治,虚树)
    【WC2018】通道(边分治,虚树,动态规划)
    【BZOJ2870】最长道路(边分治)
    【WC2018】州区划分(FWT,动态规划)
    【LOJ#6029】市场(线段树)
    【Hihocoder1413】Rikka with String(后缀自动机)
  • 原文地址:https://www.cnblogs.com/duyuiOS/p/4899908.html
Copyright © 2020-2023  润新知