切片传参默认复制结构体但指针仍指向原数组,故sort.Ints等可原地修改;仅当需避免大对象拷贝或修改元素字段时才用[]*T;sort.Slice通过索引操作零分配排序。
直接对切片底层数组操作,用指针避免复制——核心是传切片本身(引用语义)而非其副本,并在必要时传递元素指针来原地修改。
Go 中切片是包含 指向底层数组的指针、长度、容量 的结构体。每次传切片给函数,默认是复制这个结构体(3个字段),但其中的指针仍指向同一数组。所以排序函数如 sort.Ints 或自定义排序,不需要显式传指针就能原地修改元素。
sort.Ints(nums) —— 修改的是原数组内容sortInts(&nums) 或 func sortInts(ptr *[]
int) —— 多余且易出错当切片元素是大型结构体,或排序逻辑需修改字段本身(而非仅重排位置),才考虑用元素指针切片([]*T)。这样既能避免拷贝大对象,又可原地更新字段。
students := []*Student{&s1, &s2, &s3},再对 students 排序;排序后遍历 students 直接改 s.Name 或 s.Sorted = true
sort.Slice 配合索引访问对任意切片排序,推荐 sort.Slice —— 它只依赖索引比较,不强制元素可比较,也不复制元素。
sort.Slice(people, func(i, j int) bool { return people[i].Age 扩容陷阱:如果排序中切片发生扩容(如 append 导致 cap 不足),底层数组会换新,后续修改不影响原数组。确保排序前容量足够,或明确接受该行为。
if len(nums) > cap(nums) { nums = append(nums[:len(nums)], 0) }(预留空间)part := data[10:20],排序 part 会影响 data,但若 part 扩容,就断开了关联