在学习本书前半部分时,我们经常在一个表达式返回 2 个参数时使用这种模式:,ok,第一个参数是一个值或者 nil,第二个参数是 true/false 或者一个错误 error。在一个需要赋值的 if 条件语句中,使用这种模式去检测第二个参数值会让代码显得优雅简洁。这种模式在 go 语言编码规范中非常重要。下面总结了所有使用这种模式的例子:
(1)在函数返回时检测错误(参考第 5.2 小节):
value, err := pack1.Func1(param1)if err != nil {fmt.Printf(“Error %s in pack1.Func1 with parameter %v”, err.Error(), param1)return err}// 函数Func1没有错误:Process(value)e.g.: os.Open(file) strconv.Atoi(str)
这段代码中的函数将错误返回给它的调用者,当函数执行成功时,返回的错误是 nil,所以使用这种写法:
func SomeFunc() error {…if value, err := pack1.Func1(param1); err != nil {…return err}…return nil}
这种模式也常用于通过 defer 使程序从 panic 中恢复执行(参考 第 17.2(4)小节)。
要实现简洁的错误检测代码,更好的方式是使用闭包,参考 第 16.10.2 小节
(2)检测映射中是否存在一个键值(参考第 8.2 小节):key1 在映射 map1 中是否有值?
if value, isPresent = map1[key1]; isPresent {Process(value)}// key1不存在…
(3)检测一个接口类型变量 varI 是否包含了类型 T:类型断言(参考第 11.3 小节):
if value, ok := varI.(T); ok {Process(value)}// 接口类型varI没有包含类型T
(4)检测一个通道 ch 是否关闭(参考第 14.3 小节):
for input := range ch {Process(input)}
或者:
for {if input, open := <-ch; !open {break // 通道是关闭的} else {Process(input)}}
