Go file 操做

概述

这里主要简单介绍了一下在 Go 中,file 的操做。html

简单的分为:读,写,其余,三部分

首先先说其余部分(建立,截取,复制,下载,hash)

我本身定义了一个检查错误的方法,不少地方会用到, 很是简单,以下:git

func check(err error){
	if err != nil {
		panic(err)
	}
}
复制代码

建立

newfile, err := os.Create(writerfilename) // 给定一个文件名称
check(err)
复制代码

会调用 os 包的 Create 方法,给定一个文件名称便可。github

截取

func truncatefile() {
	infor, _ := os.Stat(writerfilename) // 可获取文件的信息
	fmt.Printf("origin Size %v \n",infor.Size()) // 查看文件大小
	err := os.Truncate(writerfilename, 19) // 从头截取到19位
	check(err)
	infor, _ = os.Stat(writerfilename)
	fmt.Printf("changed Size %v",infor.Size())
}
复制代码

这里主要看文件大小,便可看到文件发生的变化,这里的19,是按照byte方式计算的。bash

复制

func copyfile(){
    /*
    writerfilename: 文件名称
    flag: 这里能够指定文件是读仍是写入等(可参考下面的资料)
    FileMode: 实际上是一个权限值 就像 777,666 权限同样
    */
    // os.OpenFile(writerfilename, os.O_WRONLY,0666) 
    
	originFile, err := os.Open(writerfilename)
	check(err)
	defer originFile.Close()

	newFile, err := os.Create(copy_file)
	check(err)
	defer newFile.Close()

	bytesWritten, err := io.Copy(newFile, originFile) // 把原来文件的内容拷贝到新文件中
	check(err)
	fmt.Print(bytesWritten)
}
复制代码

flg 的取值能够参考下面的枚举app

const (
	// Exactly one of O_RDONLY, O_WRONLY, or O_RDWR must be specified.
	O_RDONLY int = syscall.O_RDONLY // open the file read-only.
	O_WRONLY int = syscall.O_WRONLY // open the file write-only.
	O_RDWR   int = syscall.O_RDWR   // open the file read-write.
	// The remaining values may be or'ed in to control behavior. O_APPEND int = syscall.O_APPEND // append data to the file when writing. O_CREATE int = syscall.O_CREAT // create a new file if none exists. O_EXCL int = syscall.O_EXCL // used with O_CREATE, file must not exist. O_SYNC int = syscall.O_SYNC // open for synchronous I/O. O_TRUNC int = syscall.O_TRUNC // if possible, truncate file when opened. ) 复制代码

downloadFile

func downFile(){
	url := "http://www.cycript.org/"
	response, err := http.Get(url)
	check(err)
	defer response.Body.Close()

	downFile := "file/downFile.html"
	file, err := os.Create(downFile)
	check(err)
	w, err := io.Copy(file, response.Body)
	check(err)
	fmt.Print(w)
}
复制代码

代码也很简单,就是给定一个url,拿到body值,把buffer拷贝到新建的文件中便可。函数

hash

func hash(){
	bytes, err := ioutil.ReadFile(writerfilename)
	check(err)
	fmt.Printf("sha1.sum: %x\n", sha1.Sum(bytes))
	fmt.Printf("sha256.sum: %x\n", sha256.Sum256(bytes))
	fmt.Printf("sha512.sum: %x\n", sha512.Sum512(bytes))
	fmt.Printf("md5.sum: %x\n", md5.Sum(bytes))
}
复制代码

这些都是内置的函数,直接使用便可,也很直观方便。ui

写入(writer)

写入其实也很简单,也分为 file ioutil bufio 三种方式 首先无论哪一种方式,都要打开文件 openurl

file, err := os.OpenFile(
		filename,
		os.O_WRONLY,
		0666)
	check(err)
	defer file.Close()
复制代码

file 写入

`byteSlice` 是一个`byte`切片
1. file.WriteAt(byteSlice, 3) // 指定位置插入切片的值
2. file.Write(byteSlice) // 从头插入 len(byteSlice) 大小位
3. file.WriteString(str) // 也是从头开始插入 len(str) 长度位
复制代码

ioutil

// Quick Write to File
err = ioutil.WriteFile(filename, byteSlice, 0666)
check(err)
复制代码

快速插入的一种方式,给定一个大小,给定一个全选值便可。spa

bufio

1. bufferWrite.Write([]byte) //使用buffer 写入一个切片数据
2. bufferWrite.WriteString(str) // 写入一个字符

byteSlice := []byte{}
for i := 0; i<4092; i++ {
	byteSlice  = append(byteSlice, byte('c'))
}
	
bufferWrite := bufio.NewWriter(file)
rs, err := bufferWrite.Write(byteSlice)//bufferWrite.WriteString("今天是圣诞节")
check(err)
fmt.Print(rs)
复制代码

值得注意的是buffer默认大小空间是 4096,当数据多的时候,系统会本身分配更大的空间来存储数据。3d

使用Available() 可查看还能够填充多少数据 使用Buffered 查看已经使用了多少数据

例如:我上面建立了一个大小位4092的byteSlice,写入到文件,那么使用Available() 能够看到还剩余4个byte。

最后,bufferWrite.Reset(bufferWrite) reset函数能够重置还未进过flush到磁盘的数据, 也就是说在调用了一系列的buffer插入数据后,必须使用Flush()写入数据到磁盘,否则他们都是在buffer里面的,而在未使用 Flush() 方法以前,buffer里面的数据是能够Reset的。

读(Reader)

其实读是很简单的,不要看思惟导图那么多,其实都大同小异。

func main() {
	file, err := os.Open(filename)
	check(err)

	defer file.Close()

	infor, err := file.Stat()
	check(err)

	// file
	byteSlice := make([]byte, infor.Size())
	rs, err := file.Read(byteSlice)
	check(err)
	fmt.Printf("reade data:%s \n", byteSlice)
	fmt.Printf("byte reader:%d \n", rs)

	// io
	//bufSlice := make([]byte, infor.Size())
	//rs, err := io.ReadFull(file, bufSlice)
	//check(err)
	//fmt.Printf("reader buf data: \n%s \n", bufSlice)
	//fmt.Printf("buf data: %d \n", rs)

	// io
	//byteSlice := make([]byte, 8)
	//min := 8
	//rs, err := io.ReadAtLeast(file, byteSlice, min)
	//fmt.Printf("reader buf data: \n%s \n", byteSlice)
	//fmt.Printf("buf data: %d \n", rs)

	// io & ioutil
	/*
		ioutil.ReadFile(filename) 快速读取一个文件的内容
		ioutil.ReadAll(file)	使用 io.Reader 读取
	*/
	//bytes, err := ioutil.ReadFile(filename) // ioutil.ReadAll(file)
	//check(err)
	//fmt.Printf("reader buf data: \n%s \n", bytes)
	//fmt.Printf("buf data: %d \n", len(bytes))

	// buffer reader
	// bufio
	//bufferReader := bufio.NewReader(file) // bufio.NewReaderSize(file, 10)

	// Read
	//bufferSlice := make([]byte, infor.Size())
	//rs, err := bufferReader.Read(bufferSlice)
	//check(err)
	//fmt.Printf("%s", bufferSlice)
	//fmt.Printf("%d", rs)

	// peek
	//bytes, err := bufferReader.Peek(int(infor.Size()))
	//check(err)
	//fmt.Printf("%s", bytes)
	//fmt.Printf("%d", len(bytes))

	// Read 1 Byte
	//byte, err := bufferReader.ReadByte()
	//check(err)
	//fmt.Printf("reader 1 byte %c", byte)

	// ReadBytes return slice
	//bytes, err := bufferReader.ReadBytes('q')
	//check(err)
	//fmt.Printf("reader buf data: \n%s \n", bytes)
	//fmt.Printf("buf data: %d \n", len(bytes))

	// ReadString return string
	//bytes, err := bufferReader.ReadString('q')
	//check(err)
	//fmt.Printf("reader buf data: \n%v \n", bytes)
	//fmt.Printf("buf data: %d \n", len(bytes))

	// NewScanner
	//scanner := bufio.NewScanner(file)
	//for scanner.Scan() {
	//	fmt.Print(scanner.Text())
	//	fmt.Println()
	//}
}
复制代码

值得注意的是 NewScanner 是读取buffer里面的数据,先读取到buffer中,而后使用Scan()方法来判断是否能够继续读取,而后来 使用 Text()方法去值的

代码

相关文章
相关标签/搜索