make loop

This commit is contained in:
chai2010
2015-12-26 20:05:30 +08:00
parent 82ec0c025d
commit e15e88dad7
74 changed files with 207 additions and 207 deletions

View File

@@ -1,8 +1,8 @@
## 1.3. 査找重複的行
文件拷貝、文件打印、文件蒐索、文件排序、文件統計類的程序一般都會有比較相似的程序結構:一個處理輸入的循環,在每一個輸入元素上執行計算處理,在處理的同時或者處理完成之後進行結果輸。我們會展示一個叫dup程序的三個版本這個程序的靈感來自於linux的uniq命令我們的程序將會找到相鄰的重複的行。這個程序提供的模式可以很方便地被脩改來完成不同的需求。
文件拷貝、文件打印、文件蒐索、文件排序、文件統計類的程序一般都會有比較相似的程序結構:一個處理輸入的循環,在每一個輸入元素上執行計算處理,在處理的同時或者處理完成之後進行結果輸。我們會展示一個叫dup程序的三個版本這個程序的靈感來自於linux的uniq命令我們的程序將會找到相鄰的重複的行。這個程序提供的模式可以很方便地被脩改來完成不同的需求。
第一個版本的dup會輸標準輸入流中的現多次的行,在行內容前會有其現次數的計數。這個程序將引入if表達式map內置數據結構和bufio的package。
第一個版本的dup會輸標準輸入流中的現多次的行,在行內容前會有其現次數的計數。這個程序將引入if表達式map內置數據結構和bufio的package。
```go
gopl.io/ch1/dup1
@@ -47,7 +47,7 @@ counts[line] = counts[line] + 1
在這里我們又用了一個range的循環來打印結果這次range是被用在map這個數據結構之上。這一次的情況和上次比較類似range會返迴兩個值一個key和在map對應這個key的value。對map進行range循環時其迭代順序是不確定的從實踐來看很可能每次運行都會有不一樣的結果譯註這是Go語言的設計者有意爲之的因爲其底層實現不保證插入順序和遍歷順序一致也希望程序員不要依賴遍歷時的順序所以榦脆直接在遍歷的時候做了隨機化處理醉了。補充好像説隨機序可以防止某種類型的攻擊雖然不太明白但是感覺還蠻厲害的來避免程序員在業務中依賴遍歷時的順序。
然後輪到我們例子中的bufio這個package了這個package主要的目的是幫助我們更方便有效地處理程序的輸入和輸。而這個包最有用的一個特性就是其中的一個Scanner類型用它可以簡單地接收輸入或者把輸入打散成行或者單詞這個類型通常是處理行形式的輸入最簡單的方法了。
然後輪到我們例子中的bufio這個package了這個package主要的目的是幫助我們更方便有效地處理程序的輸入和輸。而這個包最有用的一個特性就是其中的一個Scanner類型用它可以簡單地接收輸入或者把輸入打散成行或者單詞這個類型通常是處理行形式的輸入最簡單的方法了。
本程序中用了一個短變量聲明來創建一個buffio.Scanner對象
@@ -57,9 +57,9 @@ input := bufio.NewScanner(os.Stdin)
scanner對象可以從程序的標準輸入中讀取內容。對input.Scanner的每一次調用都會調入一個新行併且會自動將其行末的換行符去掉其結果可以用input.Text()得到。Scan方法在讀到了新行的時候會返迴true而在沒有新行被讀入時會返迴false。
例子中還有一個fmt.Printf這個函數和C繫的其它語言里的那個printf函數差不多都是格式化輸的方法。fmt.Printf的第一個參數卽是輸內容的格式規約,每一個參數如何格式化是取決於在格式化字符串里現的“轉換字符”,這個字符串是跟着%號後的一個字母。比如%d表示以一個整數的形式來打印一個變量而%s則表示以string形式來打印一個變量。
例子中還有一個fmt.Printf這個函數和C繫的其它語言里的那個printf函數差不多都是格式化輸的方法。fmt.Printf的第一個參數卽是輸內容的格式規約,每一個參數如何格式化是取決於在格式化字符串里現的“轉換字符”,這個字符串是跟着%號後的一個字母。比如%d表示以一個整數的形式來打印一個變量而%s則表示以string形式來打印一個變量。
Printf有一大堆這種轉換Go語言程序員把這些叫做verb動詞。下面的表格列了常用的動詞,當然了不是全部,但基本也夠用了。
Printf有一大堆這種轉換Go語言程序員把這些叫做verb動詞。下面的表格列了常用的動詞,當然了不是全部,但基本也夠用了。
```
%d int變量
@@ -69,12 +69,12 @@ Printf有一大堆這種轉換Go語言程序員把這些叫做verb動詞
%c rune (Unicode碼點)Go語言里特有的Unicode字符類型
%s string
%q 帶雙引號的字符串 "abc" 或 帶單引號的 rune 'c'
%v 會將任意變量以易讀的形式打印
%v 會將任意變量以易讀的形式打印
%T 打印變量的類型
%% 字符型百分比標誌(%符號本身,沒有其他操作)
```
dup1中的程序還包含了一個\t和\n的格式化字符串。在字符串中會以這些特殊的轉義字符來表示不可見字符。Printf默認不會在輸內容後加上換行符。按照慣例用來格式化的函數都會在末尾以f字母結尾譯註f後綴對應format或fmt縮寫比如log.Printffmt.Errorf同時還有一繫列對應以ln結尾的函數譯註ln後綴對應line縮寫這些函數默認以%v來格式化他們的參數併且會在輸結束後在最後自動加上一個換行符。
dup1中的程序還包含了一個\t和\n的格式化字符串。在字符串中會以這些特殊的轉義字符來表示不可見字符。Printf默認不會在輸內容後加上換行符。按照慣例用來格式化的函數都會在末尾以f字母結尾譯註f後綴對應format或fmt縮寫比如log.Printffmt.Errorf同時還有一繫列對應以ln結尾的函數譯註ln後綴對應line縮寫這些函數默認以%v來格式化他們的參數併且會在輸結束後在最後自動加上一個換行符。
許多程序從標準輸入中讀取數據像上面的例子那樣。除此之外還可能從一繫列的文件中讀取。下一個dup程序就是從標準輸入中讀到一些文件名用os.Open函數來打開每一個文件穫取內容的。
@@ -124,7 +124,7 @@ func countLines(f *os.File, counts map[string]int) {
os.Open函數會返迴兩個值。第一個值是一個打開的文件類型(*os.File)這個對象在下面的程序中被Scanner讀取。
os.Open返迴的第二個值是一個Go語言內置的error類型。如果這個error和內置值的nil譯註相當於其它語言里的NULL相等的話説明文件被成功的打開了。之後文件被讀取一直到文件的最後文件的Close方法關閉該文件併釋放相應的占用一切資源。另一方面如果err的值不是nil的話那説明在打開文件的時候了某種錯誤。這種情況下error類型的值會描述具體的問題。我們例子里的簡單錯誤處理會在標準錯誤流中用Fprintf和%v來格式化該錯誤字符串。然後繼續處理下一個文件continue語句會直接跳過之後的語句直接開始執行下一個循環迭代。
os.Open返迴的第二個值是一個Go語言內置的error類型。如果這個error和內置值的nil譯註相當於其它語言里的NULL相等的話説明文件被成功的打開了。之後文件被讀取一直到文件的最後文件的Close方法關閉該文件併釋放相應的占用一切資源。另一方面如果err的值不是nil的話那説明在打開文件的時候了某種錯誤。這種情況下error類型的值會描述具體的問題。我們例子里的簡單錯誤處理會在標準錯誤流中用Fprintf和%v來格式化該錯誤字符串。然後繼續處理下一個文件continue語句會直接跳過之後的語句直接開始執行下一個循環迭代。
我們在本書中早期的例子中做了比較詳盡的錯誤處理當然了在實際編碼過程中像os.Open這類的函數是一定要檢査其返迴的error值的爲了減少例子程序的代碼量我們姑且簡化掉這些不太可能返迴錯誤的處理邏輯。後面的例子里我們會跳過錯誤檢査。在5.4節中我們會對錯誤處理做更詳細的闡述。
@@ -167,9 +167,9 @@ func main() {
}
```
ReadFile函數返迴一個byte的slice這個slice必鬚被轉換爲string之後能夠用string.Split方法來進行處理。我們在3.5.4節中會更詳細地講解string和byte slice字節數組
ReadFile函數返迴一個byte的slice這個slice必鬚被轉換爲string之後能夠用string.Split方法來進行處理。我們在3.5.4節中會更詳細地講解string和byte slice字節數組
在更底層一些的地方bufio.Scannerioutil.ReadFile和ioutil.WriteFile使用的是*os.File的Read和Write方法不過一般程序員併不需要去直接了解到其底層實現細節在bufio和io/ioutil包中提供的方法已經足夠好用。
**練習 1.4** 脩改dup2使其可以打印重複的行分别現在哪些文件。
**練習 1.4** 脩改dup2使其可以打印重複的行分别現在哪些文件。