ch13: fix code path

pull/1/head
chai2010 2016-01-20 22:50:43 +08:00
parent 8b7a8ea019
commit 617ef87432
3 changed files with 8 additions and 16 deletions

View File

@ -18,9 +18,8 @@ fmt.Printf("%#016x\n", Float64bits(1.0)) // "0x3ff0000000000000"
許多將unsafe.Pointer指針轉爲原生數字然後再轉迴爲unsafe.Pointer類型指針的操作也是不安全的。比如下面的例子需要將變量x的地址加上b字段地址偏移量轉化爲`*int16`類型指針然後通過該指針更新x.b
<u><i>gopl.io/ch13/unsafeptr</i></u>
```Go
//gopl.io/ch13/unsafeptr
var x struct {
a bool
b int16
@ -37,10 +36,10 @@ fmt.Println(x.b) // "42"
上面的寫法盡管很繁瑣但在這里併不是一件壞事因爲這些功能應該很謹慎地使用。不要試圖引入一個uintptr類型的臨時變量因爲它可能會破壞代碼的安全性譯註這是眞正可以體會unsafe包爲何不安全的例子。下面段代碼是錯誤的
```Go
// NOTE: subtly incorrect!
tmp := uintptr(unsafe.Pointer(&x)) + unsafe.Offsetof(x.b)
pb := (*int16)(unsafe.Pointer(tmp))
*pb = 42
// NOTE: subtly incorrect!
tmp := uintptr(unsafe.Pointer(&x)) + unsafe.Offsetof(x.b)
pb := (*int16)(unsafe.Pointer(tmp))
*pb = 42
```
産生錯誤的原因很微妙。有時候垃圾迴收器會移動一些變量以降低內存碎片等問題。這類垃圾迴收器被稱爲移動GC。當一個變量被移動所有的保存改變量舊地址的指針必須同時被更新爲變量移動後的新地址。從垃圾收集器的視角來看一個unsafe.Pointer是一個指向變量的指針因此當變量被移動是對應的指針也必須被更新但是uintptr類型的臨時變量隻是一個普通的數字所以其值不應該被改變。上面錯誤的代碼因爲引入一個非指針的臨時變量tmp導致垃圾收集器無法正確識别這個是一個指向變量x的指針。當第二個語句執行時變量x可能已經被轉移這時候臨時變量tmp也就不再是現在的`&x.b`地址。第三個向之前無效地址空間的賦值語句將徹底摧譭整個程序!
@ -66,5 +65,3 @@ func (Value) Pointer() uintptr
func (Value) UnsafeAddr() uintptr
func (Value) InterfaceData() [2]uintptr // (index 1)
```

View File

@ -22,8 +22,8 @@ fmt.Println(reflect.DeepEqual(c, d)) // "false"
我們希望在這里實現一個自己的Equal函數用於比較類型的值。和DeepEqual函數類似的地方是它也是基於slice和map的每個元素進行遞歸比較不同之處是它將nil值的slicemap類似和非nil值但是空的slice視作相等的值。基礎部分的比較可以基於reflect包完成和12.3章的Display函數的實現方法類似。同樣我們也定義了一個內部函數equal用於內部的遞歸比較。讀者目前不用關心seen參數的具體含義。對於每一對需要比較的x和yequal函數首先檢測它們是否都有效或都無效然後檢測它們是否是相同的類型。剩下的部分是一個鉅大的switch分支用於相同基礎類型的元素比較。因爲頁面空間的限製我們省略了一些相似的分支。
<u><i>gopl.io/ch13/equal</i></u>
```Go
gopl.io/ch13/equal
func equal(x, y reflect.Value, seen map[comparison]bool) bool {
if !x.IsValid() || !y.IsValid() {
return x.IsValid() == y.IsValid()
@ -125,5 +125,3 @@ fmt.Println(Equal(a, c)) // "false"
**練習 13.1** 定義一個深比較函數,對於十億以內的數字比較,忽略類型差異。
**練習 13.2** 編寫一個函數,報告其參數是否循環數據結構。

View File

@ -41,9 +41,8 @@ bzip2壓縮算法是基於優雅的Burrows-Wheeler變換算法運行速度
我們可以在Go代碼中直接調用BZ2_bzCompressInit和BZ2_bzCompressEnd但是對於BZ2_bzCompress我們將定義一個C語言的包裝函數用它完成眞正的工作。下面是C代碼對應一個獨立的文件。
<u><i>gopl.io/ch13/bzip</i></u>
```C
gopl.io/ch13/bzip
/* This file is gopl.io/ch13/bzip/bzip2.c, */
/* a simple wrapper for libbzip2 suitable for cgo. */
#include <bzlib.h>
@ -168,9 +167,8 @@ func (w *writer) Close() error {
下面的bzipper程序使用我們自己包實現的bzip2壓縮命令。它的行爲和許多Unix繫統的bzip2命令類似。
<u><i>gopl.io/ch13/bzipper</i></u>
```Go
gopl.io/ch13/bzipper
// Bzipper reads input, bzip2-compresses it, and writes it out.
package main
@ -211,4 +209,3 @@ $ ./bzipper < /usr/share/dict/words | bunzip2 | sha256sum
**練習 13.3** 使用sync.Mutex以保證bzip2.writer在多個goroutines中被併發調用是安全的。
**練習 13.4** 因爲C庫依賴的限製。 使用os/exec包啟動/bin/bzip2命令作爲一個子進程提供一個純Go的bzip.NewWriter的替代實現譯註雖然是純Go實現但是運行時將依賴/bin/bzip2命令其他操作繫統可能無法運行