Go語(yǔ)言編程中對(duì)文件讀寫(xiě)的基本方法整理
1.func Copy(dst Writer, src Reader) (written int64, err error)這個(gè)函數(shù)是從一個(gè)文件讀取拷貝到另外一個(gè)文件,一直拷貝到讀取文件的EOF,所以不會(huì)返回io.EOF錯(cuò)誤,參數(shù)是寫(xiě)入目標(biāo)器和讀取目標(biāo)器,返回int64的拷貝字節(jié)數(shù)和err信息
import (
"fmt"
"io"
"os"
)
func main() {
r, _ := os.Open("test.txt")
w, _ := os.Create("write.txt")
num, err := io.Copy(w, w)
if err != nil {
fmt.Println(err)
}
fmt.Println(num) //返回int64的11 打開(kāi)我的write.txt正是test.txt里邊的hello widuu
}
2.func CopyN(dst Writer, src Reader, n int64) (written int64, err error)看函數(shù)就知道了跟上述的是一樣的,只是多加了一個(gè)讀取數(shù)的限制,然后我們看下代碼
import (
"fmt"
"io"
"io/ioutil"
"os"
)
func main() {
r, _ := os.Open("test.txt")
w, _ := os.Create("write1.txt")
num, err := io.CopyN(w, r, 5)
if err != nil {
fmt.Println(err)
}
defer r.Close()
b, _ := ioutil.ReadFile("write1.txt")
fmt.Println(string(b)) //輸出 hello
fmt.Println(num) //5
}
3.func ReadAtLeast(r Reader, buf []byte, min int) (n int, err error)這個(gè)函數(shù)就是從讀取器中讀取數(shù)據(jù)放到我們的buf中,限定了最小的讀取字節(jié)數(shù),如果我們讀取的數(shù)據(jù)小于最小讀取器,譬如你設(shè)定min的值是8,但是你讀取的數(shù)據(jù)字節(jié)數(shù)是5就會(huì)返回一個(gè)`io.ErrUnexpectedEOF`,如果大于就會(huì)返回`io.ErrShortBuffer`,讀取完畢會(huì)有`io.EOF`~~,多講一些哈,這個(gè)Reader只要我們滿足這個(gè)interface就可以用這個(gè)
type Reader interface {
Read(p []byte) (n int, err error)
}
其中*File就支持func (f *File) Read(b []byte) (n int, err error)
import (
"fmt"
"io"
"os"
)
func main() {
r, _ := os.Open("write1.txt")
b := make([]byte, 20)
defer r.Close()
var total int
for {
n, err := io.ReadAtLeast(r, b, 8)
if err == nil {
fmt.Println("Read enough value:", string(b)) // Read enough value: hello widuu
}
if err == io.ErrUnexpectedEOF { //讀取了的數(shù)據(jù)小于我們限定的最小讀取數(shù)據(jù)8
fmt.Println("Read fewer value:", string(b[0:n]))
}
if err == io.ErrShortBuffer{ //這個(gè)是我們?cè)O(shè)定的buf也就是b小于我們限定8
fmt.Println("buf too Short")
os.Exit(1)
}
if err == io.EOF { //讀完了 輸出
fmt.Println("Read end total", total) //Read end total 11
break
}
total = total + n
}
}
4.func ReadFull(r Reader, buf []byte) (n int, err error)這個(gè)函數(shù)和上邊的函數(shù)是相似,只不過(guò)是讀取len(buf)個(gè),放在buf中
import (
"fmt"
"io"
"os"
)
func main() {
r, _ := os.Open("write.txt")
b := make([]byte, 20)
num, err := io.ReadFull(r, b)
defer r.Close()
if err == io.EOF {
fmt.Println("Read end total", num)
}
if err == io.ErrUnexpectedEOF {
fmt.Println("Read fewer value:", string(b[:num])) //Read fewer value: hello widuu,依然是buf長(zhǎng)度大于讀取的長(zhǎng)度
return
}
fmt.Println("Read value:", string(b)) //如果b是5 就出現(xiàn)這里
}
5.func WriteString(w Writer, s string) (n int, err error)弄完讀了,當(dāng)然帶要寫(xiě)了,這個(gè)函數(shù)主要是向?qū)懭肽繕?biāo)中寫(xiě)入字符創(chuàng),返回是寫(xiě)入的字節(jié)數(shù)還有error錯(cuò)誤,主要是權(quán)限的錯(cuò)誤,其中寫(xiě)入呀!都是writer這個(gè)結(jié)構(gòu)就可以寫(xiě)入
type Writer interface {
Write(p []byte) (n int, err error)
}
跟read一樣我們的*File是有func (f *File) Write(b []byte) (n int, err error),當(dāng)然其實(shí)我們的*File中就已經(jīng)有WirteString了func (f *File) WriteString(s string) (ret int, err error)
import (
"fmt"
"io"
"io/ioutil"
"os"
)
func main() {
w, _ := os.OpenFile("write1.txt", os.O_RDWR, os.ModePerm)
n, err := io.WriteString(w, "ni hao ma")
if err != nil {
fmt.Println(err) //當(dāng)我用os.open()的時(shí)候木有權(quán)限 悲催的~~輸出write write1.txt: Access is denied.
}
defer w.Close()
b, _ := ioutil.ReadFile("write1.txt")
fmt.Println("write total", n) //write total 9
fmt.Println(string(b)) // ni hao ma
}
6.type LimitedReader
type LimitedReader struct {
R Reader // 讀取器了
N int64 // 最大字節(jié)限制
}
只實(shí)現(xiàn)了一個(gè)方法func (l *LimitedReader) Read(p []byte) (n int, err error)其實(shí)我們不難發(fā)現(xiàn)這個(gè)跟我們的ReadAtLast()就是親兄弟的節(jié)奏
import (
"fmt"
"io"
"os"
)
func main() {
reader, _ := os.Open("test.txt")
limitedreader := io.LimitedReader{
R: reader,
N: 20,
}
p := make([]byte, 10)
var total int
for {
n, err := limitedreader.Read(p)
if err == io.EOF {
fmt.Println("read total", total) //read total 11
fmt.Println("read value", string(p)) //read value hello widuu
break
}
total = total + n
}
}
7.type PipeReader
type PipeReader struct {
// contains filtered or unexported fields
}
(1)func Pipe() (*PipeReader, *PipeWriter)創(chuàng)建一個(gè)管道,并返回它的讀取器和寫(xiě)入器,這個(gè)會(huì)在內(nèi)存中進(jìn)行管道同步,它的開(kāi)啟會(huì)io.Reader然后等待io.Writer的輸入,沒(méi)有內(nèi)部緩沖,它是安全的調(diào)用Read和Write彼此和并行調(diào)用寫(xiě)
import (
"fmt"
"io"
"reflect"
)
func main() {
r, w := io.Pipe()
fmt.Println(reflect.TypeOf(r)) //*io.PipeReader
fmt.Println(reflect.TypeOf(w)) //*io.PipeWriter
}
(2)func (r *PipeReader) Close() error管道關(guān)閉后,正在進(jìn)行或后續(xù)的寫(xiě)入Write操作返回ErrClosedPipe
import (
"fmt"
"io"
)
func main() {
r, w := io.Pipe()
r.Close()
_, err := w.Write([]byte("hello widuu"))
if err == io.ErrClosedPipe {
fmt.Println("管道已經(jīng)關(guān)閉無(wú)法寫(xiě)入") //管道已經(jīng)關(guān)閉無(wú)法寫(xiě)入
}
}
(3)func (r *PipeReader) CloseWithError(err error) error這個(gè)就是上邊的r.Close關(guān)閉的時(shí)候,寫(xiě)入器會(huì)返回錯(cuò)誤的信息
import (
"errors"
"fmt"
"io"
)
func main() {
r, w := io.Pipe()
r.Close()
err := errors.New("管道符關(guān)閉了") //errors這個(gè)包我們前邊已經(jīng)說(shuō)過(guò)了,就一個(gè)方法New不會(huì)的可以看看前邊的
r.CloseWithError(err)
_, err = w.Write([]byte("test"))
if err != nil {
fmt.Println(err) //管道符關(guān)閉了
}
}
(4)func (r *PipeReader) Read(data []byte) (n int, err error)標(biāo)準(zhǔn)的閱讀接口,它從管道中讀取數(shù)據(jù)、阻塞一直到一個(gè)寫(xiě)入接口關(guān)閉,如果寫(xiě)入端發(fā)生錯(cuò)誤,它就會(huì)返回錯(cuò)誤,否則返回的EOF
import (
"fmt"
"io"
)
func main() {
r, w := io.Pipe()
go w.Write([]byte("hello widuu"))
d := make([]byte, 11)
n, _ := r.Read(d) //從管道里讀取數(shù)據(jù)
fmt.Println(string(d))
fmt.Println(n)
}
相關(guān)文章
Go語(yǔ)言中的匿名結(jié)構(gòu)體用法實(shí)例
這篇文章主要介紹了Go語(yǔ)言中的匿名結(jié)構(gòu)體用法,實(shí)例分析了匿名結(jié)構(gòu)體的使用技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-02-02Golang觀察者模式優(yōu)化訂單處理系統(tǒng)實(shí)例探究
當(dāng)涉及到訂單處理系統(tǒng)時(shí),觀察者設(shè)計(jì)模式可以用于實(shí)現(xiàn)訂單狀態(tài)的變化和通知,在這篇文章中,我們將介紹如何使用Golang來(lái)實(shí)現(xiàn)觀察者設(shè)計(jì)模式,并提供一個(gè)基于訂單處理系統(tǒng)的代碼示例2024-01-01Golang實(shí)現(xiàn)簡(jiǎn)易的rpc調(diào)用
RPC指(Remote Procedure Call Protocol)遠(yuǎn)程過(guò)程調(diào)用協(xié)議。本文將實(shí)現(xiàn)利用Golang進(jìn)行rpc調(diào)用(只實(shí)現(xiàn)一個(gè)rpc框架基本的功能,不對(duì)性能做保證),需要的可以參考一下2023-03-03golang提示dial?tcp?172?.217.163.49:443:?connectex:?A?con
這篇文章主要為大家介紹了golang提示dial?tcp?172?.217.163.49:443:?connectex:?A?connection?attempt?failed解決,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-07-07Mac下Vs code配置Go語(yǔ)言環(huán)境的詳細(xì)過(guò)程
這篇文章給大家介紹Mac下Vs code配置Go語(yǔ)言環(huán)境的詳細(xì)過(guò)程,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2021-07-07Go?語(yǔ)言前綴樹(shù)實(shí)現(xiàn)敏感詞檢測(cè)
這篇文章主要為大家介紹了Go語(yǔ)言前綴樹(shù)實(shí)現(xiàn)敏感詞檢測(cè)實(shí)例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-08-08手把手教你如何在Goland中創(chuàng)建和運(yùn)行項(xiàng)目
歡迎來(lái)到本指南!我們將手把手地教您在Goland中如何創(chuàng)建、配置并運(yùn)行項(xiàng)目,通過(guò)簡(jiǎn)單的步驟,您將迅速上手這款強(qiáng)大的集成開(kāi)發(fā)環(huán)境(IDE),輕松實(shí)現(xiàn)您的編程夢(mèng)想,讓我們一起開(kāi)啟這段精彩的旅程吧!2024-02-02一文帶你掌握掌握 Golang結(jié)構(gòu)體與方法
在 Golang 中,結(jié)構(gòu)體和方法是實(shí)現(xiàn)面向?qū)ο缶幊痰闹匾M成部分,也是 Golang 的核心概念之一。在本篇文章中,我們將深入介紹 Golang 結(jié)構(gòu)體與方法的概念、使用方法以及相關(guān)的編程技巧和最佳實(shí)踐2023-04-04