gopl-zh.github.com/ch7/ch7-10.md

58 lines
3.3 KiB
Markdown
Raw Normal View History

2016-02-15 03:06:34 +00:00
## 7.10. 类型断言
2016-01-18 02:52:21 +00:00
2016-02-15 03:06:34 +00:00
类型断言是一个使用在接口值上的操作。语法上它看起来像x.(T)被称为断言类型这里x表示一个接口的类型和T表示一个类型。一个类型断言检查它操作对象的动态类型是否和断言的类型匹配。
2015-12-09 07:45:11 +00:00
2016-02-15 03:06:34 +00:00
这里有两种可能。第一种如果断言的类型T是一个具体类型然后类型断言检查x的动态类型是否和T相同。如果这个检查成功了类型断言的结果是x的动态值当然它的类型是T。换句话说具体类型的类型断言从它的操作对象中获得具体的值。如果检查失败接下来这个操作会抛出panic。例如
2016-01-18 02:52:21 +00:00
2016-01-18 02:36:12 +00:00
```go
var w io.Writer
w = os.Stdout
f := w.(*os.File) // success: f == os.Stdout
c := w.(*bytes.Buffer) // panic: interface holds *os.File, not *bytes.Buffer
```
2016-01-18 02:52:21 +00:00
2016-02-15 03:06:34 +00:00
第二种如果相反断言的类型T是一个接口类型然后类型断言检查是否x的动态类型满足T。如果这个检查成功了动态值没有获取到这个结果仍然是一个有相同类型和值部分的接口值但是结果有类型T。换句话说对一个接口类型的类型断言改变了类型的表述方式改变了可以获取的方法集合通常更大但是它保护了接口值内部的动态类型和值的部分。
2016-01-18 02:36:12 +00:00
2016-02-15 03:06:34 +00:00
在下面的第一个类型断言后w和rw都持有os.Stdout因此它们每个有一个动态类型*os.File但是变量w是一个io.Writer类型只对外公开出文件的Write方法然而rw变量也只公开它的Read方法。
2016-01-18 02:52:21 +00:00
2016-01-18 02:36:12 +00:00
```go
var w io.Writer
w = os.Stdout
rw := w.(io.ReadWriter) // success: *os.File has both Read and Write
w = new(ByteCounter)
rw = w.(io.ReadWriter) // panic: *ByteCounter has no Read method
```
2016-01-18 02:52:21 +00:00
2016-02-15 03:06:34 +00:00
如果断言操作的对象是一个nil接口值那么不论被断言的类型是什么这个类型断言都会失败。我们几乎不需要对一个更少限制性的接口类型更少的方法集合做断言因为它表现的就像赋值操作一样除了对于nil接口值的情况。
2016-01-18 02:52:21 +00:00
2016-01-18 02:36:12 +00:00
```go
w = rw // io.ReadWriter is assignable to io.Writer
w = rw.(io.Writer) // fails only if rw == nil
```
2016-01-18 02:52:21 +00:00
2016-02-15 03:06:34 +00:00
经常地我们对一个接口值的动态类型是不确定的并且我们更愿意去检验它是否是一些特定的类型。如果类型断言出现在一个预期有两个结果的赋值操作中例如如下的定义这个操作不会在失败的时候发生panic但是代替地返回一个额外的第二个结果这个结果是一个标识成功的布尔值
2016-01-18 02:52:21 +00:00
2016-01-18 02:36:12 +00:00
```go
var w io.Writer = os.Stdout
f, ok := w.(*os.File) // success: ok, f == os.Stdout
b, ok := w.(*bytes.Buffer) // failure: !ok, b == nil
```
2016-01-18 02:52:21 +00:00
2016-02-15 03:06:34 +00:00
第二个结果常规地赋值给一个命名为ok的变量。如果这个操作失败了那么ok就是false值第一个结果等于被断言类型的零值在这个例子中就是一个nil的*bytes.Buffer类型。
2016-01-18 02:36:12 +00:00
2016-02-15 03:06:34 +00:00
这个ok结果经常立即用于决定程序下面做什么。if语句的扩展格式让这个变的很简洁
2016-01-18 02:52:21 +00:00
2016-01-18 02:36:12 +00:00
```go
if f, ok := w.(*os.File); ok {
2016-01-18 02:52:21 +00:00
// ...use f...
2016-01-18 02:36:12 +00:00
}
```
2016-01-18 02:52:21 +00:00
2016-02-15 03:06:34 +00:00
当类型断言的操作对象是一个变量,你有时会看见原来的变量名重用而不是声明一个新的本地变量,这个重用的变量会覆盖原来的值,如下面这样:
2016-01-18 02:52:21 +00:00
2016-01-18 02:36:12 +00:00
```go
if w, ok := w.(*os.File); ok {
2016-01-18 02:52:21 +00:00
// ...use w...
2016-01-18 02:36:12 +00:00
}
```