這是一個建立於 的文章,其中的資訊可能已經有所發展或是發生改變。
今天研究go的複製檔案的時候,發現複製出來的檔案總是比源檔案大了些,程式如下
package mainimport ( "os" "log" "io" "fmt")func main() { file, err := os.Open("C:/Users/Desktop/123123.txt") // For read access. defer file.Close() if err != nil { log.Fatal(err.Error()) } chunks := make([]byte,1024) buff := make([]byte, 1024) for{ count, err := file.Read(buff) if err != nil && err != io.EOF{panic(err)} if 0==count {break} chunks = append(chunks,buff[:count]...) } file1,err := os.Create("C:/Users/Desktop/123123123.txt") defer file1.Close() file1.Write(chunks)}
如果這麼寫程式的話,會導致複製出來的檔案比源檔案大1024個位元組,為什麼呢?
其實主要是因為go語言的內建函數append的原因
我實驗了一下,append函數在將兩個切片合在一起的時候,並不是把第二個切片放進第一個切片裡,如果是直接放進去的話,那個切片的長度不會變,如下
a:=make([]byte,10)b:=make([]byte,2)fmt.Println(len(append(a,b...)))
應該輸出10,而實際上是輸出了12!
所以,在複製檔案的例子中,
chunks := make([]byte,1024)應該改為 chunks := make([]byte,0)
這樣,複製出來的檔案就是正確的了!
繼續補充一點append的用法
第一種用法,就是第一個參數是切片,後面可以傳很多(不固定)元素,比如:
append([]byte,1,2,3)
第二種用法,就是傳兩個切片,這樣的話,只能傳兩個參數,且第二個參數要以...結尾,如上面的例子
a:=make([]byte,10)b:=make([]byte,2)fmt.Println(len(append(a,b...)))
最後,再說一點,讀檔案時候一些不懂的地方
if err != nil && err != io.EOF{panic(err)}
這一句,涉及兩個知識,一個是go語言中的異常,一個是EOF
go語言中的異常機制,有篇部落格寫的很好,看部落格
EOF表示檔案結束異常(讀完了)