本文主要介紹了Go語言中文件讀寫的相關操做。linux
文件是什麼?緩存
計算機中的文件是存儲在外部介質(一般是磁盤)上的數據集合,文件分爲文本文件和二進制文件。app
os.Open()
函數可以打開一個文件,返回一個*File
和一個err
。對獲得的文件實例調用close()
方法可以關閉文件。函數
package main import ( "fmt" "os" ) func main() { // 只讀方式打開當前目錄下的main.go文件 file, err := os.Open("./main.go") if err != nil { fmt.Println("open file failed!, err:", err) return } // 關閉文件 file.Close() }
爲了防止文件忘記關閉,咱們一般使用defer註冊文件關閉語句。命令行
Read方法定義以下:code
func (f *File) Read(b []byte) (n int, err error)
它接收一個字節切片,返回讀取的字節數和可能的具體錯誤,讀到文件末尾時會返回0
和io.EOF
。 舉個例子:字符串
func main() { // 只讀方式打開當前目錄下的main.go文件 file, err := os.Open("./main.go") if err != nil { fmt.Println("open file failed!, err:", err) return } defer file.Close() // 使用Read方法讀取數據 var tmp = make([]byte, 128) n, err := file.Read(tmp) if err == io.EOF { fmt.Println("文件讀完了") return } if err != nil { fmt.Println("read file failed, err:", err) return } fmt.Printf("讀取了%d字節數據\n", n) fmt.Println(string(tmp[:n])) }
使用for循環讀取文件中的全部數據。string
func main() { // 只讀方式打開當前目錄下的main.go文件 file, err := os.Open("./main.go") if err != nil { fmt.Println("open file failed!, err:", err) return } defer file.Close() // 循環讀取文件 var content []byte var tmp = make([]byte, 128) for { n, err := file.Read(tmp) if err == io.EOF { fmt.Println("文件讀完了") break } if err != nil { fmt.Println("read file failed, err:", err) return } content = append(content, tmp[:n]...) } fmt.Println(string(content)) }
bufio是在file的基礎上封裝了一層API,支持更多的功能。it
package main import ( "bufio" "fmt" "io" "os" ) // bufio按行讀取示例 func main() { file, err := os.Open("./main.go") if err != nil { fmt.Println("open file failed, err:", err) return } defer file.Close() reader := bufio.NewReader(file) for { line, err := reader.ReadString('\n') //注意是字符 if err == io.EOF { fmt.Println("文件讀完了") break } if err != nil { fmt.Println("read file failed, err:", err) return } fmt.Print(line) } }
io/ioutil
包的ReadFile
方法可以讀取完整的文件,只須要將文件名做爲參數傳入。io
package main import ( "fmt" "io/ioutil" ) // ioutil.ReadFile讀取整個文件 func main() { content, err := ioutil.ReadFile("./main.go") if err != nil { fmt.Println("read file failed, err:", err) return } fmt.Println(string(content)) }
os.OpenFile()
函數可以以指定模式打開文件,從而實現文件寫入相關功能。
func OpenFile(name string, flag int, perm FileMode) (*File, error) { ... }
其中:
name
:要打開的文件名 flag
:打開文件的模式。 模式有如下幾種:
模式 | 含義 |
---|---|
os.O_WRONLY |
只寫 |
os.O_CREATE |
建立文件 |
os.O_RDONLY |
只讀 |
os.O_RDWR |
讀寫 |
os.O_TRUNC |
清空 |
os.O_APPEND |
追加 |
perm
:文件權限,一個八進制數。r(讀)04,w(寫)02,x(執行)01。
func main() { file, err := os.OpenFile("xx.txt", os.O_CREATE|os.O_TRUNC|os.O_WRONLY, 0666) if err != nil { fmt.Println("open file failed, err:", err) return } defer file.Close() str := "hello 沙河" file.Write([]byte(str)) //寫入字節切片數據 file.WriteString("hello 小王子") //直接寫入字符串數據 }
func main() { file, err := os.OpenFile("xx.txt", os.O_CREATE|os.O_TRUNC|os.O_WRONLY, 0666) if err != nil { fmt.Println("open file failed, err:", err) return } defer file.Close() writer := bufio.NewWriter(file) for i := 0; i < 10; i++ { writer.WriteString("hello沙河\n") //將數據先寫入緩存 } writer.Flush() //將緩存中的內容寫入文件 }
func main() { str := "hello 沙河" err := ioutil.WriteFile("./xx.txt", []byte(str), 0666) if err != nil { fmt.Println("write file failed, err:", err) return } }
藉助io.Copy()
實現一個拷貝文件函數。
// CopyFile 拷貝文件函數 func CopyFile(dstName, srcName string) (written int64, err error) { // 以讀方式打開源文件 src, err := os.Open(srcName) if err != nil { fmt.Printf("open %s failed, err:%v.\n", srcName, err) return } defer src.Close() // 以寫|建立的方式打開目標文件 dst, err := os.OpenFile(dstName, os.O_WRONLY|os.O_CREATE, 0644) if err != nil { fmt.Printf("open %s failed, err:%v.\n", dstName, err) return } defer dst.Close() return io.Copy(dst, src) //調用io.Copy()拷貝內容 } func main() { _, err := CopyFile("dst.txt", "src.txt") if err != nil { fmt.Println("copy file failed, err:", err) return } fmt.Println("copy done!") }
使用文件操做相關知識,模擬實現linux平臺cat
命令的功能。
package main import ( "bufio" "flag" "fmt" "io" "os" ) // cat命令實現 func cat(r *bufio.Reader) { for { buf, err := r.ReadBytes('\n') //注意是字符 if err == io.EOF { break } fmt.Fprintf(os.Stdout, "%s", buf) } } func main() { flag.Parse() // 解析命令行參數 if flag.NArg() == 0 { // 若是沒有參數默認從標準輸入讀取內容 cat(bufio.NewReader(os.Stdin)) } // 依次讀取每一個指定文件的內容並打印到終端 for i := 0; i < flag.NArg(); i++ { f, err := os.Open(flag.Arg(i)) if err != nil { fmt.Fprintf(os.Stdout, "reading from %s failed, err:%v\n", flag.Arg(i), err) continue } cat(bufio.NewReader(f)) } }