2015-12-09 07:57:17 +00:00
<!DOCTYPE HTML>
< html lang = "zh-tw" >
< head >
< meta charset = "UTF-8" >
< meta http-equiv = "X-UA-Compatible" content = "IE=edge" / >
< title > 査找重復的行 | Go编程语言< / title >
< meta content = "text/html; charset=utf-8" http-equiv = "Content-Type" >
< meta name = "description" content = "" >
< meta name = "generator" content = "GitBook 2.5.2" >
< meta name = "HandheldFriendly" content = "true" / >
< meta name = "viewport" content = "width=device-width, initial-scale=1, user-scalable=no" >
< meta name = "apple-mobile-web-app-capable" content = "yes" >
< meta name = "apple-mobile-web-app-status-bar-style" content = "black" >
< link rel = "apple-touch-icon-precomposed" sizes = "152x152" href = "../gitbook/images/apple-touch-icon-precomposed-152.png" >
< link rel = "shortcut icon" href = "../gitbook/images/favicon.ico" type = "image/x-icon" >
< link rel = "stylesheet" href = "../gitbook/style.css" >
< link rel = "stylesheet" href = "../gitbook/plugins/gitbook-plugin-highlight/website.css" >
< link rel = "stylesheet" href = "../gitbook/plugins/gitbook-plugin-search/search.css" >
< link rel = "stylesheet" href = "../gitbook/plugins/gitbook-plugin-fontsettings/website.css" >
< link rel = "next" href = "../ch1/ch1-04.html" / >
< link rel = "prev" href = "../ch1/ch1-02.html" / >
< / head >
< body >
2015-12-14 04:08:47 +00:00
< div class = "book" data-level = "1.3" data-chapter-title = "査找重復的行" data-filepath = "ch1/ch1-03.md" data-basepath = ".." data-revision = "Mon Dec 14 2015 11:30:54 GMT+0800 (中国标准时间)" >
2015-12-09 07:57:17 +00:00
< div class = "book-summary" >
< nav role = "navigation" >
< ul class = "summary" >
< li class = "chapter " data-level = "0" data-path = "index.html" >
< a href = "../index.html" >
< i class = "fa fa-check" > < / i >
前言
< / a >
< ul class = "articles" >
< li class = "chapter " data-level = "0.1" data-path = "ch0/ch0-01.html" >
< a href = "../ch0/ch0-01.html" >
< i class = "fa fa-check" > < / i >
< b > 0.1.< / b >
Go語言起源
< / a >
< / li >
< li class = "chapter " data-level = "0.2" data-path = "ch0/ch0-02.html" >
< a href = "../ch0/ch0-02.html" >
< i class = "fa fa-check" > < / i >
< b > 0.2.< / b >
Go語言項目
< / a >
< / li >
< li class = "chapter " data-level = "0.3" data-path = "ch0/ch0-03.html" >
< a href = "../ch0/ch0-03.html" >
< i class = "fa fa-check" > < / i >
< b > 0.3.< / b >
本書的組織
< / a >
< / li >
< li class = "chapter " data-level = "0.4" data-path = "ch0/ch0-04.html" >
< a href = "../ch0/ch0-04.html" >
< i class = "fa fa-check" > < / i >
< b > 0.4.< / b >
更多的信息
< / a >
< / li >
< li class = "chapter " data-level = "0.5" data-path = "ch0/ch0-05.html" >
< a href = "../ch0/ch0-05.html" >
< i class = "fa fa-check" > < / i >
< b > 0.5.< / b >
緻謝
< / a >
< / li >
< / ul >
< / li >
< li class = "chapter " data-level = "1" data-path = "ch1/ch1.html" >
< a href = "../ch1/ch1.html" >
< i class = "fa fa-check" > < / i >
< b > 1.< / b >
入門
< / a >
< ul class = "articles" >
< li class = "chapter " data-level = "1.1" data-path = "ch1/ch1-01.html" >
< a href = "../ch1/ch1-01.html" >
< i class = "fa fa-check" > < / i >
< b > 1.1.< / b >
Hello, World
< / a >
< / li >
< li class = "chapter " data-level = "1.2" data-path = "ch1/ch1-02.html" >
< a href = "../ch1/ch1-02.html" >
< i class = "fa fa-check" > < / i >
< b > 1.2.< / b >
命令行參數
< / a >
< / li >
< li class = "chapter active" data-level = "1.3" data-path = "ch1/ch1-03.html" >
< a href = "../ch1/ch1-03.html" >
< i class = "fa fa-check" > < / i >
< b > 1.3.< / b >
査找重復的行
< / a >
< / li >
< li class = "chapter " data-level = "1.4" data-path = "ch1/ch1-04.html" >
< a href = "../ch1/ch1-04.html" >
< i class = "fa fa-check" > < / i >
< b > 1.4.< / b >
GIF動畫
< / a >
< / li >
< li class = "chapter " data-level = "1.5" data-path = "ch1/ch1-05.html" >
< a href = "../ch1/ch1-05.html" >
< i class = "fa fa-check" > < / i >
< b > 1.5.< / b >
穫取URL
< / a >
< / li >
< li class = "chapter " data-level = "1.6" data-path = "ch1/ch1-06.html" >
< a href = "../ch1/ch1-06.html" >
< i class = "fa fa-check" > < / i >
< b > 1.6.< / b >
併髮穫取多個URL
< / a >
< / li >
< li class = "chapter " data-level = "1.7" data-path = "ch1/ch1-07.html" >
< a href = "../ch1/ch1-07.html" >
< i class = "fa fa-check" > < / i >
< b > 1.7.< / b >
Web服務
< / a >
< / li >
< li class = "chapter " data-level = "1.8" data-path = "ch1/ch1-08.html" >
< a href = "../ch1/ch1-08.html" >
< i class = "fa fa-check" > < / i >
< b > 1.8.< / b >
本章要點
< / a >
< / li >
< / ul >
< / li >
< li class = "chapter " data-level = "2" data-path = "ch2/ch2.html" >
< a href = "../ch2/ch2.html" >
< i class = "fa fa-check" > < / i >
< b > 2.< / b >
程序結構
< / a >
< ul class = "articles" >
< li class = "chapter " data-level = "2.1" data-path = "ch2/ch2-01.html" >
< a href = "../ch2/ch2-01.html" >
< i class = "fa fa-check" > < / i >
< b > 2.1.< / b >
命名
< / a >
< / li >
< li class = "chapter " data-level = "2.2" data-path = "ch2/ch2-02.html" >
< a href = "../ch2/ch2-02.html" >
< i class = "fa fa-check" > < / i >
< b > 2.2.< / b >
聲明
< / a >
< / li >
< li class = "chapter " data-level = "2.3" data-path = "ch2/ch2-03.html" >
< a href = "../ch2/ch2-03.html" >
< i class = "fa fa-check" > < / i >
< b > 2.3.< / b >
變量
< / a >
< / li >
< li class = "chapter " data-level = "2.4" data-path = "ch2/ch2-04.html" >
< a href = "../ch2/ch2-04.html" >
< i class = "fa fa-check" > < / i >
< b > 2.4.< / b >
賦值
< / a >
< / li >
< li class = "chapter " data-level = "2.5" data-path = "ch2/ch2-05.html" >
< a href = "../ch2/ch2-05.html" >
< i class = "fa fa-check" > < / i >
< b > 2.5.< / b >
類型
< / a >
< / li >
< li class = "chapter " data-level = "2.6" data-path = "ch2/ch2-06.html" >
< a href = "../ch2/ch2-06.html" >
< i class = "fa fa-check" > < / i >
< b > 2.6.< / b >
包和文件
< / a >
< / li >
< li class = "chapter " data-level = "2.7" data-path = "ch2/ch2-07.html" >
< a href = "../ch2/ch2-07.html" >
< i class = "fa fa-check" > < / i >
< b > 2.7.< / b >
作用域
< / a >
< / li >
< / ul >
< / li >
< li class = "chapter " data-level = "3" data-path = "ch3/ch3.html" >
< a href = "../ch3/ch3.html" >
< i class = "fa fa-check" > < / i >
< b > 3.< / b >
基礎數據類型
< / a >
< ul class = "articles" >
< li class = "chapter " data-level = "3.1" data-path = "ch3/ch3-01.html" >
< a href = "../ch3/ch3-01.html" >
< i class = "fa fa-check" > < / i >
< b > 3.1.< / b >
整型
< / a >
< / li >
< li class = "chapter " data-level = "3.2" data-path = "ch3/ch3-02.html" >
< a href = "../ch3/ch3-02.html" >
< i class = "fa fa-check" > < / i >
< b > 3.2.< / b >
浮點數
< / a >
< / li >
< li class = "chapter " data-level = "3.3" data-path = "ch3/ch3-03.html" >
< a href = "../ch3/ch3-03.html" >
< i class = "fa fa-check" > < / i >
< b > 3.3.< / b >
復數
< / a >
< / li >
< li class = "chapter " data-level = "3.4" data-path = "ch3/ch3-04.html" >
< a href = "../ch3/ch3-04.html" >
< i class = "fa fa-check" > < / i >
< b > 3.4.< / b >
佈爾型
< / a >
< / li >
< li class = "chapter " data-level = "3.5" data-path = "ch3/ch3-05.html" >
< a href = "../ch3/ch3-05.html" >
< i class = "fa fa-check" > < / i >
< b > 3.5.< / b >
字符串
< / a >
< / li >
< li class = "chapter " data-level = "3.6" data-path = "ch3/ch3-06.html" >
< a href = "../ch3/ch3-06.html" >
< i class = "fa fa-check" > < / i >
< b > 3.6.< / b >
常量
< / a >
< / li >
< / ul >
< / li >
< li class = "chapter " data-level = "4" data-path = "ch4/ch4.html" >
< a href = "../ch4/ch4.html" >
< i class = "fa fa-check" > < / i >
< b > 4.< / b >
復閤數據類型
< / a >
< ul class = "articles" >
< li class = "chapter " data-level = "4.1" data-path = "ch4/ch4-01.html" >
< a href = "../ch4/ch4-01.html" >
< i class = "fa fa-check" > < / i >
< b > 4.1.< / b >
數組
< / a >
< / li >
< li class = "chapter " data-level = "4.2" data-path = "ch4/ch4-02.html" >
< a href = "../ch4/ch4-02.html" >
< i class = "fa fa-check" > < / i >
< b > 4.2.< / b >
切片
< / a >
< / li >
< li class = "chapter " data-level = "4.3" data-path = "ch4/ch4-03.html" >
< a href = "../ch4/ch4-03.html" >
< i class = "fa fa-check" > < / i >
< b > 4.3.< / b >
字典
< / a >
< / li >
< li class = "chapter " data-level = "4.4" data-path = "ch4/ch4-04.html" >
< a href = "../ch4/ch4-04.html" >
< i class = "fa fa-check" > < / i >
< b > 4.4.< / b >
結構體
< / a >
< / li >
< li class = "chapter " data-level = "4.5" data-path = "ch4/ch4-05.html" >
< a href = "../ch4/ch4-05.html" >
< i class = "fa fa-check" > < / i >
< b > 4.5.< / b >
JSON
< / a >
< / li >
< li class = "chapter " data-level = "4.6" data-path = "ch4/ch4-06.html" >
< a href = "../ch4/ch4-06.html" >
< i class = "fa fa-check" > < / i >
< b > 4.6.< / b >
文本和HTML模闆
< / a >
< / li >
< / ul >
< / li >
< li class = "chapter " data-level = "5" data-path = "ch5/ch5.html" >
< a href = "../ch5/ch5.html" >
< i class = "fa fa-check" > < / i >
< b > 5.< / b >
函數
< / a >
< ul class = "articles" >
< li class = "chapter " data-level = "5.1" data-path = "ch5/ch5-01.html" >
< a href = "../ch5/ch5-01.html" >
< i class = "fa fa-check" > < / i >
< b > 5.1.< / b >
函數聲明
< / a >
< / li >
< li class = "chapter " data-level = "5.2" data-path = "ch5/ch5-02.html" >
< a href = "../ch5/ch5-02.html" >
< i class = "fa fa-check" > < / i >
< b > 5.2.< / b >
遞歸
< / a >
< / li >
< li class = "chapter " data-level = "5.3" data-path = "ch5/ch5-03.html" >
< a href = "../ch5/ch5-03.html" >
< i class = "fa fa-check" > < / i >
< b > 5.3.< / b >
多返迴值
< / a >
< / li >
< li class = "chapter " data-level = "5.4" data-path = "ch5/ch5-04.html" >
< a href = "../ch5/ch5-04.html" >
< i class = "fa fa-check" > < / i >
< b > 5.4.< / b >
錯誤
< / a >
< / li >
< li class = "chapter " data-level = "5.5" data-path = "ch5/ch5-05.html" >
< a href = "../ch5/ch5-05.html" >
< i class = "fa fa-check" > < / i >
< b > 5.5.< / b >
函數值
< / a >
< / li >
< li class = "chapter " data-level = "5.6" data-path = "ch5/ch5-06.html" >
< a href = "../ch5/ch5-06.html" >
< i class = "fa fa-check" > < / i >
< b > 5.6.< / b >
匿名函數
< / a >
< / li >
< li class = "chapter " data-level = "5.7" data-path = "ch5/ch5-07.html" >
< a href = "../ch5/ch5-07.html" >
< i class = "fa fa-check" > < / i >
< b > 5.7.< / b >
可變參數
< / a >
< / li >
< li class = "chapter " data-level = "5.8" data-path = "ch5/ch5-08.html" >
< a href = "../ch5/ch5-08.html" >
< i class = "fa fa-check" > < / i >
< b > 5.8.< / b >
Deferred函數
< / a >
< / li >
< li class = "chapter " data-level = "5.9" data-path = "ch5/ch5-09.html" >
< a href = "../ch5/ch5-09.html" >
< i class = "fa fa-check" > < / i >
< b > 5.9.< / b >
Panic異常
< / a >
< / li >
< li class = "chapter " data-level = "5.10" data-path = "ch5/ch5-10.html" >
< a href = "../ch5/ch5-10.html" >
< i class = "fa fa-check" > < / i >
< b > 5.10.< / b >
Recover捕穫異常
< / a >
< / li >
< / ul >
< / li >
< li class = "chapter " data-level = "6" data-path = "ch6/ch6.html" >
< a href = "../ch6/ch6.html" >
< i class = "fa fa-check" > < / i >
< b > 6.< / b >
方法
< / a >
< ul class = "articles" >
< li class = "chapter " data-level = "6.1" data-path = "ch6/ch6-01.html" >
< a href = "../ch6/ch6-01.html" >
< i class = "fa fa-check" > < / i >
< b > 6.1.< / b >
方法聲明
< / a >
< / li >
< li class = "chapter " data-level = "6.2" data-path = "ch6/ch6-02.html" >
< a href = "../ch6/ch6-02.html" >
< i class = "fa fa-check" > < / i >
< b > 6.2.< / b >
基於指鍼對象的方法
< / a >
< / li >
< li class = "chapter " data-level = "6.3" data-path = "ch6/ch6-03.html" >
< a href = "../ch6/ch6-03.html" >
< i class = "fa fa-check" > < / i >
< b > 6.3.< / b >
通過嵌入結構體來擴展類型
< / a >
< / li >
< li class = "chapter " data-level = "6.4" data-path = "ch6/ch6-04.html" >
< a href = "../ch6/ch6-04.html" >
< i class = "fa fa-check" > < / i >
< b > 6.4.< / b >
方法值和方法錶達式
< / a >
< / li >
< li class = "chapter " data-level = "6.5" data-path = "ch6/ch6-05.html" >
< a href = "../ch6/ch6-05.html" >
< i class = "fa fa-check" > < / i >
< b > 6.5.< / b >
示例: Bit數組
< / a >
< / li >
< li class = "chapter " data-level = "6.6" data-path = "ch6/ch6-06.html" >
< a href = "../ch6/ch6-06.html" >
< i class = "fa fa-check" > < / i >
< b > 6.6.< / b >
封裝
< / a >
< / li >
< / ul >
< / li >
< li class = "chapter " data-level = "7" data-path = "ch7/ch7.html" >
< a href = "../ch7/ch7.html" >
< i class = "fa fa-check" > < / i >
< b > 7.< / b >
接口
< / a >
< ul class = "articles" >
< li class = "chapter " data-level = "7.1" data-path = "ch7/ch7-01.html" >
< a href = "../ch7/ch7-01.html" >
< i class = "fa fa-check" > < / i >
< b > 7.1.< / b >
接口是閤約
< / a >
< / li >
< li class = "chapter " data-level = "7.2" data-path = "ch7/ch7-02.html" >
< a href = "../ch7/ch7-02.html" >
< i class = "fa fa-check" > < / i >
< b > 7.2.< / b >
接口類型
< / a >
< / li >
< li class = "chapter " data-level = "7.3" data-path = "ch7/ch7-03.html" >
< a href = "../ch7/ch7-03.html" >
< i class = "fa fa-check" > < / i >
< b > 7.3.< / b >
實現接口的條件
< / a >
< / li >
< li class = "chapter " data-level = "7.4" data-path = "ch7/ch7-04.html" >
< a href = "../ch7/ch7-04.html" >
< i class = "fa fa-check" > < / i >
< b > 7.4.< / b >
flag.Value接口
< / a >
< / li >
< li class = "chapter " data-level = "7.5" data-path = "ch7/ch7-05.html" >
< a href = "../ch7/ch7-05.html" >
< i class = "fa fa-check" > < / i >
< b > 7.5.< / b >
接口值
< / a >
< / li >
< li class = "chapter " data-level = "7.6" data-path = "ch7/ch7-06.html" >
< a href = "../ch7/ch7-06.html" >
< i class = "fa fa-check" > < / i >
< b > 7.6.< / b >
sort.Interface接口
< / a >
< / li >
< li class = "chapter " data-level = "7.7" data-path = "ch7/ch7-07.html" >
< a href = "../ch7/ch7-07.html" >
< i class = "fa fa-check" > < / i >
< b > 7.7.< / b >
http.Handler接口
< / a >
< / li >
< li class = "chapter " data-level = "7.8" data-path = "ch7/ch7-08.html" >
< a href = "../ch7/ch7-08.html" >
< i class = "fa fa-check" > < / i >
< b > 7.8.< / b >
error接口
< / a >
< / li >
< li class = "chapter " data-level = "7.9" data-path = "ch7/ch7-09.html" >
< a href = "../ch7/ch7-09.html" >
< i class = "fa fa-check" > < / i >
< b > 7.9.< / b >
示例: 錶達式求值
< / a >
< / li >
< li class = "chapter " data-level = "7.10" data-path = "ch7/ch7-10.html" >
< a href = "../ch7/ch7-10.html" >
< i class = "fa fa-check" > < / i >
< b > 7.10.< / b >
類型斷言
< / a >
< / li >
< li class = "chapter " data-level = "7.11" data-path = "ch7/ch7-11.html" >
< a href = "../ch7/ch7-11.html" >
< i class = "fa fa-check" > < / i >
< b > 7.11.< / b >
基於類型斷言識彆錯誤類型
< / a >
< / li >
< li class = "chapter " data-level = "7.12" data-path = "ch7/ch7-12.html" >
< a href = "../ch7/ch7-12.html" >
< i class = "fa fa-check" > < / i >
< b > 7.12.< / b >
通過類型斷言査詢接口
< / a >
< / li >
< li class = "chapter " data-level = "7.13" data-path = "ch7/ch7-13.html" >
< a href = "../ch7/ch7-13.html" >
< i class = "fa fa-check" > < / i >
< b > 7.13.< / b >
類型分支
< / a >
< / li >
< li class = "chapter " data-level = "7.14" data-path = "ch7/ch7-14.html" >
< a href = "../ch7/ch7-14.html" >
< i class = "fa fa-check" > < / i >
< b > 7.14.< / b >
示例: 基於標記的XML解碼
< / a >
< / li >
< li class = "chapter " data-level = "7.15" data-path = "ch7/ch7-15.html" >
< a href = "../ch7/ch7-15.html" >
< i class = "fa fa-check" > < / i >
< b > 7.15.< / b >
補充幾點
< / a >
< / li >
< / ul >
< / li >
< li class = "chapter " data-level = "8" data-path = "ch8/ch8.html" >
< a href = "../ch8/ch8.html" >
< i class = "fa fa-check" > < / i >
< b > 8.< / b >
Goroutines和Channels
< / a >
< ul class = "articles" >
< li class = "chapter " data-level = "8.1" data-path = "ch8/ch8-01.html" >
< a href = "../ch8/ch8-01.html" >
< i class = "fa fa-check" > < / i >
< b > 8.1.< / b >
Goroutines
< / a >
< / li >
< li class = "chapter " data-level = "8.2" data-path = "ch8/ch8-02.html" >
< a href = "../ch8/ch8-02.html" >
< i class = "fa fa-check" > < / i >
< b > 8.2.< / b >
示例: 併髮的Clock服務
< / a >
< / li >
< li class = "chapter " data-level = "8.3" data-path = "ch8/ch8-03.html" >
< a href = "../ch8/ch8-03.html" >
< i class = "fa fa-check" > < / i >
< b > 8.3.< / b >
示例: 併髮的Echo服務
< / a >
< / li >
< li class = "chapter " data-level = "8.4" data-path = "ch8/ch8-04.html" >
< a href = "../ch8/ch8-04.html" >
< i class = "fa fa-check" > < / i >
< b > 8.4.< / b >
Channels
< / a >
< / li >
< li class = "chapter " data-level = "8.5" data-path = "ch8/ch8-05.html" >
< a href = "../ch8/ch8-05.html" >
< i class = "fa fa-check" > < / i >
< b > 8.5.< / b >
併行的循環
< / a >
< / li >
< li class = "chapter " data-level = "8.6" data-path = "ch8/ch8-06.html" >
< a href = "../ch8/ch8-06.html" >
< i class = "fa fa-check" > < / i >
< b > 8.6.< / b >
示例: 併髮的Web爬蟲
< / a >
< / li >
< li class = "chapter " data-level = "8.7" data-path = "ch8/ch8-07.html" >
< a href = "../ch8/ch8-07.html" >
< i class = "fa fa-check" > < / i >
< b > 8.7.< / b >
基於select的多路復用
< / a >
< / li >
< li class = "chapter " data-level = "8.8" data-path = "ch8/ch8-08.html" >
< a href = "../ch8/ch8-08.html" >
< i class = "fa fa-check" > < / i >
< b > 8.8.< / b >
示例: 併髮的字典遍歷
< / a >
< / li >
< li class = "chapter " data-level = "8.9" data-path = "ch8/ch8-09.html" >
< a href = "../ch8/ch8-09.html" >
< i class = "fa fa-check" > < / i >
< b > 8.9.< / b >
併髮的退齣
< / a >
< / li >
< li class = "chapter " data-level = "8.10" data-path = "ch8/ch8-10.html" >
< a href = "../ch8/ch8-10.html" >
< i class = "fa fa-check" > < / i >
< b > 8.10.< / b >
示例: 聊天服務
< / a >
< / li >
< / ul >
< / li >
< li class = "chapter " data-level = "9" data-path = "ch9/ch9.html" >
< a href = "../ch9/ch9.html" >
< i class = "fa fa-check" > < / i >
< b > 9.< / b >
基於共享變量的併髮
< / a >
< ul class = "articles" >
< li class = "chapter " data-level = "9.1" data-path = "ch9/ch9-01.html" >
< a href = "../ch9/ch9-01.html" >
< i class = "fa fa-check" > < / i >
< b > 9.1.< / b >
競爭條件
< / a >
< / li >
< li class = "chapter " data-level = "9.2" data-path = "ch9/ch9-02.html" >
< a href = "../ch9/ch9-02.html" >
< i class = "fa fa-check" > < / i >
< b > 9.2.< / b >
sync.Mutex互斥鎖
< / a >
< / li >
< li class = "chapter " data-level = "9.3" data-path = "ch9/ch9-03.html" >
< a href = "../ch9/ch9-03.html" >
< i class = "fa fa-check" > < / i >
< b > 9.3.< / b >
sync.RWMutex讀寫鎖
< / a >
< / li >
< li class = "chapter " data-level = "9.4" data-path = "ch9/ch9-04.html" >
< a href = "../ch9/ch9-04.html" >
< i class = "fa fa-check" > < / i >
< b > 9.4.< / b >
內存衕步
< / a >
< / li >
< li class = "chapter " data-level = "9.5" data-path = "ch9/ch9-05.html" >
< a href = "../ch9/ch9-05.html" >
< i class = "fa fa-check" > < / i >
< b > 9.5.< / b >
sync.Once初始化
< / a >
< / li >
< li class = "chapter " data-level = "9.6" data-path = "ch9/ch9-06.html" >
< a href = "../ch9/ch9-06.html" >
< i class = "fa fa-check" > < / i >
< b > 9.6.< / b >
競爭條件檢測
< / a >
< / li >
< li class = "chapter " data-level = "9.7" data-path = "ch9/ch9-07.html" >
< a href = "../ch9/ch9-07.html" >
< i class = "fa fa-check" > < / i >
< b > 9.7.< / b >
示例: 併髮的非阻塞緩存
< / a >
< / li >
< li class = "chapter " data-level = "9.8" data-path = "ch9/ch9-08.html" >
< a href = "../ch9/ch9-08.html" >
< i class = "fa fa-check" > < / i >
< b > 9.8.< / b >
Goroutines和綫程
< / a >
< / li >
< / ul >
< / li >
< li class = "chapter " data-level = "10" data-path = "ch10/ch10.html" >
< a href = "../ch10/ch10.html" >
< i class = "fa fa-check" > < / i >
< b > 10.< / b >
包和工具
< / a >
< ul class = "articles" >
< li class = "chapter " data-level = "10.1" data-path = "ch10/ch10-01.html" >
< a href = "../ch10/ch10-01.html" >
< i class = "fa fa-check" > < / i >
< b > 10.1.< / b >
簡介
< / a >
< / li >
< li class = "chapter " data-level = "10.2" data-path = "ch10/ch10-02.html" >
< a href = "../ch10/ch10-02.html" >
< i class = "fa fa-check" > < / i >
< b > 10.2.< / b >
導入路徑
< / a >
< / li >
< li class = "chapter " data-level = "10.3" data-path = "ch10/ch10-03.html" >
< a href = "../ch10/ch10-03.html" >
< i class = "fa fa-check" > < / i >
< b > 10.3.< / b >
包聲明
< / a >
< / li >
< li class = "chapter " data-level = "10.4" data-path = "ch10/ch10-04.html" >
< a href = "../ch10/ch10-04.html" >
< i class = "fa fa-check" > < / i >
< b > 10.4.< / b >
導入聲明
< / a >
< / li >
< li class = "chapter " data-level = "10.5" data-path = "ch10/ch10-05.html" >
< a href = "../ch10/ch10-05.html" >
< i class = "fa fa-check" > < / i >
< b > 10.5.< / b >
匿名導入
< / a >
< / li >
< li class = "chapter " data-level = "10.6" data-path = "ch10/ch10-06.html" >
< a href = "../ch10/ch10-06.html" >
< i class = "fa fa-check" > < / i >
< b > 10.6.< / b >
包和命名
< / a >
< / li >
< li class = "chapter " data-level = "10.7" data-path = "ch10/ch10-07.html" >
< a href = "../ch10/ch10-07.html" >
< i class = "fa fa-check" > < / i >
< b > 10.7.< / b >
工具
< / a >
< / li >
< / ul >
< / li >
< li class = "chapter " data-level = "11" data-path = "ch11/ch11.html" >
< a href = "../ch11/ch11.html" >
< i class = "fa fa-check" > < / i >
< b > 11.< / b >
測試
< / a >
< ul class = "articles" >
< li class = "chapter " data-level = "11.1" data-path = "ch11/ch11-01.html" >
< a href = "../ch11/ch11-01.html" >
< i class = "fa fa-check" > < / i >
< b > 11.1.< / b >
go test
< / a >
< / li >
< li class = "chapter " data-level = "11.2" data-path = "ch11/ch11-02.html" >
< a href = "../ch11/ch11-02.html" >
< i class = "fa fa-check" > < / i >
< b > 11.2.< / b >
測試函數
< / a >
< / li >
< li class = "chapter " data-level = "11.3" data-path = "ch11/ch11-03.html" >
< a href = "../ch11/ch11-03.html" >
< i class = "fa fa-check" > < / i >
< b > 11.3.< / b >
測試覆蓋率
< / a >
< / li >
< li class = "chapter " data-level = "11.4" data-path = "ch11/ch11-04.html" >
< a href = "../ch11/ch11-04.html" >
< i class = "fa fa-check" > < / i >
< b > 11.4.< / b >
基準測試
< / a >
< / li >
< li class = "chapter " data-level = "11.5" data-path = "ch11/ch11-05.html" >
< a href = "../ch11/ch11-05.html" >
< i class = "fa fa-check" > < / i >
< b > 11.5.< / b >
剖析
< / a >
< / li >
< li class = "chapter " data-level = "11.6" data-path = "ch11/ch11-06.html" >
< a href = "../ch11/ch11-06.html" >
< i class = "fa fa-check" > < / i >
< b > 11.6.< / b >
示例函數
< / a >
< / li >
< / ul >
< / li >
< li class = "chapter " data-level = "12" data-path = "ch12/ch12.html" >
< a href = "../ch12/ch12.html" >
< i class = "fa fa-check" > < / i >
< b > 12.< / b >
反射
< / a >
< ul class = "articles" >
< li class = "chapter " data-level = "12.1" data-path = "ch12/ch12-01.html" >
< a href = "../ch12/ch12-01.html" >
< i class = "fa fa-check" > < / i >
< b > 12.1.< / b >
為何需要反射?
< / a >
< / li >
< li class = "chapter " data-level = "12.2" data-path = "ch12/ch12-02.html" >
< a href = "../ch12/ch12-02.html" >
< i class = "fa fa-check" > < / i >
< b > 12.2.< / b >
reflect.Type和reflect.Value
< / a >
< / li >
< li class = "chapter " data-level = "12.3" data-path = "ch12/ch12-03.html" >
< a href = "../ch12/ch12-03.html" >
< i class = "fa fa-check" > < / i >
< b > 12.3.< / b >
Display遞歸打印
< / a >
< / li >
< li class = "chapter " data-level = "12.4" data-path = "ch12/ch12-04.html" >
< a href = "../ch12/ch12-04.html" >
< i class = "fa fa-check" > < / i >
< b > 12.4.< / b >
示例: 編碼S錶達式
< / a >
< / li >
< li class = "chapter " data-level = "12.5" data-path = "ch12/ch12-05.html" >
< a href = "../ch12/ch12-05.html" >
< i class = "fa fa-check" > < / i >
< b > 12.5.< / b >
通過reflect.Value脩改值
< / a >
< / li >
< li class = "chapter " data-level = "12.6" data-path = "ch12/ch12-06.html" >
< a href = "../ch12/ch12-06.html" >
< i class = "fa fa-check" > < / i >
< b > 12.6.< / b >
示例: 解碼S錶達式
< / a >
< / li >
< li class = "chapter " data-level = "12.7" data-path = "ch12/ch12-07.html" >
< a href = "../ch12/ch12-07.html" >
< i class = "fa fa-check" > < / i >
< b > 12.7.< / b >
穫取結構體字段標識
< / a >
< / li >
< li class = "chapter " data-level = "12.8" data-path = "ch12/ch12-08.html" >
< a href = "../ch12/ch12-08.html" >
< i class = "fa fa-check" > < / i >
< b > 12.8.< / b >
顯示一個類型的方法集
< / a >
< / li >
< li class = "chapter " data-level = "12.9" data-path = "ch12/ch12-09.html" >
< a href = "../ch12/ch12-09.html" >
< i class = "fa fa-check" > < / i >
< b > 12.9.< / b >
幾點忠告
< / a >
< / li >
< / ul >
< / li >
< li class = "chapter " data-level = "13" data-path = "ch13/ch13.html" >
< a href = "../ch13/ch13.html" >
< i class = "fa fa-check" > < / i >
< b > 13.< / b >
底層編程
< / a >
< ul class = "articles" >
< li class = "chapter " data-level = "13.1" data-path = "ch13/ch13-01.html" >
< a href = "../ch13/ch13-01.html" >
< i class = "fa fa-check" > < / i >
< b > 13.1.< / b >
unsafe.Sizeof, Alignof 和 Offsetof
< / a >
< / li >
< li class = "chapter " data-level = "13.2" data-path = "ch13/ch13-02.html" >
< a href = "../ch13/ch13-02.html" >
< i class = "fa fa-check" > < / i >
< b > 13.2.< / b >
unsafe.Pointer
< / a >
< / li >
< li class = "chapter " data-level = "13.3" data-path = "ch13/ch13-03.html" >
< a href = "../ch13/ch13-03.html" >
< i class = "fa fa-check" > < / i >
< b > 13.3.< / b >
示例: 深度相等判斷
< / a >
< / li >
< li class = "chapter " data-level = "13.4" data-path = "ch13/ch13-04.html" >
< a href = "../ch13/ch13-04.html" >
< i class = "fa fa-check" > < / i >
< b > 13.4.< / b >
通過cgo調用C代碼
< / a >
< / li >
< li class = "chapter " data-level = "13.5" data-path = "ch13/ch13-05.html" >
< a href = "../ch13/ch13-05.html" >
< i class = "fa fa-check" > < / i >
< b > 13.5.< / b >
幾點忠告
< / a >
< / li >
< / ul >
< / li >
< li class = "chapter " data-level = "14" data-path = "exercise/ex.html" >
< a href = "../exercise/ex.html" >
< i class = "fa fa-check" > < / i >
< b > 14.< / b >
習題解答
< / a >
< ul class = "articles" >
< li class = "chapter " data-level = "14.1" data-path = "exercise/ex-ch1.html" >
< a href = "../exercise/ex-ch1.html" >
< i class = "fa fa-check" > < / i >
< b > 14.1.< / b >
第一章 入門
< / a >
< / li >
< / ul >
< / li >
< li class = "chapter " data-level = "15" data-path = "errata.html" >
< a href = "../errata.html" >
< i class = "fa fa-check" > < / i >
< b > 15.< / b >
勘誤
< / a >
< / li >
< li class = "divider" > < / li >
< li >
< a href = "https://www.gitbook.com" target = "blank" class = "gitbook-link" >
本書使用 GitBook 釋出
< / a >
< / li >
< / ul >
< / nav >
< / div >
< div class = "book-body" >
< div class = "body-inner" >
< div class = "book-header" role = "navigation" >
<!-- Actions Left -->
<!-- Title -->
< h1 >
< i class = "fa fa-circle-o-notch fa-spin" > < / i >
< a href = "../" > Go编程语言< / a >
< / h1 >
< / div >
< div class = "page-wrapper" tabindex = "-1" role = "main" >
< div class = "page-inner" >
< section class = "normal" id = "section-" >
< h2 id = "13-査找重復的行" > 1.3. 査 找 重 復 的 行 < / h2 >
< p > 文 件 拷 貝 、 文 件 打 印 、 文 件 蒐 索 、 文 件 排 序 、 文 件 統 計 類 的 程 序 一 般 都 會 有 比 較 相 似 的 程 序 結 構 : 處 理 輸 入 的 一 個 循 環 , 在 每 一 個 輸 入 元 素 上 執 行 計 算 處 理 , 在 處 理 的 衕 時 或 者 處 理 完 成 之 後 進 行 結 果 輸 齣 。 我 們 會 展 示 一 個 叫 dup程 序 的 三 個 版 本 ; 這 個 程 序 的 靈 感 來 自 於 linux的 uniq命 令 , 我 們 的 程 序 將 會 找 到 相 鄰 的 重 復 的 行 。 這 個 程 序 提 供 的 模 式 可 以 很 方 便 地 被 脩 改 來 完 成 不 衕 的 需 求 。 < / p >
< p > 第 一 個 版 本 的 dup會 輸 齣 標 準 輸 入 流 中 的 齣 現 多 次 的 行 , 在 行 內 容 前 會 有 其 齣 現 次 數 的 計 數 。 這 個 程 序 將 引 入 if錶 達 式 , map內 置 數 據 結 果 和 bufio的 package。 < / p >
< pre > < code class = "lang-go" > gopl.io/ch1/dup1
< span class = "hljs-comment" > // Dup1 prints the text of each line that appears more than< / span >
< span class = "hljs-comment" > // once in the standard input, preceded by its count.< / span >
< span class = "hljs-keyword" > package< / span > main
< span class = "hljs-keyword" > import< / span > (
< span class = "hljs-string" > " bufio" < / span >
< span class = "hljs-string" > " fmt" < / span >
< span class = "hljs-string" > " os" < / span >
)
< span class = "hljs-keyword" > func< / span > main() {
counts := < span class = "hljs-built_in" > make< / span > (< span class = "hljs-keyword" > map< / span > [< span class = "hljs-typename" > string< / span > ]< span class = "hljs-typename" > int< / span > )
input := bufio.NewScanner(os.Stdin)
< span class = "hljs-keyword" > for< / span > input.Scan() {
counts[input.Text()]++
}
< span class = "hljs-comment" > // < span class = "hljs-doctag" > NOTE:< / span > ignoring potential errors from input.Err()< / span >
< span class = "hljs-keyword" > for< / span > line, n := < span class = "hljs-keyword" > range< / span > counts {
< span class = "hljs-keyword" > if< / span > n > < span class = "hljs-number" > 1< / span > {
fmt.Printf(< span class = "hljs-string" > " %d\t%s\n" < / span > , n, line)
}
}
}
< / code > < / pre >
< p > 和 我 們 前 麫 提 到 的 for循 環 一 樣 , 在 if條 件 的 兩 邊 , 我 們 也 不 需 要 加 括 號 , 但 是 if錶 達 式 後 的 邏 輯 體 的 花 括 號 是 不 能 省 略 的 。 如 果 需 要 的 話 , 像 其 它 語 言 一 樣 , 這 個 錶 達 式 也 可 以 有 else部 分 , 這 部 分 邏 輯 會 在 if中 的 條 件 結 果 為 false時 被 執 行 。 < / p >
< p > map是 go語 言 內 置 的 key/value數 據 結 構 , 這 個 數 據 結 構 能 夠 提 供 常 數 時 間 的 存 儲 、 穫 取 、 測 試 操 作 。 key可 以 是 任 意 數 據 類 型 , 隻 要 該 類 型 能 夠 用 ==來 進 行 比 較 , string是 最 常 用 的 key類 型 。 而 value類 型 的 範 圍 就 更 大 了 , 基 本 上 什 麼 類 型 都 是 可 以 的 。 這 個 例 子 中 的 key都 是 string類 型 , value用 的 是 int類 型 。 我 們 用 內 置 make函 數 來 創 建 一 個 空 的 map, 當 然 了 , make方 法 還 可 以 有 彆 的 用 處 。 在 4.3章 中 我 們 還 會 對 map進 行 更 深 度 的 討 論 。 < / p >
< p > dup程 序 每 次 讀 取 輸 入 的 一 行 , 這 一 行 的 內 容 會 被 當 做 一 個 map的 key, 而 其 value值 會 被 +1。 counts[input.Text()]++這 個 語 句 和 下 麫 的 兩 句 是 等 價 的 : < / p >
< pre > < code class = "lang-go" > line := input.Text()
counts[line] = counts[line] + < span class = "hljs-number" > 1< / span >
< / code > < / pre >
< p > 當 然 了 , 在 這 個 例 子 裏 我 們 併 不 用 擔 心 map在 沒 有 當 前 的 key時 就 對 其 進 行 ++操 作 會 有 什 麼 問 題 , 因 為 go語 言 在 碰 到 這 種 情 況 時 , 會 自 動 將 其 初 始 化 為 0, 然 後 再 進 行 操 作 。 < / p >
< p > 在 這 裏 我 們 又 用 了 一 個 range的 循 環 來 打 印 結 果 , 這 次 range是 被 用 在 map這 個 數 據 結 果 上 。 這 一 次 的 情 況 和 上 次 比 較 類 型 , range會 返 迴 兩 個 值 , 一 個 key和 在 map對 應 這 個 key的 value。 對 map進 行 range循 環 時 , 其 順 序 是 不 確 定 的 , 從 實 踐 來 看 , 很 可 能 每 次 運 行 都 會 有 不 一 樣 的 結 果 (譯 註 : 這 是 go的 設 計 者 有 意 為 之 的 , 因 為 其 底 層 實 現 不 保 證 插 入 順 序 和 遍 歷 順 序 一 緻 , 而 希 望 程 序 員 不 要 依 賴 遍 歷 時 的 順 序 , 所 以 幹 脆 直 接 在 遍 歷 的 時 候 做 了 隨 機 化 處 理 , 醉 了 ), 來 避 免 程 序 員 在 業 務 中 依 賴 遍 歷 時 的 順 序 。 < / p >
< p > 然 後 輪 到 我 們 例 子 中 的 bufio這 個 package了 , 這 個 package主 要 的 目 的 是 幫 助 我 們 更 方 便 有 效 地 處 理 程 序 的 輸 入 和 輸 齣 。 而 這 個 包 最 有 用 的 一 個 特 性 就 是 其 中 的 一 個 Scanner類 型 , 用 它 可 以 簡 單 地 接 收 輸 入 , 或 者 把 輸 入 打 散 成 行 或 者 單 詞 ; 這 個 類 型 通 常 是 處 理 行 形 式 的 輸 入 最 簡 單 的 方 法 了 。 < / p >
< p > 本 程 序 中 用 了 一 個 短 變 量 聲 明 , 來 創 建 一 個 buffio.Scanner對 象 : < / p >
< pre > < code > input := bufio.NewScanner(os.Stdin)
< / code > < / pre > < p > scanner對 象 可 以 從 程 序 的 標 準 輸 入 中 讀 取 內 容 。 對 input.Scanner的 每 一 次 調 用 都 會 調 入 一 個 新 行 , 併 且 會 自 動 將 其 行 末 的 換 行 符 去 掉 ; 其 結 果 可 以 用 input.Text()得 到 。 Scan方 法 在 讀 到 了 新 行 的 時 候 會 返 迴 true, 而 在 沒 有 新 行 被 讀 入 時 , 會 返 迴 false。 < / p >
< p > 例 子 中 還 有 一 個 fmt.Printf, 這 個 函 數 和 C繫 的 其 它 語 言 裏 的 那 個 printf函 數 差 不 多 , 都 是 格 式 化 輸 齣 的 方 法 。 fmt.Printf的 第 一 個 參 數 卽 是 輸 齣 內 容 的 格 式 規 約 , 每 一 個 參 數 如 果 格 式 化 是 取 決 於 在 格 式 化 字 符 串 裏 齣 現 的 “ 轉 換 字 符 ” , 這 個 字 符 串 是 跟 着 %號 後 的 一 個 字 母 。 比 如 %d錶 示 以 一 個 整 數 的 形 式 來 打 印 一 個 變 量 , 而 %s, 則 錶 示 以 string形 式 來 打 印 一 個 變 量 。 < / p >
< p > Printf有 一 大 堆 這 種 轉 換 , Go程 序 員 把 這 些 叫 做 verb(動 詞 )。 下 麫 的 錶 格 列 齣 了 常 用 的 動 詞 , 當 然 了 不 是 全 部 , 但 基 本 也 夠 用 了 。 < / p >
< pre > < code > %d int變 量
%x, %o, %b 分 彆 為 16進 製 , 8進 製 , 2進 製 形 式 的 int
%f, %g, %e 浮 點 數 : 3.141593 3.141592653589793 3.141593e+00
%t 佈 爾 變 量 : true 或 false
%c rune (Unicode code point), go語 言 裏 特 有 的 Unicode字 符 類 型
%s string
%q quoted string " abc" or rune ' c'
%v 會 將 任 意 變 量 以 易 讀 的 形 式 打 印 齣 來
%T 打 印 變 量 的 類 型
%% 字 符 型 百 分 比 標 誌 (不 確 定 ) literal percent sign (no operand)
< / code > < / pre > < p > dup1中 的 程 序 還 包 含 了 一 個 \t和 \n的 格 式 化 字 符 串 。 在 字 符 串 中 會 以 這 些 特 殊 的 轉 義 字 符 來 錶 示 不 可 見 字 符 。 Printf默 認 不 會 在 輸 齣 內 容 後 加 上 換 行 符 。 按 照 慣 例 , 用 來 格 式 化 的 函 數 都 會 在 末 尾 以 f字 母 結 尾 , 比 如 log.Printf, fmt.Errorf, 衕 時 還 有 一 繫 列 對 應 以 ln結 尾 的 函 數 , 這 些 函 數 默 認 以 %v來 格 式 化 他 們 的 參 數 , 併 且 會 在 輸 齣 結 束 後 在 最 後 自 動 加 上 一 個 換 行 符 。 < / p >
< p > 許 多 程 序 從 標 準 輸 入 中 讀 取 數 據 , 像 上 麫 的 例 子 那 樣 。 除 此 之 外 , 還 可 能 從 一 繫 列 的 文 件 中 讀 取 。 下 一 個 dup程 序 就 是 從 標 準 輸 入 中 讀 到 一 些 文 件 名 , 用 os.Open函 數 來 打 開 每 一 個 文 件 穫 取 內 容 的 。 < / p >
< pre > < code class = "lang-go" > gopl.io/ch1/dup2
< span class = "hljs-comment" > // Dup2 prints the count and text of lines that appear more than once< / span >
< span class = "hljs-comment" > // in the input. It reads from stdin or from a list of named files.< / span >
< span class = "hljs-keyword" > package< / span > main
< span class = "hljs-keyword" > import< / span > (
< span class = "hljs-string" > " bufio" < / span >
< span class = "hljs-string" > " fmt" < / span >
< span class = "hljs-string" > " os" < / span >
)
< span class = "hljs-keyword" > func< / span > main() {
counts := < span class = "hljs-built_in" > make< / span > (< span class = "hljs-keyword" > map< / span > [< span class = "hljs-typename" > string< / span > ]< span class = "hljs-typename" > int< / span > )
files := os.Args[< span class = "hljs-number" > 1< / span > :]
< span class = "hljs-keyword" > if< / span > < span class = "hljs-built_in" > len< / span > (files) == < span class = "hljs-number" > 0< / span > {
countLines(os.Stdin, counts)
} < span class = "hljs-keyword" > else< / span > {
< span class = "hljs-keyword" > for< / span > _, arg := < span class = "hljs-keyword" > range< / span > files {
f, err := os.Open(arg)
< span class = "hljs-keyword" > if< / span > err != < span class = "hljs-constant" > nil< / span > {
fmt.Fprintf(os.Stderr, < span class = "hljs-string" > " dup2: %v\n" < / span > , err)
< span class = "hljs-keyword" > continue< / span >
}
countLines(f, counts)
f.Close()
}
}
< span class = "hljs-keyword" > for< / span > line, n := < span class = "hljs-keyword" > range< / span > counts {
< span class = "hljs-keyword" > if< / span > n > < span class = "hljs-number" > 1< / span > {
fmt.Printf(< span class = "hljs-string" > " %d\t%s\n" < / span > , n, line)
}
}
}
< span class = "hljs-keyword" > func< / span > countLines(f *os.File, counts < span class = "hljs-keyword" > map< / span > [< span class = "hljs-typename" > string< / span > ]< span class = "hljs-typename" > int< / span > ) {
input := bufio.NewScanner(f)
< span class = "hljs-keyword" > for< / span > input.Scan() {
counts[input.Text()]++
}
< span class = "hljs-comment" > // < span class = "hljs-doctag" > NOTE:< / span > ignoring potential errors from input.Err()< / span >
}
< / code > < / pre >
< p > os.Open函 數 會 返 迴 兩 個 值 。 第 一 個 值 是 一 個 打 開 的 文 件 類 型 (*os.File), 這 個 對 象 在 下 麫 的 程 序 中 被 Scanner讀 取 。 < / p >
< p > os.Open返 迴 的 第 二 個 值 是 一 個 go內 置 的 error類 型 。 如 果 這 個 error和 內 置 值 的 nil(譯 註 : 相 當 於 其 它 語 言 裏 的 NULL)相 等 的 話 , 說 明 文 件 被 成 功 的 打 開 了 。 之 後 文 件 被 讀 取 , 一 直 到 文 件 的 最 後 , Close函 數 關 閉 該 文 件 , 併 釋 放 相 應 的 佔 用 一 切 資 源 。 另 一 方 麫 , 如 果 err的 值 不 是 nil的 話 , 那 說 明 在 打 開 文 件 的 時 候 齣 了 某 種 錯 誤 。 這 種 情 況 下 , error類 型 的 值 會 描 述 具 體 的 問 題 。 我 們 例 子 裏 的 簡 單 錯 誤 處 理 會 在 標 準 錯 誤 流 中 用 Fprintf和 %v來 格 式 化 該 錯 誤 字 符 串 。 然 後 繼 續 處 理 下 一 個 文 件 ; continue語 句 會 直 接 跳 過 之 後 的 語 句 , 直 接 開 始 執 行 下 一 次 循 環 。 < / p >
< p > 我 們 在 本 書 中 早 期 的 例 子 中 做 了 比 較 詳 盡 的 錯 誤 處 理 , 當 然 了 , 在 實 際 編 碼 過 程 中 , 像 os.Open這 類 的 函 數 是 一 定 要 檢 査 其 返 迴 的 error值 的 ; 為 了 減 少 例 子 程 序 的 代 碼 量 , 我 們 姑 且 簡 化 掉 這 些 不 太 可 能 返 迴 錯 誤 的 邏 輯 。 後 麫 的 例 子 裏 我 們 會 跳 過 錯 誤 檢 査 。 在 5.4節 中 我 們 會 對 錯 誤 處 理 做 更 詳 細 的 闡 述 。 < / p >
< p > 讀 者 可 以 再 觀 察 一 下 上 麫 的 例 子 , 我 們 的 countLines函 數 是 在 其 聲 明 之 前 就 被 調 用 了 。 在 Go語 言 裏 , 函 數 和 包 級 彆 的 變 量 可 以 以 任 意 的 順 序 被 聲 明 , 併 不 影 響 其 被 調 用 。 (譯 註 : 最 好 還 是 遵 循 一 定 的 規 範 )< / p >
< p > 再 來 講 講 map這 個 數 據 結 構 , map是 用 make函 數 創 建 的 數 據 結 構 的 一 個 引 用 。 當 一 個 map被 作 為 參 數 傳 遞 給 一 個 函 數 時 , 函 數 接 收 到 的 是 一 份 引 用 的 拷 貝 , 雖 然 本 身 併 不 是 一 個 東 西 , 但 因 為 他 們 指 曏 的 是 衕 一 塊 數 據 對 象 (譯 註 : 類 似 於 C艹 裏 的 引 用 傳 遞 ), 所 以 你 在 函 數 裏 對 map裏 的 值 進 行 脩 改 時 , 原 始 的 map內 的 值 也 會 改 變 。 在 我 們 的 例 子 中 , 我 們 在 countLines函 數 中 插 入 到 counts這 個 map裏 的 值 , 在 主 函 數 中 也 是 看 得 到 的 。 < / p >
< p > 上 麫 這 個 版 本 的 dup是 以 流 的 形 式 來 處 理 輸 入 , 併 將 其 打 散 為 行 。 理 論 上 這 些 程 序 也 是 可 以 以 二 進 製 形 式 來 處 理 輸 入 的 。 我 們 也 可 以 一 次 性 的 把 整 個 輸 入 內 容 全 部 讀 到 內 存 中 , 然 後 再 把 其 分 割 為 多 行 , 然 後 再 去 處 理 這 些 行 內 的 數 據 。 下 麫 的 dup3這 個 例 子 就 是 以 這 種 形 式 來 進 行 操 作 的 。 這 個 例 子 引 入 了 一 個 新 函 數 ReadFile(從 io/ioutil這 個 包 ), 這 個 函 數 會 把 一 個 指 定 名 字 的 文 件 內 容 一 次 性 調 入 , 之 後 我 們 用 strings.Split函 數 把 文 件 分 割 為 多 個 子 字 符 串 , 併 存 儲 到 slice結 構 中 。 (Split函 數 是 strings.Join的 逆 函 數 , Join函 數 之 前 提 到 過 )< / p >
< p > 我 們 簡 化 了 dup3這 個 程 序 。 首 先 , 他 隻 讀 取 命 名 的 文 件 , 而 不 去 讀 標 準 輸 入 , 因 為 ReadFile函 數 需 要 一 個 文 件 名 參 數 。 其 次 , 我 們 將 行 計 數 邏 輯 移 迴 到 了 main函 數 , 因 為 現 在 這 個 邏 輯 隻 有 一 個 地 方 需 要 用 到 。 < / p >
< pre > < code class = "lang-go" > gopl.io/ch1/dup3
< span class = "hljs-keyword" > package< / span > main
< span class = "hljs-keyword" > import< / span > (
< span class = "hljs-string" > " fmt" < / span >
< span class = "hljs-string" > " io/ioutil" < / span >
< span class = "hljs-string" > " os" < / span >
< span class = "hljs-string" > " strings" < / span >
)
< span class = "hljs-keyword" > func< / span > main() {
counts := < span class = "hljs-built_in" > make< / span > (< span class = "hljs-keyword" > map< / span > [< span class = "hljs-typename" > string< / span > ]< span class = "hljs-typename" > int< / span > )
< span class = "hljs-keyword" > for< / span > _, filename := < span class = "hljs-keyword" > range< / span > os.Args[< span class = "hljs-number" > 1< / span > :] {
data, err := ioutil.ReadFile(filename)
< span class = "hljs-keyword" > if< / span > err != < span class = "hljs-constant" > nil< / span > {
fmt.Fprintf(os.Stderr, < span class = "hljs-string" > " dup3: %v\n" < / span > , err)
< span class = "hljs-keyword" > continue< / span >
}
< span class = "hljs-keyword" > for< / span > _, line := < span class = "hljs-keyword" > range< / span > strings.Split(< span class = "hljs-typename" > string< / span > (data), < span class = "hljs-string" > " \n" < / span > ) {
counts[line]++
}
}
< span class = "hljs-keyword" > for< / span > line, n := < span class = "hljs-keyword" > range< / span > counts {
< span class = "hljs-keyword" > if< / span > n > < span class = "hljs-number" > 1< / span > {
fmt.Printf(< span class = "hljs-string" > " %d\t%s\n" < / span > , n, line)
}
}
}
< / code > < / pre >
< p > ReadFile函 數 返 迴 一 個 byte的 slice, 這 個 slice必 鬚 被 轉 換 為 string, 之 後 纔 能 夠 用 string.Split方 法 來 進 行 處 理 。 我 們 在 3.5.4節 中 會 更 詳 細 地 講 解 string和 byte slice(字 節 數 組 )。 < / p >
< p > 在 更 底 層 一 些 的 地 方 , bufio.Scanner, ioutil.ReadFile和 ioutil.WriteFile使 用 的 是 *os.File的 Read和 Write方 法 , 不 過 一 般 程 序 員 併 不 需 要 去 直 接 了 解 到 其 底 層 實 現 細 節 , 在 bufio和 io/ioutil包 中 提 供 的 方 法 已 經 足 夠 好 用 。 < / p >
< pre > < code > Exercise 1.4: 脩 改 dup2, 使 其 可 以 打 印 重 復 的 行 分 彆 齣 現 在 哪 些 文 件 。
< / code > < / pre >
< / section >
< / div >
< / div >
< / div >
< a href = "../ch1/ch1-02.html" class = "navigation navigation-prev " aria-label = "Previous page: 命令行參數" > < i class = "fa fa-angle-left" > < / i > < / a >
< a href = "../ch1/ch1-04.html" class = "navigation navigation-next " aria-label = "Next page: GIF動畫" > < i class = "fa fa-angle-right" > < / i > < / a >
< / div >
< / div >
< script src = "../gitbook/app.js" > < / script >
< script src = "../gitbook/plugins/gitbook-plugin-search/lunr.min.js" > < / script >
< script src = "../gitbook/plugins/gitbook-plugin-search/search.js" > < / script >
< script src = "../gitbook/plugins/gitbook-plugin-sharing/buttons.js" > < / script >
< script src = "../gitbook/plugins/gitbook-plugin-fontsettings/buttons.js" > < / script >
< script >
require(["gitbook"], function(gitbook) {
var config = {"highlight":{},"search":{},"sharing":{"facebook":true,"twitter":true,"google":false,"weibo":false,"instapaper":false,"vk":false,"all":["facebook","google","twitter","weibo","instapaper"]},"fontsettings":{"theme":"white","family":"sans","size":2}};
gitbook.start(config);
});
< / script >
< / body >
< / html >