кодирование гоба падает - PullRequest
0 голосов
/ 09 марта 2019

Я хотел разработать type data struct для реализации хранилища на диске. Я столкнулся с проблемой при кодировании объекта данных в байтах. Для кодирования я хотел использовать пакет encoding/gob.

package data

import (
    "encoding/gob"
    "io"
)
func NewKey(key string) Data {
    ret := Data{MsgType: "key", Value: key}
    return ret
}

func NewData(d string) Data {
    ret := Data{MsgType: "data", Value: d}
    return ret
}

type Data struct {
    MsgType string
    Value   string
}

func (dat Data) Encode () io.Writer {
    var ret io.Writer
    enc := gob.NewEncoder(ret)
    _ = enc.Encode(dat)
    // if err != nil {
    //  panic(err)
    // }
    return ret
}

Это код пакета данных. Моя цель - преобразовать тип Data в байтовый фрагмент, чтобы я мог сохранить его на diskv. Мой основной код выглядит так: основной пакет

import (
    "fmt"
    "pkg"
)

func main() {
    testData := data.Data{MsgType:"string", Value:"1"}
    ret := testData.Encode()
    // fmt.Println(ret)
    fmt.Sprintf("%T", ret)
}

Но когда я запускаю код с go run, я получаю следующий вывод:

panic: runtime error: invalid memory address or nil pointer dereference
[signal 0xc0000005 code=0x0 addr=0x18 pc=0x4a15b4]

goroutine 1 [running]:
encoding/gob.(*Encoder).writeMessage(0xc0000440a0, 0x0, 0x0, 0xc0000440d8)
    C:/Program Files/Go/src/encoding/gob/encoder.go:82 +0x134
encoding/gob.(*Encoder).sendActualType(0xc0000440a0, 0x0, 0x0, 0xc000037600, 0xc0000375c0, 0x506800, 0x4d2580, 0x38)
    C:/Program Files/Go/src/encoding/gob/encoder.go:107 +0x223
encoding/gob.(*Encoder).sendType(0xc0000440a0, 0x0, 0x0, 0xc000037600, 0x506800, 0x4d2580, 0xc000067de0)
    C:/Program Files/Go/src/encoding/gob/encoder.go:169 +0xda
encoding/gob.(*Encoder).sendTypeDescriptor(0xc0000440a0, 0x0, 0x0, 0xc000037600, 0xc0000375c0)
    C:/Program Files/Go/src/encoding/gob/encoder.go:191 +0xd9
encoding/gob.(*Encoder).EncodeValue(0xc0000440a0, 0x4d2580, 0xc000004c00, 0x99, 0x0, 0x0)
    C:/Program Files/Go/src/encoding/gob/encoder.go:244 +0x2f3
encoding/gob.(*Encoder).Encode(0xc0000440a0, 0x4d2580, 0xc000004c00, 0xc000004c00, 0x0)
    C:/Program Files/Go/src/encoding/gob/encoder.go:176 +0xad
pkg.Data.Encode(0x4e66e9, 0x6, 0x4e614a, 0x1, 0x0, 0x0)
    C:/Users/eliap/OneDrive/Dokumente/Programming/Go/src/pkg/data.go:25 +0xa2
main.main()
    C:/Users/eliap/OneDrive/Dokumente/Programming/Go/src/test.go:10 +0x56
exit status 2
[Finished in 1.1s with exit code 1]
[cmd: ['go', 'run', 'C:\\Users\\eliap\\OneDrive\\Dokumente\\Programming\\Go\\src\\test.go']]
[dir: C:\Users\eliap\OneDrive\Dokumente\Programming\Go\src]
[path: C:\ProgramData\DockerDesktop\version-bin;C:\Program Files\Docker\Docker\Resources\bin;C:\Program Files\Python37\;C:\Windows\System32;C:\Program Files\Typora\bin;C:\Windows;C:\Windows\System32\Wbem;C:\Windows\System32\WindowsPowerShell\v1.0\;C:\Windows\System32\OpenSSH\;C:\Program Files\Python37\Scripts;C:\MinGW\bin;C:\Program Files\Go\bin;C:\Program Files\Git\cmd;C:\Program Files\micro-1.4.1;C:\Program Files\Microsoft VS Code\bin;C:\Program Files (x86)\Midnight Commander;C:\Ruby26-x64\bin;C:\Users\eliap\AppData\Local\Microsoft\WindowsApps;C:\Users\eliap\go\bin]

Я использую go version go1.12 windows/amd64, Windows 10 и Sublime Text 3 (если это имеет значение) Если у вас есть лучшее решение / реализация для метода Encode, это было бы еще лучше.

Ответы [ 2 ]

3 голосов
/ 09 марта 2019

В

func (dat Data) Encode () io.Writer {
    var ret io.Writer
    enc := gob.NewEncoder(ret)

вы

  1. Объявить переменную ret типа io.Writer
  2. Не инициализируйте его, поэтому для его типа у него нулевое значение, nil.
  3. Вы передаете это nil в gob.NewEncoder, которое предположительно записывает это значение во внутреннее состояние.
  4. Как только вы пытаетесь что-то кодировать, используя этот кодер, и он решает, что пришло время действительно что-то записать в предоставленный io.Writer, он завершается с ошибкой «неверный адрес памяти или разыменование нулевого указателя».
2 голосов
/ 09 марта 2019

Проблема в том, что вы пытаетесь записать в ноль io.Writer:

var ret io.Writer
enc := gob.NewEncoder(ret)

ret ноль. Так что это будет паниковать (как это происходит), когда пытается написать в него.

Обычный способ написать Encode() метод - это взять io.Writer в качестве аргумента:

func (dat Data) Encode(w io.Writer) error {
    enc := gob.NewEncoder(ret)
    return enc.Encode(dat)
}

Тогда вы бы вызвали этот метод с существующим io.Writer (который может быть файлом, или сетевым сокетом, или чем-то еще):

file, err := os.Create("somefile")
...
if err := dat.Encode(file); err != nil {
    fmt.Printf("Failed to encode: %s\n", err)
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...