在 Swift 中使用 Obj-C 完成块

Vee*_*Vee 3 objective-c swift completion-block

在 Objective-C 中,我有一个完成块类定义为:

文件.h

typedef void (^MYCompletionBlock)(BOOL success, NSDictionary *result, NSError *error);
Run Code Online (Sandbox Code Playgroud)

然后,在 Swift 文件中,我尝试使用完成块,如下所示:

斯威夫特.斯威夫特

class MyClass: NSObject{
     ...

     func MyFunction() -> Void {
          ...
          objcMethod(param1, withCompletion: {(MYCompletionBlock) -> Void in
               if (success){ // Error:"Use of unresolved identifier 'success'"
               }
          }
          ... 
     }
     ...
}
Run Code Online (Sandbox Code Playgroud)

但是,我不断收到错误:“使用未解析的标识符‘成功’”。

我也尝试过以下方法:

objcMethod(param1, withCompletion: {(success:Bool, result: NSDictionary, error:NSError) -> Void in
     if (success){ // Error:"Cannot convert value of type '(Bool, NSDictionary, NSError) -> Void' to expected argument type "MYCompletionBlock!" 
     }
}
Run Code Online (Sandbox Code Playgroud)

有人可以帮助我理解如何在 Swift 中正确指定 Obj-C 完成块吗?

Rob*_*Rob 5

鉴于您的闭包没有指定可空性限定符(它们几乎肯定是可选的),人们可以安全地假设您的 Objective-C API 尚未经过可空性审核。因此,Swift 会将指针视为隐式解包的选项。此外,现在NSDictionary已映射到[NSObject : AnyObject]Swift 字典。

因此,它将是:

obj.objcMethod(param) { (success: Bool, result: [NSObject : AnyObject]!, error: NSError!) in
    if success {
        // do something
    }
}
Run Code Online (Sandbox Code Playgroud)

或者,正如 Kobi 指出的那样,您可以让编译器推断类型:

obj.objcMethod(param) { success, result, error in
    if success {
        // do something
    }
}
Run Code Online (Sandbox Code Playgroud)

请注意,您不必自己记住这一点。您可以在输入代码时利用 Xcode 的代码完成功能。因此,输入足以匹配方法名称的内容,当匹配时objcMethod,然后按 Enter:

在此输入图像描述

当你到达 时MYCompletionBlock,再次按回车键,它将显示正确的签名:

在此输入图像描述


如果这个 Objective-C 方法是我自己的类,我会审核它的可为空性。因此,例如,假设 是param可选的,闭包是必需的,并且 和resulterror可选的,您可以这样定义它:

NS_ASSUME_NONNULL_BEGIN

typedef void (^MYCompletionBlock)(BOOL success, NSDictionary * _Nullable result, NSError * _Nullable error);

@interface MyObject : NSObject

- (void)objcMethod:(NSDictionary * _Nullable)param1 withCompletionHandler:(MYCompletionBlock)completionHandler;

@end

NS_ASSUME_NONNULL_END
Run Code Online (Sandbox Code Playgroud)

而且,如果是这种情况,您的 Swift 代码将像这样调用它:

obj.objcMethod(param) { (success: Bool, result: [NSObject : AnyObject]?, error: NSError?) in
    if success {
        // do something
    }
}
Run Code Online (Sandbox Code Playgroud)

或者,再次让编译器为您推断类型(但这次它们将被推断为未隐式解包的可选值):

obj.objcMethod(param) { success, result, error in
    if success {
        // do something
    }
}
Run Code Online (Sandbox Code Playgroud)