iOS中Block的使用

作者: 夏天的枫_ | 来源:发表于2016-12-29 18:15 被阅读234次

    1.什么是Block

    带有局部变量的匿名函数,与C语言中的函数指针类似,可当做参数进行传值,且可以没有名字。
    格式如下:


    • block的代码是内联的,效率高于函数调用;
    • block对于外部变量默认是只读属性;
    • block被Objective-C看成是对象处理。

    2.Block的使用

    1.无参数无返回值
    //1,无参数,无返回值,声明和定义
    
    void(^MyBlockOne)(void) = ^(void){
    
    NSLog(@"无参数,无返回值");  
    
    };  
    MyBlockOne();//block的调用
    
    2.有参数无返回值
    //2,有参数,无返回值,声明和定义
    
    void(^MyblockTwo)(int a) = ^(int a){
    
    NSLog(@"@ = %d我就是block,有参数,无返回值",a);
    
      };  
    MyblockTwo(100);
    
    3.有参数有返回值
    //3,有参数,有返回值
    
    int(^MyBlockThree)(int,int) = ^(inta,intb){    
    
      NSLog(@"%d我就是block,有参数,有返回值",a + b);returna + b; 
    
     };  
    MyBlockThree(12,56);
    
    4.无参数有返回值(很少用到)
    //4,无参数,有返回值
    
    int(^MyblockFour)(void) = ^{NSLog(@"无参数,有返回值");
            return45;
      };
    MyblockFour();
    

    以上情况都是局部变量的block 参考链接

    5.为避免使用同类型block时都要编辑大量代码,可以使用typedef 定义
    typedef int (^MyBlock)(int , int);
    

    这时,MyBlock就成为了一种Block类型
    在定义类的属性时可以这样:

    @property (nonatomic,copy) MyBlock myBlockOne;
    

    使用时:

    self.myBlockOne = ^int (int ,int){
                //TODO
    }
    

    现在假设一个使用场景,B页面要反向传递一个值到A页面。在B.h类中定义了一个属性的@property (nonatomic,copy) void (^myBlock)(NSString * data);,在B.m中调用的时候一般要进行一个判断(学习的时候大家应该都会,啰嗦下~)

    if (self. myBlock) {
            self. myBlock(//需要传递的对象值);
        }
    

    在A.m中接受传递值的时候:

    B.myBlock = ^(NSString * data){
        //TODO
        //data 就是B页面传递到A页面的数据
    }
    //B 为B页面实例化的对象
    
    6.截获自动变量(自动变量=局部变量)

    看下面代码:

    int main() 
    {
         int dmy = 256;
         int val = 10; 
         const char *fmt = "val = %d\n";
        void (^blk)(void) = ^{printf(fmt, val);}; 
        val = 2; 
        fmt = "These values were changed. val = %d\n"; 
        blk(); 
        return 0;
    }
    执行结果:val = 10
    

    解释:在该源代码中,Block语法的表达式使用的是它之前声明的自动变量fmt 和val.Block语法中,Block表达式截获的自动变量,级保存该自动变量瞬间的值。因为Block表达式保存了自动变量的值,所以在执行Block语法之后,即使概念Block中的自动变量的值也不会影响Block执行时自动变量的值。这就是所谓的截获

    7.用_ _block修饰符修饰,改变局部变量的值
    int val = 0;
    void (^block)(void) = ^{
          val = 1;
    }
    block();
    NSLog(@"val = %d",val);
    

    它会直接报错 error: variable is not assignable (missing __block type specifier)

    这时候就是在告诉您要想改变Block中局部变量,可使用__block 修饰符修饰变量。

    __block int val = 0;
    void (^block)(void) = ^{
          val = 1;
    }
    block();
    NSLog(@"val = %d",val); //val的值为1
    

    还有一种纯属使用的情况,不改变值,如:

    id array = [[NSMutableArray alloc] init]; 
    void (^blk)(void) = ^{id obj = [[NSObject alloc] init];
    [array addObject:obj]; };
    

    没有向array赋值,仅仅是使用,是截获到了NSMutableArray类对象的结构体指针,不会报错;当向它赋值时就会编译报错。

    8. 使用__weak关键字,避免循环使用造成内存泄漏

    假设使用场景:在二级页面B中定义了Block属性@property (nonatomic,copy) void (^stateBlock)(NSString * data);
    在B.m中调用了block,self.stateBlock(//需要传递的值);
    在A.m中

    - (void)buttonAction {  
       B *myVC = [[B alloc] init];
        [self presentViewController:myVC animated:YES completion:^{    
        }];
        __weak typeof(self) weakSelf = self;//防止循环引用
    //用属性定义的注意:这里属性是不会自动补全的,方法就会自动补全
        [myVC.stateBlock = ^(NSString * data){
            weakSelf.labelA.text = data;
        }];
    }
    
    9. Block的回调

    开发者在block没发布前,实现回调基本都是通过代理的方式进行的。比如负责网络请求的原生类NSURLConnection类,通过多个协议方法实现请求中的事件处理。而在最新的环境下,使用的NSURLSession已经采用block的方式处理任务请求了。各种第三方网络请求框架也都在使用block进行回调处理。这种转变很大一部分原因在于block使用简单,逻辑清晰,灵活等原因。接下来我会完成一次网络请求,然后通过block进行回调处理。这些回调包括请求完成

    按照returnValue(^blockName)(parameters)的方式进行block的声明未免麻烦了些,我们可以通过关键字typedef来为block起类型名称,然后直接通过类型名进行block的创建:

    //DownloadManager.h
    #import <Foundation/Foundation.h>
    
    @interface DownloadManager : NSObject <NSURLSessionDownloadDelegate>
    
    // block 重命名
    typedef void (^DownloadHandler)(NSData * receiveData, NSError * error);
    
    - (void)downloadWithURL:(NSString *)URL parameters:(NSDictionary *)parameters handler:(DownloadHandler)handler ;
    
    @end
    
    //DownloadManager.m
    #import "DownloadManager.h"
    
    @implementation DownloadManager
    
    - (void)downloadWithURL:(NSString *)URL parameters:(NSDictionary *)parameters handler:(DownloadHandler)handler
    {
        NSURLRequest * request = [NSURLRequest requestWithURL:[NSURL URLWithString:URL]];
        NSURLSession * session = [NSURLSession sharedSession];
    
        //执行请求任务
        NSURLSessionDataTask * task = [session dataTaskWithRequest:request completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
            if (handler) {
                dispatch_async(dispatch_get_main_queue(), ^{
                    handler(data,error);
                });
            }
        }];
        [task resume];
        
    }
    

    上面通过封装NSURLSession的请求,传入一个处理请求结果的block对象,就会自动将请求任务放到工作线程中执行实现,我们在网络请求逻辑的代码中调用如下:

    - (IBAction)buttonClicked:(id)sender {
        #define SOGOUURL @"http://dlsw.baidu.com/sw-search-sp/soft/9d/25765/sogou_mac_32c_V3.2.0.1437101586.dmg"
        //下载类
        DownloadManager * downloadManager = [[DownloadManager alloc] init];
        [downloadManager downloadWithURL: SOGOUURL parameters:nil handler:^(NSData *receiveData, NSError *error) {
            if (error) {
                NSLog(@"下载失败:%@",error);
            }else {
                NSLog(@"下载成功,%@",receiveData);
            }
        }];
    }
    

    参考链接


    BlockDemo

    相关文章

      网友评论

      • 独木舟的木:无参数无返回值调用名是不是少写了几个字母
        MyBlockOne()
        夏天的枫_:@独木舟的木 是的,感谢。已更正

      本文标题:iOS中Block的使用

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