slice元素为值类型时复制数据副本,修改互不影响;为指针类型时共享同一对象,修改全局可见;选择依据场景大小、共享需求及性能权衡。
在Go语言中,slice(切片)是引用类型,但其元素可以是指针或值类型。理解slice中指针与值类型的区别,对编写高效、安全的代码至关重要。关键在于:slice本身是引用底层数组的“窗口”,而其中存储的是元素——这些元素是值还是指针,直接影响内存使用和数据共享行为。
当slice的元素是值类型(如int、string、结构体等),每个元素都是独立的数据副本。
示例:
type Person struct {
Name string
Age int
}
people := []Person{{"Alice", 25}, {"Bob", 30}}
copyPeople := append([]Person(nil), people...) // 深拷贝值
copyPeople[0].Name = "Alicia"
// people[0].Name 仍是 "Alice"
当slice的元素是指针(如*Person),存储的是指向数据的地址。
示例:
p1 := &Person{"Alice", 25}
pointers := []*Person{p1, {Name: "Bob", Age: 30}}
alias := pointers[:1] // 共享底层数组
alias[0].Name = "Alicia"
// pointers[0].Name 也变成 "Alicia"
无论元素是指针还是值,slice的append可能导致底层数组扩容,产生新数组。
注意:
a := []int{1, 2}
b := a
b = append(b, 3)
b[0] = 99
// a[0] 仍是 1(扩容后底层数组分离)
没有绝对答案,取决于使用场景。

基本上就这些。掌握slice元素是值还是指针,能更好控制数据访问和内存行为。不复杂但容易忽略细节。