計算機中文件時存儲在外部介質(通常是磁盤)上的數據集合,文件分為文本文件和二進制文件。打開和關閉文件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注冊文件關閉語句。
讀取文件一、file.Read()1.1 基本使用Read方法定義如下:
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循環讀取文件中所有數據。
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,支持更多的功能。
package main
import (
"bufio"
"fmt"
"io"
"os"
)
// bufio按行讀取示例
func main() {
file, err := os.Open("./xx.txt")
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 {
if len(line) != 0 {
fmt.Println(line)
}
fmt.Println("文件讀完了")
break
}
if err != nil {
fmt.Println("read file failed, err:", err)
return
}
fmt.Print(line)
}
}
Io/ioutil包的ReadFile方法能夠讀取完整的文件,隻需要将文件名作為參數傳入。
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:打開文件的模式。 模式有以下幾種:
perm:文件權限,一個八進制數。r(讀)04,w(寫)02,x(執行)01。
四、Write和WriteString
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.open("user.txt")
if err == nil {
defer file.close()
scanner := bufio.NewScanner(file)
for scanner.Scan() {
fmt.Println(scanner.Text())
}
}
}
func main() {
file, err := os.open("user.txt")
if err == nil {
defer file.close()
reader := bufio.NewReader(file)
for {
line, err := reader.ReaderString('\n')
if err != nil {
if err != io.EOF {
fmt.Println(err)
}
break
} else {
fmt.Println(err, line)
}
}
}
}
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 {
// 退出之前将已讀到的内容輸出
fmt.Fprintf(os.Stdout, "%s", buf)
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))
}
}
更多精彩资讯请关注tft每日頭條,我们将持续为您更新最新资讯!