Compare commits

...

2 Commits

Author SHA1 Message Date
github-actions[bot]
040421dfcc deploy: c58743acce 2025-09-03 09:00:18 +00:00
github-actions[bot]
b0bec4a36b deploy: cc738922ed 2025-09-03 08:59:41 +00:00
2 changed files with 2 additions and 2 deletions

View File

@@ -582,7 +582,7 @@ any = new(bytes.Buffer)
</code></pre>
<p>尽管不是很明显从本书最早的例子中我们就已经在使用空接口类型。它允许像fmt.Println或者5.7章中的errorf函数接受任何类型的参数。</p>
<p>对于创建的一个interface{}值持有一个booleanfloatstringmappointer或者任意其它的类型我们当然不能直接对它持有的值做操作因为interface{}没有任何方法。我们会在7.10章中学到一种用类型断言来获取interface{}中值的方法。</p>
<p>因为接口与实现只依赖于判断两个类型的方法,所以没有必要定义一个具体类型和它实现的接口之间的关系。也就是说,有意地在文档里说明或者程序上断言这种关系偶尔是有用的,但程序上不强制这么做。下面的定义在编译期断言一个<code>*bytes.Buffer</code>的值实现了io.Writer接口类型:</p>
<p>因为接口与实现只依赖于判断两个类型的方法,所以没有必要定义一个具体类型和它实现的接口之间的关系。也就是说,有意地在文档里说明或者程序上断言这种关系偶尔是有用的,但程序上不强制这么做。这种写法还可用于让编译器在编译期确保某个类型确实满足接口要求,从而提前发现实现遗漏或接口变更导致的不匹配问题。下面的定义在编译期断言一个<code>*bytes.Buffer</code>的值实现了io.Writer接口类型:</p>
<pre><code class="language-go">// *bytes.Buffer must satisfy io.Writer
var w io.Writer = new(bytes.Buffer)
</code></pre>

View File

@@ -622,7 +622,7 @@ func f(out io.Writer) {
<p>当main函数调用函数f时它给f函数的out参数赋了一个*bytes.Buffer的空指针所以out的动态值是nil。然而它的动态类型是*bytes.Buffer意思就是out变量是一个包含空指针值的非空接口如图7.5所以防御性检查out!=nil的结果依然是true。</p>
<p><img src="../images/ch7-05.png" alt=""></p>
<p>动态分配机制依然决定(*bytes.Buffer).Write的方法会被调用但是这次的接收者的值是nil。对于一些如*os.File的类型nil是一个有效的接收者§6.2.1),但是*bytes.Buffer类型不在这些种类中。这个方法会被调用但是当它尝试去获取缓冲区时会发生panic。</p>
<p>问题在于尽管一个nil的*bytes.Buffer指针有实现这个接口的方法它也不满足这个接口具体的行为上的要求。特别是这个调用违反了(*bytes.Buffer).Write方法的接收者非空的隐含先条件所以将nil指针赋给这个接口是错误的。解决方案就是将main函数中的变量buf的类型改为io.Writer因此可以避免一开始就将一个不完整的值赋值给这个接口</p>
<p>问题在于尽管一个nil的*bytes.Buffer指针有实现这个接口的方法它也不满足这个接口具体的行为上的要求。特别是这个调用违反了(*bytes.Buffer).Write方法的接收者非空的隐含先条件所以将nil指针赋给这个接口是错误的。解决方案就是将main函数中的变量buf的类型改为io.Writer因此可以避免一开始就将一个不完整的值赋值给这个接口</p>
<pre><code class="language-go">var buf io.Writer
if debug {
buf = new(bytes.Buffer) // enable collection of output