1. 程式人生 > 實用技巧 >swift中閉包和OC中block的用法比較

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

轉自:https://www.jianshu.com/p/5f98941b4c71

在OC中習慣用block來傳值,而swift中,block被重新定義了一下,叫閉包;

使用的技巧:誰定義誰傳值;

案例使用A、B控制器:

1~4步在B中執行,最後在A中執行;

  • B控制器:

1-定義

格式: typealias 閉包名稱 = (引數名稱: 引數型別) -> 返回值型別

typealias block = (str: String) -> void

2- 宣告

var callBack = block?()

3- 賦值

需要定義一個方法,引數是和block型別一致得閉包,並賦值給block


func callBackFunction ( block: (str: String) -> Void ) {

      callBackBlock = block

}

4- 傳值

func buttonClick () { //需要傳值的方法

if callBackBlock != nil {

    callBackBlock!( "傳這個值給A")    //注意,這裡是使用屬性傳值,不是方法

}

}

5 - A控制器中

B.callBackFunction { (str) in

  print("這裡使用傳過來的值")

}

swift中的閉包和oc中的block的定義和用法對比

一.閉包的介紹

閉包是功能性自包含模組,可以在程式碼中被傳遞和使用。 Swift 中的閉包與 C 和 Objective-C中的 blocks 以及其他一些程式語言中的 lambdas 比較相似。
閉包可以 捕獲 和儲存其所在上下文中任意常量和變數的引用。 這就是所謂的閉合幷包裹著這些常量和變數,俗稱閉包。Swift會為您管理在 捕獲 過程中涉及到的記憶體操作。
OC中的block是匿名的函式
Swift中的閉包是一個特殊的函式
block和閉包都經常用於回撥
二.block的用法回顧

<1>. block寫法總結:

block的寫法:

型別:

返回值型別(^block的名稱)(block的引數)



值:

^(引數列表) {

    // 執行的程式碼

}



//例子

int (^sumOfNumbers)(int a, int b) = ^(int a, int b) {

    return a + b;

};

<2>. block實現兩個介面之間的傳值

①在後面控制器的 .h檔案 中宣告block

// 一會要傳的值為NSString型別

typedef void (^newBlock)(NSString *);

@interface NewViewController : UIViewController

// 宣告block屬性

@property (nonatomic, copy) newBlock block;

②在後面控制器的 .m檔案 中設定block

- (void)viewWillDisappear:(BOOL)animated

{

  [super viewWillDisappear:YES];

  if (self.block != nil) {

    self.block(@"呵呵");

  }

}



③在前面控制器的 .m檔案 中接收傳來的值

NewViewController *newVC = [[NewViewController alloc] init];

// 接收block傳來的值

__weak ViewController *weakSelf = self;

newVC.block = ^(NSString *str){

    NSLog(@"%@,%@", weakSelf,str);

};

<3>. block作為引數進行延時回撥

定義網路請求的類
@interface HttpTool : NSObject

-(void)loadRequest:(void (^)())callBackBlock;

@end

@implementation HttpTool

-(void)loadRequest:(void (^)())callBackBlock

{

dispatch_async(dispatch_get_global_queue(0, 0), ^{

    NSLog(@"非同步延時請求操作在這裡,載入網路資料:%@", [NSThread currentThread]);

    dispatch_async(dispatch_get_main_queue(), ^{

        callBackBlock();

    });

});

}

@end

進行網路請求,請求到資料後利用block進行回撥
-(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event

{

[self.httpTool loadRequest:^{

    NSLog(@"主執行緒中,將資料回撥.%@", [NSThread currentThread]);

}];

}

三.閉包的用法

<1>. 閉包寫法總結:

型別:(形參列表)->(返回值)

技巧:初學者定義閉包型別,直接寫()->().再填充引數和返回值



值:

{

    (形參) -> 返回值型別 in

    // 執行程式碼

}



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

   print(parm)

}



//呼叫

b(100)

<2>.閉包的簡寫

如果閉包沒有引數,沒有返回值,in和in之前的內容可以省略
httpTool.loadRequest({

    print("回到主執行緒", NSThread.currentThread());

})

尾隨閉包寫法:
如果閉包是函式的最後一個引數,則可以將閉包寫在()後面
如果函式只有一個引數,並且這個引數是閉包,那麼()可以不寫
httpTool.loadRequest() {

    print("回到主執行緒", NSThread.currentThread());

}



// 開發中建議該寫法

httpTool.loadRequest {

    print("回到主執行緒", NSThread.currentThread());

}

<3>.使用閉包代替block,閉包作為引數進行延時回撥

定義網路請求的類
class HttpTool: NSObject {

func loadRequest(callBack : ()->()){

    dispatch_async(dispatch_get_global_queue(0, 0)) { () -> Void in

        print("載入資料", [NSThread.currentThread()])



         dispatch_async(dispatch_get_main_queue(), { () -> Void in

            callBack()

         })

    }

}

}

進行網路請求,請求到資料後利用閉包進行回撥
override func touchesBegan(touches: Set, withEvent event: UIEvent?) {

    // 網路請求

    httpTool.loadRequest ({ () -> () in

        print("回到主執行緒", NSThread.currentThread());

    })

}

<3>.例項二,閉包的回撥傳值

//[weak self]:解決迴圈引用導致的記憶體洩露

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:逃逸閉包。它的定義非常簡單而且易於理解。如果一個閉包被作為一個引數傳遞給一個函式,並且在函式return之後才被喚起執行,那麼這個閉包是逃逸閉包。

func delayMethod(comletion: @escaping (_ results: String,_ resultss:String) -> ()) ->(){

    //開啟一個全域性非同步子執行緒

    DispatchQueue.global().async {

        Thread.sleep(forTimeInterval: 2.0)

        //回撥到主執行緒

        DispatchQueue.main.async(execute: {

            print("主執行緒更新 UI \(Thread.current)")

            comletion("qwertyui","asdf")

        })

    }

}

<4>.閉包進行兩個介面的傳值

我們要實現點選第二個介面後,關掉第二個介面,並且傳值給第一個介面
<1>.首先在第二個介面宣告閉包進行操作
class NewViewController: UIViewController {

//宣告閉包

typealias lewisCloser = (_ paramOne : String? ) -> ()

//定義個變數 型別就是上面宣告的閉包

var customeCloser: lewisCloser?

override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {

    if(customeCloser != nil) {

        customeCloser!("要發給第一個介面的值")

    }

    self.dismiss(animated: true, completion: nil)

}

override func viewDidLoad() {

    super.viewDidLoad()

    // Do any additional setup after loading the view.

}

}

<2>.在第一個介面實現閉包,取得要穿的值

let vc = NewViewController()

//實現閉包

vc.customeCloser = {(cusValue) -> () in

  //cusValue就是傳過來的值

  print("^^^^^^^^^^^^^^^^^^^^^:\(cusValue!)")

}

self.present(vc, animated: true, completion: nil)