golang 数组和切片

区别

1.数组长度固定,切片长度可变

2.数组是深拷贝,切片是浅拷贝,切片是引用类型

扩容规则 不同版本不一样

https://www.jb51.net/article/280481.htm#_lab2_2_1

go1.18

1.如果期望容量大于当前容量的两倍就会使用期望容量;

2.如果当前切片的容量小于阈值(默认 256)就会将容量翻倍;

3.如果当前切片的容量大于等于阈值(默认 256),小切片的2倍生长,大切片生长1.25倍;

go 复制代码
package main

import "fmt"

func main() {

	arr1 := [3]int{1, 2, 3}
	arr2 := arr1
	fmt.Printf("%p\n", &arr1[0])
	fmt.Printf("%p\n", &arr2[0])
	arr2[0] = 10
	fmt.Println(arr1)
	fmt.Println(arr2)

	sli1 := []int{1, 2}
	sli2 := sli1
	fmt.Printf("%p\n", &sli1[0])
	fmt.Printf("%p\n", &sli2[0])
	sli2[0] = 10
	fmt.Println(sli1)
	fmt.Println(sli2)

	sliceAppend(sli1)
	fmt.Println(sli1)
}

// 不会改变原有slice长度
func sliceAppend(s []int) {
	s = append(s, []int{2, 2, 2}...)
	fmt.Printf("sliceAppend %d %d\n", len(s), len(s))
}
go 复制代码
package main

import "fmt"

func main() {
	sli1 := []int{1}
	fmt.Printf("len=%d cap=%d\n", len(sli1), cap(sli1))

	// 期望容量大于当前容量的两倍
	sli1 = append(sli1, []int{2, 3, 4}...)
	fmt.Printf("len=%d cap=%d\n", len(sli1), cap(sli1))

	// 如果当前切片的容量小于阈值(默认 256)就会将容量翻倍
	sli1 = append(sli1, 1)
	fmt.Printf("len=%d cap=%d\n", len(sli1), cap(sli1))

	// 如果当前切片的容量大于等于阈值(默认 256),就会每次增加 25% 的容量
	/*
		//从小切片的2倍生长过渡
		//大切片生长1.25倍
		newcap += (newcap + 3*threshold) / 4
	*/
	sli2 := make([]int, 256, 256)
	fmt.Printf("len=%d cap=%d\n", len(sli2), cap(sli2))
	sli2 = append(sli2, 1)
	fmt.Printf("len=%d cap=%d\n", len(sli2), cap(sli2))

	for i := 0; i < 10; i++ {
		sli2 = append(sli2, sli2...)
		fmt.Printf("len=%d cap=%d\n", len(sli2), cap(sli2))
	}

}

runtime/slice

go 复制代码
type slice struct {
	array unsafe.Pointer
	len   int
	cap   int
}

func growslice(et *_type, old slice, cap int) slice {
	if raceenabled {
		callerpc := getcallerpc()
		racereadrangepc(old.array, uintptr(old.len*int(et.size)), callerpc, abi.FuncPCABIInternal(growslice))
	}
	if msanenabled {
		msanread(old.array, uintptr(old.len*int(et.size)))
	}
	if asanenabled {
		asanread(old.array, uintptr(old.len*int(et.size)))
	}

	if cap < old.cap {
		panic(errorString("growslice: cap out of range"))
	}

	if et.size == 0 {
		// append should not create a slice with nil pointer but non-zero len.
		// We assume that append doesn't need to preserve old.array in this case.
		return slice{unsafe.Pointer(&zerobase), old.len, cap}
	}

	newcap := old.cap
	doublecap := newcap + newcap
	if cap > doublecap {
		newcap = cap
	} else {
		const threshold = 256
		if old.cap < threshold {
			newcap = doublecap
		} else {
			// Check 0 < newcap to detect overflow
			// and prevent an infinite loop.
			for 0 < newcap && newcap < cap {
				// Transition from growing 2x for small slices
				// to growing 1.25x for large slices. This formula
				// gives a smooth-ish transition between the two.
				newcap += (newcap + 3*threshold) / 4
			}
			// Set newcap to the requested cap when
			// the newcap calculation overflowed.
			if newcap <= 0 {
				newcap = cap
			}
		}
	}
相关推荐
冒泡的肥皂11 分钟前
MVCC初学demo(一
数据库·后端·mysql
啊阿狸不会拉杆31 分钟前
《算法导论》第 32 章 - 字符串匹配
开发语言·c++·算法
颜如玉1 小时前
ElasticSearch关键参数备忘
后端·elasticsearch·搜索引擎
卡拉叽里呱啦2 小时前
缓存-变更事件捕捉、更新策略、本地缓存和热key问题
分布式·后端·缓存
David爱编程2 小时前
线程调度策略详解:时间片轮转 vs 优先级机制,面试常考!
java·后端
武当豆豆2 小时前
C++编程学习(第25天)
开发语言·c++·学习
码事漫谈3 小时前
C++继承中的虚函数机制:从单继承到多继承的深度解析
后端
阿冲Runner3 小时前
创建一个生产可用的线程池
java·后端
写bug写bug3 小时前
你真的会用枚举吗
java·后端·设计模式
喵手4 小时前
如何利用Java的Stream API提高代码的简洁度和效率?
java·后端·java ee