3

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

文件.h

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

然后,在 Swift 文件中,我尝试按如下方式使用完成块:

斯威夫特.swift

class MyClass: NSObject{
     ...

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

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

我也尝试过以下方法:

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!" 
     }
}

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

4

2 回答 2

5

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

因此,它将是:

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

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

obj.objcMethod(param) { success, result, error in
    if success {
        // do something
    }
}

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

在此处输入图像描述

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

在此处输入图像描述


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

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

而且,如果是这样的话,你的 Swift 代码会这样调用它:

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

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

obj.objcMethod(param) { success, result, error in
    if success {
        // do something
    }
}
于 2016-05-20T23:57:18.210 回答
3

您不应该为完成块参数指定类型,因为某些类型在 Swift 和 Objective C 之间存在延迟(例如BOOL,实际上是ObjCBool在 Swift 中)。这应该有效:

objcMethod(param1) { (success, result, error) in
    if (success){ 
        // Do something
    }
}
于 2016-05-20T22:32:17.103 回答