IO 操作是我们在编程中不可避免会遇到的,例如读写文件,Go语言的 io 包中提供了相关的接口,定义了相应的规范,不同的数据类型可以根据规范去实现相应的方法,提供更加丰富的功能。

Go 语言提倡小接口 + 接口组合的方式,来扩展程序的行为以及增加程序的灵活性。io代码包恰恰就可以作为这样的一个标杆,它可以成为我们运用这种技巧时的一个参考标准。io包中包含了大量接口,本篇文章我们就先来学习四个核心接口以及对应的接口组合。
io.Reader接口定义了 Read 方法,用于读取数据到字节数组中:
type Reader interface {
Read(p []byte) (n int, err error)
}
方法功能详解
何时返回error
方法实现和调用需注意
io.Writer接口定义了 Write 方法,用于写数据到文件中
type Writer interface {
Write(p []byte) (n int, err error)
}
方法功能详解
方法实现需注意
实现该方法后,一定不要持有字节数组p,只是用来读取数据
io.Closer接口定义了 Close 方法,该方法用于关闭连接。
方法实现需注意
第一次调用该方法后,再次调用该方法应该产生什么行为,该接口没有定义,依赖实现方法自定义。
type Closer interface {
Close() error
}
io.Seeker接口定义了 Seek 方法,该方法用于指定下次读取或者写入时的偏移量
入参:计算新偏移量的起始值 whence, 基于whence的偏移量offset
返回值:基于 whence 和 offset 计算后新的偏移量值,以及可能产生的错误
type Seeker interface {
Seek(offset int64, whence int) (int64, error)
}
方法功能详解
io包中定义了如下三种 whence
const ( SeekStart = 0 // 基于文件开始位置 SeekCurrent = 1 // 基于当前偏移量 SeekEnd = 2 // 基于文件结束位置 )
如果计算后新的偏移量,在文件起始位置之前,返回 error!=nil
任意正数的偏移量都是合法的,但是对数据源如何进行I/O操作,依赖具体的实现方法
在go语言中,可以利用接口的组合,来囊括其他接口中的方法,类似于定义了一个父接口,可以包含多个子接口。如果一个 struct 实现了所有子接口的方法,也就相当于实现了父接口。小接口 + 接口组合的方式,很大程度上增加了程序的灵活性,在我们自己业务开发过程中,可以借鉴这种做法。
针对上面四个最小粒度的接口,io包定义了如下几种组合接口:
// ReadWriter 是 Read 和 Write 方法的组合
type ReadWriter interface {
Reader
Writer
}
// ReadCloser 是 Read 和 Close 方法的组合
type ReadCloser interface {
Reader
Closer
}
// WriteCloser 是 Write 和 Close 方法的组合
type WriteCloser interface {
Writer
Closer
}
// ReadWriteCloser 是 Read、Write 和 Close 方法的组合
type ReadWriteCloser interface {
Reader
Writer
Closer
}
// ReadSeeker 是 Read 和 Seek 方法的组合
type ReadSeeker interface {
Reader
Seeker
}
// WriteSeeker 是 Write 和 Seek 方法的组合
type WriteSeeker interface {
Writer
Seeker
}
// ReadWriteSeeker 是 Read、Write 和 Seek 方法的组合
type ReadWriteSeeker interface {
Reader
Writer
Seeker
}
本篇文章介绍了 io包 中的四大核心接口:
可以看到 Reader 和 Writer 接口中定义的方法中,都有字节数组p,而底层要操作的文件在方法中都没有体现。Read方法是将文件的数据读入字节数组p,Write 是将字节数组p的数据写入文件,这一点不要记混。