我们如何知道字节数组中的结构类型呢?

huangapple go评论121阅读模式
英文:

How we can know what kind of struct is on byte array

问题

我正在寻找一种方法来确定哈希的结构类型。有没有可能在不尝试错误方法(将其转换为特定类型并查看转换是否成功)的情况下实现这一点?

请检查以下代码:

  1. import (
  2. "bytes"
  3. "encoding/binary"
  4. "fmt"
  5. "reflect"
  6. )
  7. type T struct {
  8. A int64
  9. B float64
  10. }
  11. type D struct {
  12. A int64
  13. B float64
  14. C string
  15. }
  16. func main() {
  17. // 创建一个结构体并将其写入缓冲区。
  18. t := T{A: 0xEEFFEEFF, B: 3.14}
  19. buf := &bytes.Buffer{}
  20. err := binary.Write(buf, binary.BigEndian, t)
  21. if err != nil {
  22. panic(err)
  23. }
  24. fmt.Println(buf.Bytes())
  25. out := getType(buf)
  26. fmt.Println(out)
  27. }
  28. func getType(v interface{})(r string){
  29. fmt.Println(reflect.TypeOf(v))
  30. switch t := v.(type) {
  31. case T:
  32. return "是类型 T"
  33. case D:
  34. return "是类型 D"
  35. default:
  36. _ = t
  37. return "未知类型";
  38. }
  39. }

希望这可以帮助你。

英文:

I'm looking for some solution to know whats the struct type of the hash. It is possible to do that without try an error method (casting to a specific type and see the cast is successfully)?

Please check the code:

  1. import (
  2. "bytes"
  3. "encoding/binary"
  4. "fmt"
  5. "reflect"
  6. )
  7. type T struct {
  8. A int64
  9. B float64
  10. }
  11. type D struct {
  12. A int64
  13. B float64
  14. C string
  15. }
  16. func main() {
  17. // Create a struct and write it.
  18. t := T{A: 0xEEFFEEFF, B: 3.14}
  19. buf := &bytes.Buffer{}
  20. err := binary.Write(buf, binary.BigEndian, t)
  21. if err != nil {
  22. panic(err)
  23. }
  24. fmt.Println(buf.Bytes())
  25. out := getType(buf)
  26. fmt.Println(out)
  27. }
  28. func getType(v interface{})(r string){
  29. fmt.Println(reflect.TypeOf(v))
  30. switch t := v.(type) {
  31. case T:
  32. return "Is type T"
  33. case D:
  34. return "Is type D"
  35. default:
  36. _ = t
  37. return "unknown"
  38. }
  39. }

答案1

得分: 1

由于encoding/binary包不会写出类型信息,因此无法确定写入/序列化的类型。

而且你的处境比你最初想的还要糟糕:即使尝试将其解码为不同类型的值,也可能成功而没有错误,因此甚至没有可靠的方法来确定类型。

例如,如果你序列化了这种类型的值:

  1. type T struct {
  2. A int64
  3. B float64
  4. }

你可以将其读入这种类型的值:

  1. type T2 struct {
  2. B float64
  3. A int64
  4. }

它不会报错,因为两个结构体的大小相同,但显然字段中的数字是不同的。

如果你使用encoding/gob,你的处境会稍微好一些,因为gob包会传输类型信息,将类型为T的值编码然后解码为类型为T2的值是可行的:字段的顺序不重要,额外或缺少的字段也不会引起问题。

看看这个例子:

  1. // 创建一个结构体并写入它。
  2. t := T{A: 0xEEFFEEFF, B: 3.14}
  3. fmt.Println("编码:", t)
  4. buf := &bytes.Buffer{}
  5. fmt.Println(binary.Write(buf, binary.BigEndian, t))
  6. fmt.Println(buf.Bytes())
  7. fmt.Println(gob.NewEncoder(buf).Encode(t))
  8. t2 := T2{}
  9. fmt.Println(binary.Read(buf, binary.BigEndian, &t2))
  10. fmt.Println(t2)
  11. t2 = T2{}
  12. fmt.Println(gob.NewDecoder(buf).Decode(&t2))
  13. fmt.Println(t2)

输出结果(在Go Playground上尝试):

  1. 编码: {4009750271 3.14}
  2. <nil>
  3. [0 0 0 0 238 255 238 255 64 9 30 184 81 235 133 31]
  4. <nil>
  5. <nil>
  6. {1.9810798573e-314 4614253070214989087}
  7. <nil>
  8. {3.14 4009750271}

如果你想在读取之前能够检测到类型,你必须自己处理:你必须传输类型信息(例如类型的名称)。或者更好的是,使用一个已经实现了这一点的序列化方法,例如Google的协议缓冲区,这里是它的Go实现:github.com/golang/protobuf

英文:

Since the encoding/binary package does not write out type information, it is not possible to tell what type was written / serialized.

And you're in a worse position that you might originally think: even trying to decode into a value of different type might succeed without errors, so there isn't even a reliable way to tell the type.

For example if you serialize a value of this type:

  1. type T struct {
  2. A int64
  3. B float64
  4. }

You can read it into a value of this type:

  1. type T2 struct {
  2. B float64
  3. A int64
  4. }

It will give no errors because the size of both structs is the same, but obviously you will get different numbers in the fields.

You are in a little better position if you use encoding/gob, as the gob package does transmit type information, and encoding a value of type T and then decoding it into a value of type T2 would work: order of fields does not matter, and extra or missing fields also do not cause trouble.

See this example:

  1. // Create a struct and write it.
  2. t := T{A: 0xEEFFEEFF, B: 3.14}
  3. fmt.Println(&quot;Encoding:&quot;, t)
  4. buf := &amp;bytes.Buffer{}
  5. fmt.Println(binary.Write(buf, binary.BigEndian, t))
  6. fmt.Println(buf.Bytes())
  7. fmt.Println(gob.NewEncoder(buf).Encode(t))
  8. t2 := T2{}
  9. fmt.Println(binary.Read(buf, binary.BigEndian, &amp;t2))
  10. fmt.Println(t2)
  11. t2 = T2{}
  12. fmt.Println(gob.NewDecoder(buf).Decode(&amp;t2))
  13. fmt.Println(t2)

Output (try it on the Go Playground):

  1. Encoding: {4009750271 3.14}
  2. &lt;nil&gt;
  3. [0 0 0 0 238 255 238 255 64 9 30 184 81 235 133 31]
  4. &lt;nil&gt;
  5. &lt;nil&gt;
  6. {1.9810798573e-314 4614253070214989087}
  7. &lt;nil&gt;
  8. {3.14 4009750271}

If you want to be able to detect the type before reading it, you have to take care of it yourself: you have to transmit type information (e.g. name of the type). Or even better, use a serialization method that already does this, for example Google's protocol buffers, and here is the Go implementation for it: github.com/golang/protobuf.

huangapple
  • 本文由 发表于 2017年1月11日 18:59:25
  • 转载请务必保留本文链接:https://go.coder-hub.com/41589055.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定