diff --git a/ch5/ch5-02.md b/ch5/ch5-02.md index 9a06d21..6b0b639 100644 --- a/ch5/ch5-02.md +++ b/ch5/ch5-02.md @@ -6,8 +6,8 @@ 例子中調用golang.org/x/net/html的部分api如下所示。html.Parse函數讀入一組bytes.解析後,返迴html.node類型的HTML頁面樹狀結構根節點。HTML擁有很多類型的結點如text(文本),commnets(註釋)類型,在下面的例子中,我們 隻關註< name key='value' >形式的結點。 +golang.org/x/net/html ```Go -golang.org/x/net/html package html type Node struct { @@ -37,8 +37,8 @@ func Parse(r io.Reader) (*Node, error) main函數解析HTML標準輸入,通過遞歸函數visit獲得links(鏈接),併打印出這些links: +gopl.io/ch5/findlinks1 ```Go -gopl.io/ch5/findlinks1 // Findlinks1 prints the links in an HTML document read from standard input. package main @@ -106,8 +106,8 @@ http://www.google.com/intl/en/policies/privacy/ 在函數outline中,我們通過遞歸的方式遍歷整個HTML結點樹,併輸出樹的結構。在outline內部,每遇到一個HTML元素標籤,就將其入棧,併輸出。 +gopl.io/ch5/outline ```Go -gopl.io/ch5/outline func main() { doc, err := html.Parse(os.Stdin) if err != nil { diff --git a/ch5/ch5-03.md b/ch5/ch5-03.md index 2b6e815..8bfe324 100644 --- a/ch5/ch5-03.md +++ b/ch5/ch5-03.md @@ -4,8 +4,8 @@ 下面的程序是findlinks的改進版本。脩改後的findlinks可以自己發起HTTP請求,這樣我們就不必再運行fetch。因爲HTTP請求和解析操作可能會失敗,因此findlinks聲明了2個返迴值:鏈接列表和錯誤信息。一般而言,HTML的解析器可以處理HTML頁面的錯誤結點,構造出HTML頁面結構,所以解析HTML很少失敗。這意味着如果findlinks函數失敗了,很可能是由於I/O的錯誤導致的。 +gopl.io/ch5/findlinks2 ```Go -gopl.io/ch5/findlinks2 func main() { for _, url := range os.Args[1:] { links, err := findLinks(url) diff --git a/ch5/ch5-04-1.md b/ch5/ch5-04-1.md index 1cd15ab..8c9c1ba 100644 --- a/ch5/ch5-04-1.md +++ b/ch5/ch5-04-1.md @@ -37,8 +37,8 @@ genesis: crashed: no parachute: G-switch failed: bad relay orientation 讓我們來看看處理錯誤的第二種策略。如果錯誤的發生是偶然性的,或由不可預知的問題導致的。一個明智的選擇是重新嚐試失敗的操作。在重試時,我們需要限製重試的時間間隔或重試的次數,防止無限製的重試。 +gopl.io/ch5/wait ```Go -gopl.io/ch5/wait // WaitForServer attempts to contact the server of a URL. // It tries for one minute using exponential back-off. // It reports an error if all attempts fail. diff --git a/ch5/ch5-05.md b/ch5/ch5-05.md index 1cf680f..0c2d142 100644 --- a/ch5/ch5-05.md +++ b/ch5/ch5-05.md @@ -47,8 +47,8 @@ 5.2節的findLinks函數使用了輔助函數visit,遍歷和操作了HTML頁面的所有結點。使用函數值,我們可以將遍歷結點的邏輯和操作結點的邏輯分離,使得我們可以複用遍歷的邏輯,從而對結點進行不同的操作。 +gopl.io/ch5/outline2 ```Go -gopl.io/ch5/outline2 // forEachNode針對每個結點x,都會調用pre(x)和post(x)。 // pre和post都是可選的。 // 遍歷孩子結點之前,pre被調用 diff --git a/ch5/ch5-06.md b/ch5/ch5-06.md index 42446a1..9be5819 100644 --- a/ch5/ch5-06.md +++ b/ch5/ch5-06.md @@ -10,8 +10,8 @@ strings.Map(func(r rune) rune { return r + 1 }, "HAL-9000") 更爲重要的是,通過這種方式定義的函數可以訪問完整的詞法環境(lexical environment),這意味着在函數中定義的內部函數可以引用該函數的變量,如下例所示: +gopl.io/ch5/squares ```Go -gopl.io/ch5/squares // squares返迴一個匿名函數。 // 該匿名函數每次被調用時都會返迴下一個數的平方。 func squares() func() int { @@ -38,8 +38,8 @@ squares的例子證明,函數值不僅僅是一串代碼,還記録了狀態 接下來,我們討論一個有點學術性的例子,考慮這樣一個問題:給定一些計算機課程,每個課程都有前置課程,隻有完成了前置課程才可以開始當前課程的學習;我們的目標是選擇出一組課程,這組課程必須確保按順序學習時,能全部被完成。每個課程的前置課程如下: +gopl.io/ch5/toposort ```Go -gopl.io/ch5/toposort // prereqs記録了每個課程的前置課程 var prereqs = map[string][]string{ "algorithms": {"data structures"}, @@ -121,8 +121,8 @@ visitAll := func(items []string) { 讓我們迴到findLinks這個例子。我們將代碼移動到了links包下,將函數重命名爲Extract,在第八章我們會再次用到這個函數。新的匿名函數被引入,用於替換原來的visit函數。該匿名函數負責將新連接添加到切片中。在Extract中,使用forEachNode遍歷HTML頁面,由於Extract隻需要在遍歷結點前操作結點,所以forEachNode的post參數被傳入nil。 +gopl.io/ch5/links ```Go -gopl.io/ch5/links // Package links provides a link-extraction function. package links import ( @@ -172,8 +172,8 @@ func Extract(url string) ([]string, error) { 下面的函數實現了廣度優先算法。調用者需要輸入一個初始的待訪問列表和一個函數f。待訪問列表中的每個元素被定義爲string類型。廣度優先算法會爲每個元素調用一次f。每次f執行完畢後,會返迴一組待訪問元素。這些元素會被加入到待訪問列表中。當待訪問列表中的所有元素都被訪問後,breadthFirst函數運行結束。爲了避免同一個元素被訪問兩次,代碼中維護了一個map。 +gopl.io/ch5/findlinks3 ```Go -gopl.io/ch5/findlinks3 // breadthFirst calls f for each item in the worklist. // Any items returned by f are added to the worklist. // f is called at most once for each item. @@ -219,7 +219,7 @@ func main() { 讓我們從 https://golang.org 開始,下面是程序的輸出結果: -```bash +``` $ go build gopl.io/ch5/findlinks3 $ ./findlinks3 https://golang.org https://golang.org/ diff --git a/ch5/ch5-07.md b/ch5/ch5-07.md index c45e819..1be6aff 100644 --- a/ch5/ch5-07.md +++ b/ch5/ch5-07.md @@ -4,8 +4,8 @@ 在聲明可變參數函數時,需要在參數列表的最後一個參數類型之前加上省略符號“...”,這表示該函數會接收任意數量的該類型參數。 +gopl.io/ch5/sum ```Go -gopl.io/ch5/sum func sum(vals...int) int { total := 0 for _, val := range vals { @@ -14,6 +14,7 @@ func sum(vals...int) int { return total } ``` + sum函數返迴任意個int型參數的和。在函數體中,vals被看作是類型爲[] int的切片。sum可以接收任意數量的int型參數: ```Go diff --git a/ch5/ch5-08.md b/ch5/ch5-08.md index 59eb470..02e4e68 100644 --- a/ch5/ch5-08.md +++ b/ch5/ch5-08.md @@ -4,8 +4,8 @@ 下面的例子獲取HTML頁面併輸出頁面的標題。title函數會檢査服務器返迴的Content-Type字段,如果發現頁面不是HTML,將終止函數運行,返迴錯誤。 +gopl.io/ch5/title1 ```Go -gopl.io/ch5/title1 func title(url string) error { resp, err := http.Get(url) if err != nil { @@ -34,7 +34,7 @@ func title(url string) error { 下面展示了運行效果: -```powershell +``` $ go build gopl.io/ch5/title1 $ ./title1 http://gopl.io The Go Programming Language @@ -50,8 +50,8 @@ resp.Body.close調用了多次,這是爲了確保title在所有執行路徑下 defer語句經常被用於處理成對的操作,如打開、關閉、連接、斷開連接、加鎖、釋放鎖。通過defer機製,不論函數邏輯多複雜,都能保證在任何執行路徑下,資源被釋放。釋放資源的defer應該直接跟在請求資源的語句後。在下面的代碼中,一條defer語句替代了之前的所有resp.Body.Close +gopl.io/ch5/title2 ```Go -gopl.io/ch5/title2 func title(url string) error { resp, err := http.Get(url) if err != nil { @@ -73,8 +73,8 @@ func title(url string) error { 在處理其他資源時,也可以采用defer機製,比如對文件的操作: +io/ioutil ```Go -io/ioutil package ioutil func ReadFile(filename string) ([]byte, error) { f, err := os.Open(filename) @@ -100,8 +100,8 @@ func lookup(key string) int { 調試複雜程序時,defer機製也常被用於記録何時進入和退出函數。下例中的bigSlowOperation函數,直接調用trace記録函數的被調情況。bigSlowOperation被調時,trace會返迴一個函數值,該函數值會在bigSlowOperation退出時被調用。通過這種方式, 我們可以隻通過一條語句控製函數的入口和所有的出口,甚至可以記録函數的運行時間,如例子中的start。需要註意一點:不要忘記defer語句後的圓括號,否則本該在進入時執行的操作會在退出時執行,而本該在退出時執行的,永遠不會被執行。 +gopl.io/ch5/trace ```Go -gopl.io/ch5/trace func bigSlowOperation() { defer trace("bigSlowOperation")() // don't forget the extra parentheses @@ -195,8 +195,8 @@ func doFile(filename string) error { 下面的代碼是fetch(1.5節)的改進版,我們將http響應信息寫入本地文件而不是從標準輸出流輸出。我們通過path.Base提出url路徑的最後一段作爲文件名。 +gopl.io/ch5/fetch ```Go -gopl.io/ch5/fetch // Fetch downloads the URL and returns the // name and length of the local file. func fetch(url string) (filename string, n int64, err error) {