切片作为参数传递的切片指针

use*_*999 3 pointers go slice

我有以下代码:

func main() {
    var buf []byte{1, 2, 3, 4, 5}
    buf = buf[2:]
    fmt.Println(buf)
    panic(1)
}
Run Code Online (Sandbox Code Playgroud)

但是我想将指向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:]
}
Run Code Online (Sandbox Code Playgroud)

它给了我一个错误,我不能在函数的参数中使用type []byte作为类型,并且我不能切片类型.怎么了?默认情况下,切片是否通过引用传递?我尝试在没有指针的情况下执行此操作,但它不起作用 - 正在复制数组.我怎样才能做到这一点?*[]bytesliceArr()*[]byte

icz*_*cza 14

错误cannot use type []byte as type *[]byte in argument to sliceArr()来自您尚未发布的拼写错误(您尝试传递切片而不是指向切片的指针sliceArr()).

至于另一个error(cannot slice type *[]byte),你只需要使用括号将指针解除引用:

*buf = (*buf)[i:]
Run Code Online (Sandbox Code Playgroud)

你不小心=从变量声明中遗漏了一个标志.除此之外,一切都按你写的方式工作:

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:]
}
Run Code Online (Sandbox Code Playgroud)

输出(在Go Playground上试试):

[3 4 5]
Run Code Online (Sandbox Code Playgroud)

注意:

请注意,规范声明 if p是指向数组的指针,p[low:high]是简写(*p)[low:high],也就是说,指针会自动解除引用.如果p是指向切片的指针,则不会自动执行此操作,p[low:high]因为您无法切片指针,因此无效.因此,在指向切片的指针的情况下,您必须手动取消引用指针.

之所以这样,偏差是一个指针指向一个切片是非常罕见的,因为片是已经"仅仅"描述符(到底层阵列的邻接的部分),和切片没有指针的大部分时间过去了,因此,如果在稀有case(就像这个)你需要传递一个指针,你需要明确处理它.另一方面,数组代表所有元素; 分配或传递数组会复制所有值,因此使用指向数组的指针(而不是指向切片的指针)更为常见 - 因此,使用数组指针获得更多语言支持是合理的.

还要注意,您的任务可以通过只需要一个(非指针)切片作为参数类型并返回新的切片值来实现 - 当然必须在调用者处分配(这是一个很好的例子就是内置append()函数).

笔记2:

如果参数类型是切片(而不是指向切片的指针),并且传递切片,则不会复制基础数组,只复制切片标头.但是在这种情况下,如果切片参数(它是切片头的副本),并将新切片(切片头)分配给参数,则只需更改参数的值(局部变量)和不是传递的原始值(变量),这就是为什么它不起作用.

阅读以下博文,了解有关切片的更多详细信息:

去切片:用法和内部

数组,切片(和字符串):'追加'的机制