如何在 swift 中的闭包中抛出错误?

how to throw errors in a closure in swift?

请看下面的代码:

override func tableView(tableView: UITableView, editActionsForRowAtIndexPath indexPath: NSIndexPath) -> [UITableViewRowAction]? {

   let deleteAction = UITableViewRowAction(style: UITableViewRowActionStyle.Default, title: "Delete", handler: {
        (action : UITableViewRowAction, indexPath : NSIndexPath)  -> Void  in

        if let managedObjectContext = (UIApplication.sharedApplication().delegate as! AppDelegate).managedObjectContext{
            let restaurantToDelete = self.fetchResultController.objectAtIndexPath(indexPath) as! Restaurant
            managedObjectContext.deleteObject(restaurantToDelete)

            // Saving managedObjectContext instance, and catch errors if it fails
            do {
                try managedObjectContext.save()
            } catch let error as NSError {
                print("Error: \(error.localizedDescription)")
            }

        }

    })
    return deleteAction
}

来自 Xcode 的错误消息是:从类型为“(UITableViewRowAction, NSIndexPath) throws -> Void”的抛出函数类型到非抛出函数类型“(UITableViewRowAction, NSIndexPath) -> Void”的无效转换

我知道问题是 managedObjectContext.save() 会抛出错误,这在完成处理程序中是不允许的。我发现一些博客文章在其中修改了闭包参数以使闭包中的错误处理可行。虽然这里函数的定义是由苹果给出的,但我该如何解决这个问题呢?非常感谢! :D

不可能,因为可以随时调用闭包,可能不会在您的函数执行时调用,那么错误应该传播到哪里?

你必须调用另一个可以处理错误的函数:

func handleError(error: ErrorType) {
    switch error {
        ...
    }
}

然后在闭包中使用捕获到的错误调用此函数

编译器正在将 throws 添加到您的块的签名中,因为您的 catch 子句并不详尽:模式匹配 let error as NSError 可能会失败...请参阅 documentation

闭包参数的签名是 (UITableViewRowAction, NSIndexPath) -> Void,但是编译器推断您提供的闭包类型是 (UITableViewRowAction, NSIndexPath) throws -> Void

通过在你已有的子句之后添加另一个 catch 子句(没有模式),编译器将看到你正在本地捕获异常,并且它不再推断你是闭包的签名提供包括 throws:

do {
  try managedObjectContext.save()
} catch let error as NSError {
  print("Error: \(error.localizedDescription)")
} catch {}