swift基础-闭包

一,Block的回顾

1,Block通常会用在网络请求中,请求到数据后然后回调,先自定义HttpTool的工具类
实现下请求数据的方法

@implementation HttpTool

-(void)loadData:(void(^)(NSString * json))callBack;
{
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
   
        NSLog(@"%@",[NSThread currentThread]);
    
        dispatch_async(dispatch_get_main_queue(), ^{
        
            NSLog(@"%@",[NSThread currentThread]);
            callBack(@"222");
        });
    });
}
@end

2,然后在控制器里创建对象调用请求方法,等到请求数据后就能再Block里做其他的处理了

@interface ViewController ()

@property(nonatomic,strong) HttpTool * tool;

@end

@implementation ViewController

- (void)viewDidLoad {
     [super viewDidLoad];
    self.tool = [HttpTool new];

}
-(void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event
{
        [self.tool loadData:^(NSString *json) {
        NSLog(@"%@",json);
  }];
}
@end
二,用闭包来重写上面的逻辑
1,闭包的类型:(参数列表) -> (返回值列表)
import UIKit

class HttpTool: NSObject {

func loadData(callBack : (json : String)->()) {
  
    dispatch_async(dispatch_get_global_queue(0, 0)) {
        
        print("\(NSThread.currentThread())")
        
        dispatch_sync(dispatch_get_main_queue(), {
            print("\(NSThread.currentThread())")
            callBack(json: "json")
            
        })
    }
    
  }
}

2,在控制器调用

import UIKit

class ViewController: UIViewController {

var tool : HttpTool = HttpTool()

override func viewDidLoad() {
    super.viewDidLoad()
    
}

override func touchesBegan(touches: Set, withEvent event: UIEvent?) {

    tool.loadData { (json) in
        print("\(json)")
    }

 }   
}
三,闭包的循环引用
swift基础-闭包_第1张图片
40CFA1B3-5340-4795-88BF-14526C407707.png
1,第一个例子中由于HttpTool并没有对callBack这个Block产生强引用,所以不会循环引用,所以在HttpTool中属性描述一个Block,然后强引用callBack,就会发生循环引用了

#import "HttpTool.h"

@interface HttpTool ()

@property(nonatomic,copy) void(^callBack)(NSString * json);

@end

@implementation HttpTool

-(void)loadData:(void(^)(NSString * json))callBack;
{
    //强引用
    self.callBack = callBack;
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
   
        NSLog(@"%@",[NSThread currentThread]);
    
        dispatch_async(dispatch_get_main_queue(), ^{
        
            NSLog(@"%@",[NSThread currentThread]);
            callBack(@"222");
        });
    
    });
}
@end
如果不对循环引用做处理的话,执行结果是:
18E55EAC-1D2A-4DF2-B159-33CB89DBA6A6.png
只要我们weak self 就可以解决循环引用了
-(void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event
{
    __weak ViewController * weakSelf = self;
    [self.tool loadData:^(NSString *json) {
        NSLog(@"%@",json);
        weakSelf.view.backgroundColor = [UIColor redColor];
    }];
}
swift基础-闭包_第2张图片
EFFF6E8F-57D3-47CC-B337-63B17860751F.png
2,同样要解决闭包的循环引用让HttpTool强引用这个闭包callBack
import UIKit
class HttpTool: NSObject {
    //声明成可选类型的闭包
    var callBack :((json : String)->())?
    func loadData(callBack : (json : String)->()) {
    //强引用
    self.callBack = callBack
    dispatch_async(dispatch_get_global_queue(0, 0)) {
        
        print("\(NSThread.currentThread())")
        
        dispatch_sync(dispatch_get_main_queue(), {
            print("\(NSThread.currentThread())")
            callBack(json: "json")
            
        })
        }
    }
}

3,swift中解决闭包循环引用的3种方式

import UIKit

class ViewController: UIViewController {

    var tool : HttpTool = HttpTool()

    override func viewDidLoad() {
        super.viewDidLoad()
    
    }

    override func touchesBegan(touches: Set, withEvent event: UIEvent?) {
        /*
           weak var weakSelf = self
           tool.loadData { (json) in
            print("\(json)")
          //weakSelf?.view : 如果前面的可选类型,没有值,后面的所有代码不会执行
          //如果有值,系统会自动将weakSelf进行解包,并使用weakSelf
          weakSelf?.view.backgroundColor = UIColor.redColor()
       }
       */
    
    
    tool.loadData {[weak self] (json) in
        print("\(json)")
        //weakSelf?.view : 如果前面的可选类型,没有值,后面的所有代码不会执行
        //如果有值,系统会自动将weakSelf进行解包,并使用weakSelf
        self?.view.backgroundColor = UIColor.redColor()
    }
    
  
    /*
    //这种写法一旦self为空,就会报错
    //unowned相当于oc中的__unsafe_unretained,__unsafe_unretained 修饰的弱引用,如果指向的对象销毁,那么指针指向之前的内存地址,很容易产生坏内存访问,野指针。僵尸对象
    tool.loadData {[unowned self] (json) in
        print("\(json)")
        //weakSelf?.view : 如果前面的可选类型,没有值,后面的所有代码不会执行
        //如果有值,系统会自动将weakSelf进行解包,并使用weakSelf
        self.view.backgroundColor = UIColor.redColor()
    }
    */

    
}
    //相当于oc中的dealloc方法
    deinit{
        print("deinit")
    
    }

}
四,尾随闭包
    //普通写法
    tool.loadData ({[weak self] (json) in
        print("\(json)")
        //weakSelf?.view : 如果前面的可选类型,没有值,后面的所有代码不会执行
        //如果有值,系统会自动将weakSelf进行解包,并使用weakSelf
        self?.view.backgroundColor = UIColor.redColor()
    })
    //尾随闭包一
    tool.loadData (){[weak self] (json) in
        print("\(json)")
        //weakSelf?.view : 如果前面的可选类型,没有值,后面的所有代码不会执行
        //如果有值,系统会自动将weakSelf进行解包,并使用weakSelf
        self?.view.backgroundColor = UIColor.redColor()
    }
   //尾随闭包二,当闭包为函数的最后一个参数时,会自动省略()
    tool.loadData {[weak self] (json) in
        print("\(json)")
        //weakSelf?.view : 如果前面的可选类型,没有值,后面的所有代码不会执行
        //如果有值,系统会自动将weakSelf进行解包,并使用weakSelf
        self?.view.backgroundColor = UIColor.redColor()
    }

你可能感兴趣的:(swift基础-闭包)