我有以下代码:
func main() {
var buf []byte{1, 2, 3, 4, 5}
buf = buf[2:]
fmt.Println(buf)
panic(1)
}
但是我想将指向buf
字节切片的指针传递给另一个函数,并将其切片到那里,所以像这样:
func main() {
var buf []byte{1, 2, 3, 4, 5}
sliceArr(&buf, 2)
fmt.Println(buf)
panic(1)
}
func sliceArr(buf *[]byte, i int) {
*buf = *buf[i:]
}
它给我一个错误,我不能在[]byte
函数的参数中使用类型*[]byte
作为类型sliceArr()
,并且我无法对类型*[]byte
进行切片。怎么了?默认情况下,切片是否通过引用传递?我尝试在没有指针的情况下执行此操作,但它不起作用 - 正在复制数组。我怎么能这样做?
答案 0 :(得分:12)
错误cannot use type []byte as type *[]byte in argument to sliceArr()
来自您尚未发布的拼写错误(您尝试将切片而不是指向切片的指针传递给sliceArr()
)。
关于其他错误(cannot slice type *[]byte
),您只需要使用括号将指针分组解除引用:
*buf = (*buf)[i:]
你不小心从变量声明中遗漏了=
个符号。除此之外,一切都按照你的写作:
func main() {
var buf = []byte{1, 2, 3, 4, 5}
sliceArr(&buf, 2)
fmt.Println(buf)
}
func sliceArr(buf *[]byte, i int) {
*buf = (*buf)[i:]
}
输出(在Go Playground上尝试):
[3 4 5]
注意:强>
请注意specification states如果p
是指向数组的指针,p[low:high]
是(*p)[low:high]
的简写,即指针会自动解除引用。如果p
是指向切片的指针,则不会自动执行此操作,p[low:high]
无效,因为您无法切片指针。因此,在指向切片的情况下,您必须手动取消引用指针。
这种偏差的原因是指向切片的指针非常罕见,因为切片已经是“正好”描述符(对于底层数组的连续部分),并且切片在大多数时间没有指针传递,所以如果在极少数情况下(像这一个)你需要传递指针,你需要明确处理它。另一方面,数组代表所有元素;分配或传递数组会复制所有值,因此使用指向数组的指针(而不是指向切片的指针)更为常见 - 因此,使用数组指针获得更多语言支持是合理的。
另请注意,您的任务可以通过只需要一个(非指针)切片作为参数类型并返回新的切片值来实现 - 当然必须在调用者处分配(一个很好的例子就是内置{ {3}}功能)。
注意#2:
如果参数类型是切片(而不是指向切片的指针),并且传递切片,则不会复制基础数组,只会复制切片标头。但是在这种情况下,如果切片参数(它是切片头的副本),并将新切片(切片头)分配给参数,则只需更改参数的值(局部变量)和不是传递的原始值(变量),这就是为什么它不起作用。
阅读以下博客文章,了解有关切片的更多详细信息: