在Golang中,指针类型的切片指的是切片的元素类型为指针的切片结构,这种结构在实际开发中常用于需要批量操作多个同类型对象的场景,能够有效减少值拷贝带来的性能开销,同时也方便对多个独立对象进行修改。理解指针类型切片的特性和操作方法,是Golang进阶开发的必备技能。

什么是指针类型的切片
普通的切片元素可以是任意基础类型或者自定义类型,当切片的元素类型为某个类型的指针时,这个切片就是指针类型的切片。比如*int类型的切片,每个元素都存储着一个int类型变量的内存地址,而不是int值本身。我们可以通过下面的代码直观感受指针类型切片的声明方式:
package main
import "fmt"
func main() {
// 声明一个元素为*int类型的切片
var intPtrSlice []*int
fmt.Printf("切片类型: %Tn", intPtrSlice)
}
指针类型切片的初始化方式
指针类型切片的初始化和普通切片类似,只是元素需要是指针类型,常见的初始化方式有以下几种:
直接字面量初始化
可以在声明时直接给切片赋值指针元素,需要先准备对应的变量,再取地址放入切片:
package main
import "fmt"
func main() {
a := 10
b := 20
// 直接初始化指针类型切片
intPtrSlice := []*int{&a, &b}
fmt.Println("切片元素指向的值:", *intPtrSlice[0], *intPtrSlice[1])
}
使用make初始化后赋值
可以先通过make创建指定长度和容量的指针切片,再逐个给元素赋值指针:
package main
import "fmt"
func main() {
// 创建长度为2的*int类型切片
intPtrSlice := make([]*int, 2)
a := 30
b := 40
intPtrSlice[0] = &a
intPtrSlice[1] = &b
fmt.Println("切片元素指向的值:", *intPtrSlice[0], *intPtrSlice[1])
}
修改指针类型切片的元素内容
操作指针类型切片时,修改操作分为两种情况,一种是修改指针指向的变量的值,另一种是修改切片中指针本身的指向,下面分别说明。
修改指针指向的变量值
如果我们要修改切片中某个元素指针指向的变量的内容,只需要通过解引用操作符*找到对应的变量,再赋值即可,这种修改不会改动切片中指针的地址,只会改变指针指向的内存空间里存储的值:
package main
import "fmt"
func main() {
a := 10
intPtrSlice := []*int{&a}
fmt.Println("修改前的值:", *intPtrSlice[0])
// 修改指针指向的变量的值
*intPtrSlice[0] = 100
fmt.Println("修改后的值:", *intPtrSlice[0])
// 原始变量的值也会被同步修改
fmt.Println("原始变量a的值:", a)
}
修改切片中指针的指向
如果我们需要让切片中某个元素的指针指向新的变量,直接给对应索引的元素赋值新的指针地址即可:
package main
import "fmt"
func main() {
a := 10
b := 20
intPtrSlice := []*int{&a}
fmt.Println("修改前指针指向的值:", *intPtrSlice[0])
// 修改切片中第一个元素的指针指向,改为指向b
intPtrSlice[0] = &b
fmt.Println("修改后指针指向的值:", *intPtrSlice[0])
// 原始变量a的值不会受到影响
fmt.Println("原始变量a的值:", a)
}
遍历修改指针类型切片的元素
实际开发中经常需要遍历指针类型切片批量修改元素,遍历时需要注意如果通过值拷贝的方式拿到指针元素,修改指针指向的变量值会影响所有指向该变量的指针,而修改指针本身的指向只会影响当前遍历到的切片元素:
package main
import "fmt"
func main() {
a := 10
b := 20
c := 30
intPtrSlice := []*int{&a, &b, &c}
// 遍历修改每个指针指向的变量值为原来的2倍
for i := range intPtrSlice {
*intPtrSlice[i] = *intPtrSlice[i] * 2
}
// 打印修改后的结果
for _, ptr := range intPtrSlice {
fmt.Println(*ptr)
}
}
操作注意事项
- 不要对nil指针进行解引用操作,否则会触发panic,操作前可以先判断指针是否为nil。
- 切片扩容时,指针类型的切片元素会被拷贝,拷贝后新旧切片中对应索引的指针指向同一个变量,修改其中一个指针指向的变量值,另一个切片中对应元素也会感知到变化。
- 如果指针类型切片的元素是自定义结构体指针,修改结构体字段时同样通过解引用操作即可,不需要额外处理。
指针类型切片的核心在于区分修改的是指针指向的内容,还是指针本身的指向,只要理清这两者的区别,大部分操作问题都可以避免。