這是一個建立於 的文章,其中的資訊可能已經有所發展或是發生改變。
在C語言中,字串的記憶體模型定義為以NUL(\x0)結尾的位元組數組。這是為大家所熟知的。
但是在Golang中並不是如此,Golang中的字串abc和abc\x0\x0並不相當,所以說Golang明確規定了字串的長度,而不是以\x0為結尾來判斷的。
下面看範例程式碼:
package mainimport ( "fmt" "os")func main() { var a[5]byte = [5]byte{'a','b','c'} var b[]byte = []byte{'a','b','c'} fmt.Printf("len(a): %d, %q\n", len(a), a) fmt.Printf("len(b): %d, %q\n", len(b), b) slice_a := a[:] str_a := string(slice_a) str_b := string(b) fmt.Printf("len(str_a): %d, %q:%s\n", len(str_a), str_a, str_a) fmt.Printf("len(str_b): %d, %q:%s\n", len(str_b), str_b, str_b) if str_a == str_b { fmt.Println("str_a == str_b") } else { fmt.Println("str_a != str_b") } file, err := os.Create(str_a) if err != nil { fmt.Println(err) } fmt.Println(file)}
代碼輸出為:
len(a): 5, "abc\x00\x00"len(b): 3, "abc"len(str_a): 5, "abc\x00\x00":abclen(str_b): 3, "abc":abcstr_a != str_bopen abc: invalid argument
數組a長度為5,最後兩個位元組為\x00。切片b長度為3,有效內容為abc。分別將a和b轉換為string類型後,數組a中包含的\x00也被保留下來,所以在對比時,這兩者的長度不可能相等的。
當然這還不是最關鍵的,問題處在當用str_a作為檔案名稱建立檔案時file, err := os.Create(str_a),系統會報錯:open abc: invalid argument。說明這不是一個有效路徑名稱。
Golang是一種強類的語言,對於數組來說[3]byte和[5]byte並不是同一個類型,不能通用。當然,數組和切片也更不可能是同一個類型。
解決辦法是,迭代數組中的位元組,跳過為0的位元組:
func GetValidByte(src []byte) []byte { var str_buf []byte for _, v := range src { if v != 0 { str_buf = append(str_buf, v) } } return str_buf}