使用`unsafe.Pointer`时,切片边界超出范围。

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

slice bounds out of range when using unsafe.Pointer

问题

当我使用类似于b[1:2]的语法将[]byte转换为字符串,然后再转换回[]byte时,我遇到了一种奇怪的恐慌情绪。

我的Go版本是go1.7.3 darwin/amd64。以下是详细的代码。

  1. package main
  2. import (
  3. "reflect"
  4. "unsafe"
  5. "fmt"
  6. )
  7. func BytesToString(b []byte) string {
  8. bh := (*reflect.SliceHeader)(unsafe.Pointer(&b))
  9. sh := reflect.StringHeader{bh.Data, bh.Len}
  10. return *(*string)(unsafe.Pointer(&sh))
  11. }
  12. func StringToBytes(s string) []byte {
  13. sh := (*reflect.StringHeader)(unsafe.Pointer(&s))
  14. bh := reflect.SliceHeader{sh.Data, sh.Len, 0}
  15. return *(*[]byte)(unsafe.Pointer(&bh))
  16. }
  17. func main() {
  18. b := []byte{'b', 'y', 't', 'e'}
  19. // No1 here you can trim []byte using b[1:2]
  20. _ = b[1:2]
  21. fmt.Println("No1")
  22. // convert []byte to string
  23. s := BytesToString(b)
  24. // convert string to []byte
  25. b = StringToBytes(s)
  26. // create new []byte variant using content of b
  27. bb := make([]byte, len(b))
  28. for i := 0; i < len(b); i++ {
  29. bb[i] = b[i]
  30. }
  31. // No2 here you also can trim []byte using bb[1:2]
  32. _ = bb[1:2]
  33. fmt.Println("No2")
  34. // No3 here you can not trim []byte. I don't know why. why?
  35. _ = b[1:2]
  36. fmt.Println("No3")
  37. }

运行这段代码,会得到以下错误信息:

  1. No1
  2. No2
  3. panic: runtime error: slice bounds out of range
  4. goroutine 1 [running]:
  5. panic(0x8f060, 0xc42000a100)
  6. /usr/local/Cellar/go/1.7.3/libexec/src/runtime/panic.go:500 +0x1a1
  7. main.main()
  8. /tmp/unsafe.go:45 +0x274
  9. exit status 2

我对导致这个恐慌的原因很好奇。

英文:

I am faced up with a curious panic when I trim a byte array using syntax like b[1:2] which is converted from []byte to string and then back to []byte.

My go version is go1.7.3 darwin/amd64. What belows is the detail code.

  1. package main
  2. import (
  3. &quot;reflect&quot;
  4. &quot;unsafe&quot;
  5. &quot;fmt&quot;
  6. )
  7. func BytesToString(b []byte) string {
  8. bh := (*reflect.SliceHeader)(unsafe.Pointer(&amp;b))
  9. sh := reflect.StringHeader{bh.Data, bh.Len}
  10. return *(*string)(unsafe.Pointer(&amp;sh))
  11. }
  12. func StringToBytes(s string) []byte {
  13. sh := (*reflect.StringHeader)(unsafe.Pointer(&amp;s))
  14. bh := reflect.SliceHeader{sh.Data, sh.Len, 0}
  15. return *(*[]byte)(unsafe.Pointer(&amp;bh))
  16. }
  17. func main() {
  18. b := []byte{&#39;b&#39;, &#39;y&#39;, &#39;t&#39;, &#39;e&#39;}
  19. // No1 here you can trim []byte using b[1:2]
  20. _ = b[1:2]
  21. fmt.Println(&quot;No1&quot;)
  22. // convert []byte to string
  23. s := BytesToString(b)
  24. // convert string to []byte
  25. b = StringToBytes(s)
  26. // create new []byte variant using content of b
  27. bb := make([]byte, len(b))
  28. for i := 0; i &lt; len(b); i++ {
  29. bb[i] = b[i]
  30. }
  31. // No2 here you also can trim []byte using bb[1:2]
  32. _ = bb[1:2]
  33. fmt.Println(&quot;No2&quot;)
  34. // No3 here you can not trim []byte. I don&#39;t know why. why?
  35. _ = b[1:2]
  36. fmt.Println(&quot;No3&quot;)
  37. }

Run this code, and get error as follows:

  1. No1
  2. No2
  3. panic: runtime error: slice bounds out of range
  4. goroutine 1 [running]:
  5. panic(0x8f060, 0xc42000a100)
  6. /usr/local/Cellar/go/1.7.3/libexec/src/runtime/panic.go:500 +0x1a1
  7. main.main()
  8. /tmp/unsafe.go:45 +0x274
  9. exit status 2

I'm curious about what caused this panic?

答案1

得分: 3

StringToBytes函数创建的切片的容量为零。

for循环不会引发恐慌,因为索引表达式会检查len(b)

表达式b[1:2]会引发恐慌,因为切片表达式会检查cap(b)

修复的方法之一是将容量设置为字符串的长度:

  1. func StringToBytes(s string) []byte {
  2. sh := (*reflect.StringHeader)(unsafe.Pointer(&s))
  3. bh := reflect.SliceHeader{sh.Data, sh.Len, sh.Len}
  4. return *(*[]byte)(unsafe.Pointer(&bh))
  5. }
英文:

The capacity of the slice created by StringToBytes is zero.

The for loop does not panic because index expressions check len(b).

The expression b[1:2] panics because slice expressions check cap(b).

One fix is to set capacity to the length of the string:

  1. func StringToBytes(s string) []byte {
  2. sh := (*reflect.StringHeader)(unsafe.Pointer(&amp;s))
  3. bh := reflect.SliceHeader{sh.Data, sh.Len, sh.Len}
  4. return *(*[]byte)(unsafe.Pointer(&amp;bh))
  5. }

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

发表评论

匿名网友

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

确定