首页 > 实验楼的一段关于golang错误处理的代码,中间有两个疑问,求好心人解答一下,不甚感激。

实验楼的一段关于golang错误处理的代码,中间有两个疑问,求好心人解答一下,不甚感激。

package main

import "errors"
import "fmt"

// 按照惯例,错误通常是最后一个返回值并且是 `error` 类
// 型,一个内建的接口。
func f1(arg int) (int, error) {
    if arg == 42 {

        // `errors.New` 构造一个使用给定的错误信息的基本
        // `error` 值。
        return -1, errors.New("can't work with 42")

    }

    // 返回错误值为 nil 代表没有错误。
    return arg + 3, nil
}

// 通过实现 `Error` 方法来自定义 `error` 类型是可以得。
// 这里使用自定义错误类型来表示上面的参数错误。
type argError struct {
    arg  int
    prob string
}

func (e *argError) Error() string {
    return fmt.Sprintf("%d - %s", e.arg, e.prob)
}

func f2(arg int) (int, error) {
    if arg == 42 {

        // 在这个例子中,我们使用 `&argError` 语法来建立一个
        // 新的结构体,并提供了 `arg` 和 `prob` 这个两个字段
        // 的值。
        return -1, &argError{arg, "can't work with it"}
    }
    return arg + 3, nil
}

func main() {

    // 下面的两个循环测试了各个返回错误的函数。注意在 `if`
    // 行内的错误检查代码,在 Go 中是一个普遍的用法。
    for _, i := range []int{7, 42} {
        if r, e := f1(i); e != nil {
            fmt.Println("f1 failed:", e)
        } else {
            fmt.Println("f1 worked:", r)
        }
    }
    for _, i := range []int{7, 42} {
        if r, e := f2(i); e != nil {
            fmt.Println("f2 failed:", e)
        } else {
            fmt.Println("f2 worked:", r)
        }
    }

    // 你如果想在程序中使用一个自定义错误类型中的数据,你
    // 需要通过类型断言来得到这个错误类型的实例。
    _, e := f2(42)
    if ae, ok := e.(*argError); ok {
        fmt.Println(ae.arg)
        fmt.Println(ae.prob)
    }
}

我的疑问:1. f2(42)返回一个&argError类型的值同时改变argError结构体中的属性值,但是,这个Error()方法是什么时候调用的?因为最后输出的是一个字符串; 2. main函数中最后if语句 ae = e.(argError)看不懂(类型断言我已经谷歌过了)。其实说到底是对& 和 了解的不够,只知道&是取地址,是解析地址,一般用&引用就可以改变值,用就相当于本体。求高人解答。


  1. f2 的第一个 return 的返回值有两个。其中第二值是一个指针,指向的是类(叫结构体也可以) argError 的一个实例对象。由于 argError 类实现了 Error 方法,按照 Golang 的 duck type 原则,这个类是 error 接口的实现。

Error() 方法被调用是因为:

fmt.Println("f2 failed:", e)

追踪 golang 的 Println 方法。在 print.go 文件 724 行调用了 Error() 。当对象实现了 error 接口时 Println 会尝试调用参数对象的 Error() 方法,取其返回值当作对象的值打印到标准输出。

  1. 这个叫做 type assertion。类型断言,官方文档在这里 =〉 https://golang.org/ref/spec#Type_assertions
    当类型断言用在赋值语句的右侧时 (例如 x, ok := a.(T) ) The value of ok is true if the assertion holds. Otherwise it is false and the value of v is the zero value for type T. No run-time panic occurs in this case. 当 e 的值为一个指针且指向非 nil 的 argError 值时,打印这该值的各个属性。

------------------更新----------------

  1. error 是一个接口不是类,golang build in 的。

【热门文章】
【热门文章】