Go 语言从入门到实战
蔡超
Mobvista 技术副总裁兼首席架构师,前亚马逊(中国)首席软件架构师
48919 人已学习
新⼈⾸单¥59
课程目录
已完结/共 55 讲
第一章:Go语言简介 (4讲)
第二章:基本程序结构 (4讲)
第三章:常用集合 (3讲)
第四章:字符串 (1讲)
时长 16:47
第五章:函数 (2讲)
第六章:面向对象编程 (4讲)
第七章:编写好的错误处理 (2讲)
第八章:包和依赖管理 (2讲)
第九章:并发编程 (7讲)
第十章:典型并发任务 (5讲)
第十一章:测试 (3讲)
时长 11:48
时长 07:12
时长 06:15
第十二章:反射和Unsafe (3讲)
时长 08:18
时长 08:03
第十三章:常见架构模式的实现 (2讲)
第十四章:常见任务 (4讲)
时长 04:27
时长 05:14
第十五章:性能调优 (4讲)
第十六章:高可用性服务设计 (5讲)
Go 语言从入门到实战
登录|注册
留言
59
收藏
沉浸
阅读
分享
手机端
回顶部
当前播放: 09 | 数组和切片
00:00 / 00:00
高清
  • 高清
1.0x
  • 2.0x
  • 1.5x
  • 1.25x
  • 1.0x
  • 0.75x
  • 0.5x
网页全屏
全屏
00:00
付费课程,可试看
01 | Go语言课程介绍
02 | 内容综述
03 | Go语言简介:历史背景、发展现状及语言特性
04 | 编写第一个Go程序
05 | 变量、常量以及与其他语言的差异
06 | 数据类型
07 | 运算符
08 | 条件和循环
09 | 数组和切片
10 | Map声明、元素访问及遍历
11 | Map与工厂模式,在Go语言中实现Set
12 | 字符串
13 | Go语言的函数
14 | 可变参数和defer
15 | 行为的定义和实现
16 | Go语言的相关接口
17 | 扩展与复用
18 | 不一样的接口类型,一样的多态
19 | 编写好的错误处理
20 | panic和recover
21 | 构建可复用的模块(包)
22 | 依赖管理
23 | 协程机制
24 | 共享内存并发机制
25 | CSP并发机制
26 | 多路选择和超时
27 | channel的关闭和广播
28 | 任务的取消
29 | Context与任务取消
30 | 只运行一次
31 | 仅需任意任务完成
32 | 所有任务完成
33 | 对象池
34 | sync.pool对象缓存
35 | 单元测试
36 | Benchmark
37 | BDD
38 | 反射编程
39 | 万能程序
40 | 不安全编程
41 | 实现pipe-filter framework
42 | 实现micro-kernel framework
43 | 内置JSON解析
44 | easyjson
45 | HTTP服务
46 | 构建RESTful服务
47 | 性能分析工具
48 | 性能调优示例
49 | 别让性能被锁住
50 | GC友好的代码
51 | 高效字符串连接
52 | 面向错误的设计
53 | 面向恢复的设计
54 | Chaos Engineering
55 | 结课测试&结束语
登录 后留言

全部留言(59)

  • 最新
  • 精选
忽然之间
收获:老师演示的slice cap容量增长每次翻倍,下意识觉得这个内存成本太高,应该会有更优化的逻辑,至少有个增长因子的规则。为了求证,代码如下 func TestSlice(t *testing.T) { s3 := make([]int, 1) for n < 1500 { t.Log(n, len(s3), cap(s3)) //1 2 4 8 16 32 .. 512 1024 1280 1696 s3 = append(s3, 1) n++ } } 容量增长情况为:1 2 4 8 16 32..512 1024 1280 1696 可以看出slice的cap规则为: len <=1204,cap*2, len>1024 , cap = cap * ( 1+1/4 ) 那就看看源码,找了很久,才找到slice 结构体的源码位置:$GOROOT/src/runtime/slice.go 其中growslice函数有这么一段: newcap := old.cap doublecap := newcap + newcap if cap > doublecap { newcap = cap } else { if old.len < 1024 { newcap = doublecap } else { // Check 0 < newcap to detect overflow // and prevent an infinite loop. for 0 < newcap && newcap < cap { newcap += newcap / 4 } // Set newcap to the requested cap when // the newcap calculation overflowed. if newcap <= 0 { newcap = cap } } } 以上,请老师指点。

作者回复: 非常喜欢你深入学习的精神! * 首先判断,如果新申请容量(cap)大于2倍的旧容量(old.cap),最终容量(newcap)就是新申请的容量(cap) * 否则判断,如果旧切片的长度小于1024,则最终容量(newcap)就是旧容量(old.cap)的两倍,即(newcap=doublecap) * 否则判断,如果旧切片长度大于等于1024,则最终容量(newcap)从旧容量(old.cap)开始循环增加原来的 1/4,即(newcap=old.cap,for {newcap += newcap/4})直到最终容量(newcap)大于等于新申请的容量(cap),即(newcap >= cap) * 如果最终容量(cap)计算值溢出,则最终容量(cap)就是新申请容量(cap)

2019-03-17
82
Jason
老师,经过我自己测试,slice的共享内存存在一个隐患,就是使用共享内存的双方slice如果有一方出现扩容,则扩容一方将不再指向原有空间。所以程序中想以修改一方达到数据同步时需要小心

作者回复: 很好的观察,因为扩容以后,会分配一个更大的空间,则使原有的指针指向了不同的地方。

2019-03-09
2
29
Vincent
老师说的:“切片变量可以和nil比较,看看该变量是否初始化了“ 归纳得很准确,做实验: func TestSliceCompare(t *testing.T){ var a []int var b = make([]int,0,0) c := []int{} t.Log(a, len(a), cap(a)) t.Log(b, len(b), cap(b)) t.Log(c, len(c), cap(c)) t.Log(a==nil, b==nil, c==nil) } 运行结果: --- PASS: TestSliceCompare (0.00s) slice_test.go:45: [] 0 0 slice_test.go:46: [] 0 0 slice_test.go:47: [] 0 0 slice_test.go:48: true false false 其中,只有声明方式var a []int得到的切片==nil

作者回复: 你细致的学习和通过实践的学习方式太赞了

2019-03-11
15
Aliliin
有个疑问?go不支持 array 或者 slice 里面有不同的类型值吗? ``` [ 1, "nam", "age", ] ```

作者回复: 1. 声明数组或者切片时是要指定类型的 2. 一种变通的方式是通过interface{}类型 S5:= []interface{}{“hello”,1} 可以参加接下来课程里的“空接口和断言”相关部分

2019-03-05
3
10
Vincent
实验中发现,多维数组如果这样声明赋值: marr := [...][...]int {{1,2,3},{2,3,4}} 会报错:use of [...] array outside of array literal 只能这样:marr := [...][3]int {{1,2,3},{2,3,4}} 好像只有第一维的声明能简写成[...],其他维必须写明大小。 在三维数组实验了下,也是这样。想请问老师,是这样吗? 另外,实验了下,marr := [...][]{{1,2,3},{2,3,4}}可以,不过第二维是Slice。

作者回复: 我在使用过程中的观察和你一致。非常感谢你的反馈。

2019-03-10
2
9
flykyle
老师,我用的也是atom,为什么我保存之后执行test的结果是: PASS coverage: 0.0% of statements ok ch2/test 0.012s Done 而并没有打印出t.Log的内容

作者回复: 按以下检查你的go-plus相关的test设置: Atom->preferences->packages 搜索找到go-plus 在settings中,Test配置中选中 Run with verbose flag setting

2019-03-06
6
碧雪天虹
负数的话, 前面加个len(arr)就行了, 目前语法糖上还不支持, 但本质上也就那样. func TestArraySection(t *testing.T) { arr3 := [...]int{1, 2, 3, 4, 5} // 最前面3个元素 arr3_sec := arr3[:3] t.Log(arr3_sec) // 最后面2个元素 arr3_sec2 := arr3[len(arr3)-2:] t.Log(arr3_sec2) // 倒数第1个元素 t.Log(arr3[len(arr3)-1]) }

作者回复: 赞

2019-10-16
5
gamtin
听您的讲解,切片是非常灵活且比较节省内存空间的。 请问,在实际应用过程中,什么时候应该选择用数组,什么时候选择用切片呢?是否可以理解:大部分情况下,都应该优先选择用切片呢?

作者回复: 长度已知的情况可以用数组。 切片的方便之处主要来自于自动增长,但切片的自动增长会导致内存分配和数据复制,以及未来相关的GC开销。

2019-03-06
2
5
fatty Jack
老师以下两种数组的初始化方式有本质不同,各自有使用场景? arr1 := [4]int{1,2,3,4} arr2 := [...]int{1,2,3,4}

作者回复: 主要差别是第二种初始化方式,不用你自己算元素个数,在元素较多,或者你修改程序增加元素时更为简便

2019-03-05
2
5
___
a := [...]int{1, 2, 3, 4, 5} b := []int{1, 2, 3, 4, 5} 请问这2种有什么区别?

作者回复: 第一个是数组,第二个是切片。 数组和切片的差别请参考课程相关内容

2019-03-20
2
4
收起评论