Go语言采坑记录gob序列化坑

package main

import (
	"bytes"
	"encoding/gob"
	"fmt"
	"log"
	"os"
)

func main() {
	/////////////// 序列化编码&写入文件开始 ///////////////
	// 实例化一个学生对象s1
	s1 := &Student{id: 1, name: "jack ma", age: 2, cash: 43242334.35}

	// 实例化一个gob编码对象,并对s1学生对象进行编码
	var buf bytes.Buffer
	encoder := gob.NewEncoder(&buf)
	error2Log("Encode: ", encoder.Encode(s1))

	// 输出序列化后的类容
	fmt.Printf("Serialized: %x\n", buf.Bytes())

	// 创建data.data文件,存放序列化数据
	fs, err := os.Create("./data.data")
	error2Log("Creat File: ", err)
	// 写入数据,并打印写入数据长度
	wlen, err := fs.Write(buf.Bytes())
	error2Log("Write file: ", err)
	fmt.Println("Wirte bytes:", wlen)
	fs.Close()

	/////////////// 序列化编码&写入文件结束 ///////////////

	/////////////// 读入文件&反序列化编码开始 ///////////////

	// 从文件打开写入的序列化数据
	fs, err = os.Open("./data.data")
	error2Log("Open file: ", err)
	// 获取文件信息接口
	finfo, err := fs.Stat()
	error2Log("Get fileinfo: ", err)
	// 根据文件大小创建字节数组,然后将文件中数据读入到字节数组中,并打印读入数据长度
	readBuf := make([]byte, finfo.Size())
	rlen, err := fs.Read(readBuf)
	error2Log("Read file: ", err)
	fmt.Println("Read bytes:", rlen)
	fs.Close()

	// 实例化一个用于接收反序列化数据的学生对象
	s2 := new(Student)

	// 实例化一个gob解码对象
	decoder := gob.NewDecoder(bytes.NewReader(readBuf))
	// 将数据解码到学生对象s2
	error2Log("Decode: ", decoder.Decode(s2))
	// 将反序列化后的数据,s2学生对象打印出来
	fmt.Println("Unserialized:", s2)

	/////////////// 读入文件&反序列化编码结束 ///////////////
}

func error2Log(m string, e error) {
	if e != nil {
		log.Fatal(m, e)
	}
}

// Student 学生对象
type Student struct {
	id   uint64
	name string
	age  uint
	cash float64
}

编译后运行结构如下:

2019/04/17 15:07:46 Encode: gob: type main.Student has no exported fields

害的我郁闷了一个多小时。查询了一些信息,比如:
https://github.com/golang/go/issues/5819
但是这不是我的需要的问题解决方法,我就想将一个结构体对象写到文件中去,也没打算嵌入结构体呀。
后来突然灵光一闪,我犯了个超级低级错误:

type Student struct {
	id   uint64
	name string
	age  uint
	cash float64
}

尼玛… 结构体成员(golang叫字段)小写…,突然想起来关于包级的变量名和函数名,小写相当于private… 难道是这样?
马上改了:

type Student struct {
	ID   uint64
	Name string
	Age  uint
	Cash float64
}

14行那里的实例化s1改成:

s1 := &Student{ID: 1, Name: "jack ma", Age: 2, Cash: 43242334.35}

然后编译运行结构终于正确:

Serialized: 36ff810301010753747564656e7401ff820001040102494401060001044e616d65010c00010341676501060001044361736801080000001aff82010101076a61636b206d61010201f8cdccccf29a9e844100
Wirte bytes: 82
Read bytes: 82
Unserialized: &{1 jack ma 2 4.324233435e+07}

这相当于回过头来看,在golang中没有养成的代码习惯。
所以今后写结构体的时候,得养成好习惯:字段名应该使用首字母大写。(555555555555555~)

你可能感兴趣的:(Go语言)