初级:1-34
- 1. 左大括号 { 不能单独放一行
- 2. 未使用的变量
- 3. 未使用的 import
- 4. 简短声明的变量只能在函数内部使用
- 5. 使用简短声明来重复声明变量
- 6. 不能使用简短声明来设置字段的值
- 7. 不小心覆盖了变量
- 8. 显式类型的变量无法使用 nil 来初始化
- 9. 直接使用值为 nil 的 slice、map
- 10. map 容量
- 11. string 类型的变量值不能为 nil
- 12. Array 类型的值作为函数参数
- 13. range 遍历 slice 和 array 时混淆了返回值
- 14. slice 和 array 其实是一维数据
- 15. 访问 map 中不存在的 key
- 16. string 类型的值是常量,不可更改
- 17. string 与 byte slice 之间的转换
- 18. string 与索引操作符
- 19. 字符串并不都是 UTF8 文本
- 20. 字符串的长度
- 21. 在多行 array、slice、map 语句中缺少 , 号
- 22. log.Fatal 和 log.Panic 不只是 log
- 23. 对内建数据结构的操作并不是同步的
- 24. range 迭代 string 得到的值
- 25. range 迭代 map
- 26. switch 中的 fallthrough 语句
- 27. 自增和自减运算
- 28. 按位取反
- 29. 运算符的优先级
- 30. 不导出的 struct 字段无法被 encode
- 31. 程序退出时还有 goroutine 在执行
- 32. 向无缓冲的 channel 发送数据,只要 receiver 准备好了就会立刻返回
- 33. 向已关闭的 channel 发送数据会造成 panic
- 34. 使用了值为 nil 的 channel
- 34. 若函数 receiver 传参是传值方式,则无法修改参数的原有值
中级:35-50
- 35. 关闭 HTTP 的响应体
- 36. 关闭 HTTP 连接
- 37. 将 JSON 中的数字解码为 interface 类型
- 38. struct、array、slice 和 map 的值比较
- 39. 从 panic 中恢复
- 40. 在 range 迭代 slice、array、map 时通过更新引用来更新元素
- 41. slice 中隐藏的数据
- 42. Slice 中数据的误用
- 43. 旧 slice
- 44. 类型声明与方法
- 45. 跳出 for-switch 和 for-select 代码块
- 46. for 语句中的迭代变量与闭包函数
- 47. defer 函数的参数值
- 48. defer 函数的执行时机
- 49. 失败的类型断言
- 50. 阻塞的 gorutinue 与资源泄露
高级:50-57
文章来源于网络收集而来,版权归原创者所有,如有侵权请及时联系!
42. Slice 中数据的误用
举个简单例子,重写文件路径(存储在 slice 中)
分割路径来指向每个不同级的目录,修改第一个目录名再重组子目录名,创建新路径:
// 错误使用 slice 的拼接示例
func main() {
path := []byte("AAAA/BBBBBBBBB")
sepIndex := bytes.IndexByte(path, '/') // 4
println(sepIndex)
dir1 := path[:sepIndex]
dir2 := path[sepIndex+1:]
println("dir1: ", string(dir1)) // AAAA
println("dir2: ", string(dir2)) // BBBBBBBBB
dir1 = append(dir1, "suffix"...)
println("current path: ", string(path)) // AAAAsuffixBBBB
path = bytes.Join([][]byte{dir1, dir2}, []byte{'/'})
println("dir1: ", string(dir1)) // AAAAsuffix
println("dir2: ", string(dir2)) // uffixBBBB
println("new path: ", string(path)) // AAAAsuffix/uffixBBBB // 错误结果
}
拼接的结果不是正确的 AAAAsuffix/BBBBBBBBB
,因为 dir1、 dir2 两个 slice 引用的数据都是 path
的底层数组,第 13 行修改 dir1
同时也修改了 path
,也导致了 dir2
的修改
解决方法:
- 重新分配新的 slice 并拷贝你需要的数据
- 使用完整的 slice 表达式:
input[low:high:max]
,容量便调整为 max - low
// 使用 full slice expression
func main() {
path := []byte("AAAA/BBBBBBBBB")
sepIndex := bytes.IndexByte(path, '/') // 4
dir1 := path[:sepIndex:sepIndex] // 此时 cap(dir1) 指定为 4, 而不是先前的 16
dir2 := path[sepIndex+1:]
dir1 = append(dir1, "suffix"...)
path = bytes.Join([][]byte{dir1, dir2}, []byte{'/'})
println("dir1: ", string(dir1)) // AAAAsuffix
println("dir2: ", string(dir2)) // BBBBBBBBB
println("new path: ", string(path)) // AAAAsuffix/BBBBBBBBB
}
第 6 行中第三个参数是用来控制 dir1 的新容量,再往 dir1 中 append 超额元素时,将分配新的 buffer 来保存。而不是覆盖原来的 path 底层数组
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论