美文网首页
【译】捕获上下文信息

【译】捕获上下文信息

作者: mbillchan | 来源:发表于2016-06-30 21:47 被阅读8次

作者:Erica Sadun,原文链接,原文日期:2015-08-27
译者:CMB;校对:numbbbbb;定稿:千叶知风

假设你正在使用一个类型,当有错误时发生时你想要输出异常发生时的上下文。通常你会使用一些内置的编译器关键字:__FUNCTION____LINE____FILE__ ,这些关键词提供了有关函数调用详细的文本插值:

public struct Error: ErrorType {
    let source: String; let reason: String
    public init(_ reason: String, source: String = __FUNCTION__,
        file: String = __FILE__, line: Int = __LINE__) {
            self.reason = reason; self.source = "\(source):\(file):\(line)"
    }
}

一行典型的 Error 输出如下所示:

Error(source: "myFunction():<EXPR>:14", reason: "An important reason")

虽然这种结构能够让你捕获出现异常的函数、文件和行号,但你无法捕捉没有类型参数的原始父类型。为了捕获该类型,需要在 Error 结构体构造器中包含“原始类型”,并向构造器中传递 self.dynamicType 参数。

public struct Error: ErrorType {
    let source: String; let reason: String
    public init(_ reason: String, type: Any = "", 
        source: String = __FUNCTION__,
        file: String = __FILE__, 
        line: Int = __LINE__) {
            self.reason = reason; self.source = "\(source):\(file):\(line):\(type)"
    }
}

我很不喜欢这种额外添加类型参数的方式,它唯一的作用就是简化错误生成。

public struct Parent {
    func myFunction() throws {
        throw Error("An important reason", type: self.dynamicType)}
}

do {try Parent().myFunction()} catch{print(error)}
// Error(source: "myFunction():<EXPR>:14:Parent", reason: "An important reason")

我更喜欢扩展 Contextualizable 来实现自动捕获类型上下文。注意,默认实现的协议方法中用到了 self.dynamicType,它不能被用在方法签名中(译者注:也就是说不能当做函数参数或者返回值)。

protocol Contextualizable {}
extension Contextualizable {
    func currentContext(file : String = __FILE__, function : String = __FUNCTION__, line : Int = __LINE__) -> String {
        return "\(file):\(function):\(line):\(self.dynamicType)"
    }
}

结合上述两种方法可以简化整个过程轻松实现我们的目标。共享 Error 类型之后就可以把变量改成常量,并且把上下文相关代码从 Error 构造器移动到遵循协议的类型中,这样就可以自动继承 currentContext 方法。

public struct Error: ErrorType {
    let source: String; let reason: String
    public init(_ source: String = __FILE__, _ reason: String) {
        self.reason = reason; self.source = source
    }
}
public struct Parent: Contextualizable {
    func myFunction() throws {
        throw Error(currentContext(), "An important reason")}

更新之后,错误输出中会包含原始类型。

正如读者 Kametrixom 所指出的,你还可以扩展 Contextualizable 协议并创建你自己的错误。(他还写了一个非常棒的错误类型,可以选择是否添加上下文。)

本文的所有代码可以在 这个 Gist 中找到(译者注:Gist 已经被墙,需要翻墙查看)。

相关文章

  • 【译】捕获上下文信息

    作者:Erica Sadun,原文链接,原文日期:2015-08-27译者:CMB;校对:numbbbbb;定稿:...

  • Block 使用心得

    Block 与 Method 最大的区别在于,Block 能够捕获上下文中的变量; Block 能够捕获上下文中的...

  • 我的信息管理方法论与实践(二)- 信息捕获

    信息捕获的目的 及时记录,不让有价值的信息「溜走」 信息捕获是将有价值的信息储存下来,为后续「吸收信息价值」、「搭...

  • Swift编程八(闭包)

    案例代码下载 闭包 函数闭包可以从定义它们的上下文中捕获和存储对任何常量和变量的引用,Swift处理捕获的所有内存...

  • Swift编程八(闭包)

    案例代码下载 闭包 函数闭包可以从定义它们的上下文中捕获和存储对任何常量和变量的引用,Swift处理捕获的所有内存...

  • C++11之lambda

    lambda函数的语法如下: [] capture list 捕获列表,定义了lambda函数如何引用上下文中既有...

  • swift闭包中的值捕获

    闭包可以捕获和存储其所在上下文中任意常量和变量的引用。被称为包裹常量和变量。 Swift 会为你管理在捕获过程中涉...

  • 8.闭包

    闭包可以捕获和存储其所在上下文中任意常量和变量的引用。被称为包裹常量和变量。 Swift 会为你管理在捕获过程中涉...

  • 箭头函数的特点

    不绑定arguments,用rest参数…解决 本身没有this的概念,捕获其所在上下文的 this 值,作为自己...

  • Fu*king Closures

    闭包能在其被定义的上下文中捕获和存储任何常量和变量的引用,这叫关闭这些常量和变量。 全局函数是有名字且不捕获任何值...

网友评论

      本文标题:【译】捕获上下文信息

      本文链接:https://www.haomeiwen.com/subject/mpdfjttx.html