Golang 意外的 EOF
Golang unexpected EOF
这是我的代码,我是 Go 新手。
我试着用谷歌搜索这个问题,但我不能完全理解它。
我认为这与Read()
方法有关。
package main
import (
...
)
type compressor struct {
content []byte
}
func (r *compressor) compress() []byte {
...
}
func (r *compressor) decompress() []byte {
var buffer bytes.Buffer
dc := flate.NewReader(&buffer)
_, err := dc.Read(r.content)
if err != nil {
if err != io.EOF {
log.Fatal(err)
}
}
return buffer.Bytes()
}
func main() {
fileName := os.Args[1]
fmt.Println(os.Args)
contents, err := ioutil.ReadFile(fileName)
if err != nil {
log.Fatal(err)
}
fmt.Print("Uncompressed data: ")
fmt.Println(len(contents))
comp := compressor{contents}
buffer := comp.decompress()
fmt.Print("Uncompressed data: ")
fmt.Println(len(comp.decompress()))
err = ioutil.WriteFile(fileName+".decjc", buffer, 0644)
if err != nil {
log.Fatal(err)
}
}
这是输出
dylan@skynet:~/Documents/EXP/jc$ ./jc data.txt.jc
[./jc data.txt.jc]
Uncompressed data: 2364480
2018/06/29 21:41:35 unexpected EOF
你混淆了输入和输出。
flate.NewReader
将压缩输入作为一个 io.Reader
并且它 returns 一个 io.ReadCloser
可用于获取未压缩的输出:
func (r *compressor) decompress() []byte {
dc := flate.NewReader(bytes.NewReader(r.content))
defer dc.Close()
rb, err := ioutil.ReadAll(dc)
if err != nil {
if err != io.EOF {
log.Fatalf("Err %v\n read %v", err, rb)
}
}
return rb
}
在对有问题的特定代码进行跟踪后,我得到了以下答案。
func (r *Reader) ReadByte() (byte, error) {
...
if r.i >= int64(len(r.s)) {
return 0, io.EOF
}
....
}
bytes/reader 中有四个函数可以 return io.EOF
,零个函数可以 return io.ErrUnexpectedEOF
。可以returnio.EOF
的四个函数是:
Read(b []byte)
ReadAt(b []byte, off int64)
ReadByte()
ReadRune()
/src/compress/flate/inflate.go 698
func (f *decompressor) moreBits() error {
c, err := f.r.ReadByte()
if err != nil {
return noEOF(err)
}
...
}
在可以return io.EOF
的四个函数中,flate/inflate.go中只有一个函数调用其中任何一个:moreBits()
调用ReadByte()
/src/compress/flate/inflate.go 690
func noEOF(e error) error {
if e == io.EOF {
return io.ErrUnexpectedEOF
}
...
}
当 moreBits()
收到一个错误时,它调用 noEOF()
,检查它是否收到了 io.EOF
。如果是这种情况,那么 io.ErrUnexpectedEOF
就得到了 return 的支持。一切似乎都按预期工作,而且用户有责任注意这种特殊情况。建议对上面的代码进行编辑以处理似乎已定义的行为:
func (r *compressor) decompress() []byte {
dc := flate.NewReader(bytes.NewReader(r.content))
defer dc.Close()
rb, err := ioutil.ReadAll(dc)
if err != nil {
if err != io.EOF && err != io.ErrUnexpectedEOF {
log.Fatalf("Err %v\n read %v", err, rb)
}
}
return rb
}
这是在 go1.12.9
下检查的
这是我的代码,我是 Go 新手。
我试着用谷歌搜索这个问题,但我不能完全理解它。
我认为这与Read()
方法有关。
package main
import (
...
)
type compressor struct {
content []byte
}
func (r *compressor) compress() []byte {
...
}
func (r *compressor) decompress() []byte {
var buffer bytes.Buffer
dc := flate.NewReader(&buffer)
_, err := dc.Read(r.content)
if err != nil {
if err != io.EOF {
log.Fatal(err)
}
}
return buffer.Bytes()
}
func main() {
fileName := os.Args[1]
fmt.Println(os.Args)
contents, err := ioutil.ReadFile(fileName)
if err != nil {
log.Fatal(err)
}
fmt.Print("Uncompressed data: ")
fmt.Println(len(contents))
comp := compressor{contents}
buffer := comp.decompress()
fmt.Print("Uncompressed data: ")
fmt.Println(len(comp.decompress()))
err = ioutil.WriteFile(fileName+".decjc", buffer, 0644)
if err != nil {
log.Fatal(err)
}
}
这是输出
dylan@skynet:~/Documents/EXP/jc$ ./jc data.txt.jc [./jc data.txt.jc] Uncompressed data: 2364480 2018/06/29 21:41:35 unexpected EOF
你混淆了输入和输出。
flate.NewReader
将压缩输入作为一个 io.Reader
并且它 returns 一个 io.ReadCloser
可用于获取未压缩的输出:
func (r *compressor) decompress() []byte {
dc := flate.NewReader(bytes.NewReader(r.content))
defer dc.Close()
rb, err := ioutil.ReadAll(dc)
if err != nil {
if err != io.EOF {
log.Fatalf("Err %v\n read %v", err, rb)
}
}
return rb
}
在对有问题的特定代码进行跟踪后,我得到了以下答案。
func (r *Reader) ReadByte() (byte, error) {
...
if r.i >= int64(len(r.s)) {
return 0, io.EOF
}
....
}
bytes/reader 中有四个函数可以 return io.EOF
,零个函数可以 return io.ErrUnexpectedEOF
。可以returnio.EOF
的四个函数是:
Read(b []byte)
ReadAt(b []byte, off int64)
ReadByte()
ReadRune()
/src/compress/flate/inflate.go 698
func (f *decompressor) moreBits() error {
c, err := f.r.ReadByte()
if err != nil {
return noEOF(err)
}
...
}
在可以return io.EOF
的四个函数中,flate/inflate.go中只有一个函数调用其中任何一个:moreBits()
调用ReadByte()
/src/compress/flate/inflate.go 690
func noEOF(e error) error {
if e == io.EOF {
return io.ErrUnexpectedEOF
}
...
}
当 moreBits()
收到一个错误时,它调用 noEOF()
,检查它是否收到了 io.EOF
。如果是这种情况,那么 io.ErrUnexpectedEOF
就得到了 return 的支持。一切似乎都按预期工作,而且用户有责任注意这种特殊情况。建议对上面的代码进行编辑以处理似乎已定义的行为:
func (r *compressor) decompress() []byte {
dc := flate.NewReader(bytes.NewReader(r.content))
defer dc.Close()
rb, err := ioutil.ReadAll(dc)
if err != nil {
if err != io.EOF && err != io.ErrUnexpectedEOF {
log.Fatalf("Err %v\n read %v", err, rb)
}
}
return rb
}
这是在 go1.12.9