swift中閉包和OC中block的定義和用法比較

一.閉包的介紹

閉包和OC中的block非常相似

  • OC中的block是匿名的函數(shù)
  • Swift中的閉包是一個(gè)特殊的函數(shù)
  • block和閉包都經(jīng)常用于回調(diào)

二.block的用法回顧

1.block寫法總結(jié)

    類型:
    返回值(^block的名稱)(block的參數(shù))

    值:
    ^(參數(shù)列表) {
        // 執(zhí)行的代碼
    }

    int (^sumOfNumbers)(int a, int b) = ^(int a, int b) {
        return a + b;
    };

2.block實(shí)現(xiàn)兩個(gè)界面之間的傳值

    ①在后面控制器的 .h文件 中聲明block
    // 一會(huì)要傳的值為NSString類型
    typedef void (^newBlock)(NSString *);

    @interface NewViewController : UIViewController
    // 聲明block屬性
    @property (nonatomic, copy) newBlock block;


    ②在后面控制器的 .m文件 中設(shè)置block
    - (void)viewWillDisappear:(BOOL)animated
    {
      [super viewWillDisappear:YES];
      if (self.block != nil) {
        self.block(@"呵呵");
      }
    }

    ③在前面控制器的 .m文件 中接收傳來(lái)的值

    NewViewController *newVC = [[NewViewController alloc] init];
    // 接收block傳來(lái)的值
    __weak ViewController *weakSelf = self;
    在 Block 中引用self,也就是 ViewController ,而 ViewController 創(chuàng)建并引用了 NewViewController,而NewViewController引用 block,此時(shí)就形成了一個(gè)循環(huán)引用,所以要用__weak修飾。

    newVC.block = ^(NSString *str){
        NSLog(@"%@", str);
    };

三.閉包的用法

1.閉包的寫法總結(jié)

    類型:(形參列表)->(返回值)
    技巧:初學(xué)者定義閉包類型,直接寫()->().再填充參數(shù)和返回值

    值:
    {
        (形參) -> 返回值類型 in
        // 執(zhí)行代碼
    }



    let b = { (parm : Int) -> (Int) in 
       print(parm)
       ……
    }

    //調(diào)用
    b(100)

2.閉包的簡(jiǎn)寫

如果閉包沒(méi)有參數(shù),沒(méi)有返回值,in和in之前的內(nèi)容可以省略.

  httpTool.loadRequest({
        print("回到主線程", NSThread.currentThread());
   })

3.尾隨閉包寫法

  • 如果閉包是函數(shù)的最后一個(gè)參數(shù),則可以將閉包寫在()后面
  • 如果函數(shù)只有一個(gè)參數(shù),并且這個(gè)參數(shù)是閉包,那么()可以不寫
    httpTool.loadRequest() {
        print("回到主線程", NSThread.currentThread());
    }
    // 開(kāi)發(fā)中建議該寫法
    httpTool.loadRequest {
        print("回到主線程", NSThread.currentThread());
    }

4.閉包的使用例子:延遲回調(diào)

 //[weak self]:解決馴化引用導(dǎo)致的內(nèi)存泄露
    override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {
        delayMethod {[weak self] (re) ->() in
            print("$$$$$$$$$$$$$$$$$:\(re)%%%%%%%%%%%\(String(describing: self))")
        }
        delayMethod(comletion: {[weak self] (re)->() in
            print("********:\(re)*************\(String(describing: self))")
        })
    }
    
    //@escaping:逃逸閉包。它的定義非常簡(jiǎn)單而且易于理解。如果一個(gè)閉包被作為一個(gè)參數(shù)傳遞給一個(gè)函數(shù),并且在函數(shù)return之后才被喚起執(zhí)行,那么這個(gè)閉包是逃逸閉包。
    func delayMethod(comletion: @escaping (_ results: String,_ resultss:String) -> ()) ->(){
        //開(kāi)啟一個(gè)全局異步子線程
        DispatchQueue.global().async {
            Thread.sleep(forTimeInterval: 2.0)
            //回調(diào)到主線程
            DispatchQueue.main.async(execute: {
                print("主線程更新 UI \(Thread.current)")
                comletion("qwertyui","asdf")
            })
        }
    }

運(yùn)行結(jié)果:

屏幕快照 2017-05-23 下午6.13.59.png

5.閉包的使用例子:兩個(gè)界面之間的回調(diào)

    //實(shí)現(xiàn)B界面要傳值給上個(gè)界面A
    //首先在B界面聲明閉包
    typealias lewisCloser = (_ paramOne : String? ) -> ()
    //定義個(gè)變量 類型就是上面聲明的閉包
    var customeCloser: lewisCloser?
    //點(diǎn)擊事件 回跳到A界面 實(shí)現(xiàn)傳值
    @IBAction func tap(_ sender: Any) {
        if(customeCloser != nil) {
            customeCloser!("qq")
        }
        self.dismiss(animated: true, completion: nil)  
     }

    //在A界面接受B界面?zhèn)鬟^(guò)來(lái)的值
    @IBAction func tap(_ sender: UIButton) {
        let vc = NewViewController()
        vc.customeCloser = {(cusValue) -> () in
            
            self.slipTap.titleLabel!.text! = cusValue!;
            print("^^^^^^^^^^^^^^^^^^^^^:\(cusValue!)")
        
        }
        self.present(vc, animated: true, completion: nil)
    }

That's all,THX! 歡迎小伙伴的交流評(píng)論!

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容