这里列举的Go语言常见坑都是符合Golang语法的,可以正常的编译,但是可能是运行结果错误,或者是有资源泄漏的风险。

可变参数是空接口类型

当参数的可变参数是空接口类型时,传人空接口的切片时需要注意参数展开的问题。

func main() {

var a = []interface{}{1, 2, 3}

fmt.Println(a)

fmt.Println(a...)

}

// 不管是否展开,编译器都无法发现错误,但是输出是不同的:

// [1 2 3]

// 1 2 3

数组是值传递

在函数调用参数中,数组是值传递,无法通过修改数组类型的参数返回结果,必要时需要使用切片。

func main() {

x := [3]int{1, 2, 3}

func(arr [3]int) {

arr[0] = 7

fmt.Println(arr)

}(x)

fmt.Println(x)

}

// [7 2 3]

// [1 2 3]

map遍历顺序是不固定的

map是一种hash表实现,每次遍历的顺序都可能不一样。Golang会提前取一个随机数,把桶的遍历顺序随机化。

func main() {

m := map[string]string{

"1": "1",

"2": "2",

"3": "3",

}

for k, v := range m {

println(k, v)

}

}

recover必须在defer函数中运行

recover捕获的是祖父级调用时的异常,直接调用无效;

直接defer调用无效

func main() {

defer recover()

panic(1)

}

defer调用时多层嵌套依然无效

func main() {

defer func() {

func() { recover() }()

}()

panic(1)

}

必须在defer函数中直接调用才有效

func main() {

defer func() {

recover()

}()

panic(1)

}

main函数提前退出

导致后台Goroutine无法保证完成任务。

休眠:

time.Sleep(time.Second)

插入调度语句:

runtime.Gosched() //用于让出当前CPU时间片

利用for:

for {}

利用for+调度语句:

for {

runtime.Gosched()

}

利用阻塞:

select {}

闭包错误引用同一个变量

func main() {

for i := 0; i < 5; i++ {

defer func() {

println(i)

}()

}

}

/*

5

5

5

5

5

*/

改进的方法是在每轮迭代中生成一个局部变量

func main() {

for i := 0; i < 5; i++ {

i := i

defer func() {

println(i)

}()

}

}

/*

4

3

2

1

0

*/

或者是通过函数参数传入:

func main() {

for i := 0; i < 5; i++ {

defer func(i int) {

println(i)

}(i)

}

}

/*

4

3

2

1

0

*/

在循环内部执行defer语句

defer在函数退出时才能执行,在for执行defer会导致资源延迟释放:

func main() {

for i := 0; i < 5; i++ {

f, err := os.Open("/path/to/file")

if err != nil {

log.Fatal(err)

}

defer f.Close()

}

}

解决的方法可以在for中构造一个局部函数,在局部函数内部执行defer:

func main() {

for i := 0; i < 5; i++ {

func() {

f, err := os.Open("/path/to/file")

if err != nil {

log.Fatal(err)

}

defer f.Close()

}()

}

}

切片会导致整个底层数组被锁定

切片会导致整个底层数组被锁定,底层数组无法释放内存。如果底层数组较大会对内存产生很大的压力。

headerMap[name] = data[:1]

解决的方法是将结果克隆一份,这样可以释放底层的数组:

eaderMap[name] = append([]byte{}, data[:1]...)

空指针和空接口不等价

返回了一个错误指针,但是并不是空的error接口:

内存地址会变化

Go语言中对象的地址可能发生变化,因此指针不能从其它非指针类型的值生成:

func main() {

var x int = 42

var p uintptr = uintptr(unsafe.Pointer(&x))

runtime.GC()

var px *int = (*int)(unsafe.Pointer(p))

println(*px)

}

当内存发送变化的时候,相关的指针会同步更新,但是非指针类型的uintptr不会做同步更新。

同理CGO中也不能保存Go对象地址。

Goroutine泄露

Go语言是带内存自动回收的特性,因此内存一般不会泄漏。但是Goroutine确存在泄漏的情况,同时泄漏的Goroutine引用的内存同样无法被回收。

func main() {

ch := func()

ch := make(chan int)

go func() {

for i := 0; ; i++ {

ch

}

} ()

return ch

}()

for v := range ch {

fmt.Println(v)

if v == 5 {

break

}

}

}

上面的程序中后台Goroutine向管道输入自然数序列,main函数中输出序列。但是当break跳出for循环的时候,后台Goroutine就处于无法被回收的状态了。

我们可以通过context包来避免这个问题:

func main() {

ctx, cancel := context.WithCancel(context.Background())

ch := func(ctx context.Context)

ch := make(chan int)

go func() {

for i := 0; ; i++ {

select {

case

return

case ch

}

}

} ()

return ch

}(ctx)

for v := range ch {

fmt.Println(v)

if v == 5 {

cancel()

break

}

}

}

当main函数在break跳出循环时,通过调用cancel()来通知后台Goroutine退出,这样就避免了Goroutine的泄漏。

golang 所有进程休眠_Golang基础--常见坑相关推荐

  1. golang 所有进程休眠_golang 垃圾回收(三)插入写屏障

    并发的垃圾回收 STW 安全的回收 并发的垃圾回收 插入写屏障 伪代码 对象丢失的必要条件 写屏障是怎么解决问题? 并发的垃圾回收 golang 语言设计的根本性追求就是高并发,低延迟,所以golan ...

  2. golang 所有进程休眠_Golang-进程无故消失?

    导致进程消失原因:64位操作系统中,写一个溢出指针到内存的非指针字符(a non-pointer word of memory),导致GC的时,不能看到该溢出块,所以导致GC过早回收该块,然后就导致崩 ...

  3. c++时间片轮转rr进程调度算法_进程,线程基础(—)

    进程 进程简单的定义是指装载到内存的指令集并且正在由cpu执行其中的每一条指令的这个程序叫做进程. 进程控制块 process control block 简称PCB,主要包含了标识符,状态,优先级, ...

  4. 进程和线程基础知识(已经是最详细的啦)

    进程和线程基础知识 文章目录 进程和线程基础知识 一.前言 二.进程 1.引入 2.并发和并行有什么区别? 3.进程与程序的关系的类比 4.进程的状态 5.进程的控制结构 6.进程的控制 7.进程的上 ...

  5. golang中container/list包中的坑

    转载地址:golang中container/list包中的坑 - Go语言中文网 - Golang中文社区 golang中list包用法可以参看golang中container/list包用法_che ...

  6. Linux进程管理:进程和线程基础知识

    <Linux进程管理:进程和线程基础知识> <Linux-进程管理> <C语言进程的内存地址空间分配> <进程和线程模型> <(1)Linux进程 ...

  7. 快速get进程、线程基础知识

    前言 为了方便大家理解进线程的工作流程及运行状态,先来看一篇小故事: 我们写好的一行行代码,为了让其工作起来,我们还得把它送进城(进程)里,那既然进了城里,那肯定不能胡作非为了. 但是城里有城里的规矩 ...

  8. Linux进程休眠和唤醒

    当进程以阻塞的方式通信,在得到结果前进程会挂起休眠. 为了将进程以一种安全的方式进入休眠,我们需要牢记两条规则: 一.永远不要在原子上下文中进入休眠. 二.进程休眠后,对环境一无所知.唤醒后,必须再次 ...

  9. SpringBoot 快速开启事务(附常见坑点)

    SpringBoot 快速开启事务(附常见坑点) 序言:此前,我们主要通过XML配置Spring来托管事务.在SpringBoot则非常简单,只需在业务层添加事务注解(@Transactional ) ...

最新文章

  1. 在GridView内访问特定控件
  2. 安装SQL Server 2012示例数据库
  3. dba mysql命令_Mysql常用DBA命令
  4. 数据库管理专才四步走
  5. win10如何下载适合自己python版本的pygame?
  6. 关于压缩jar包时提示*.*没有这个文件或目录的问题以及解决办法:
  7. ironpython console是否可以卸载_IronPython的第十块鳞片
  8. android杀掉进程顺序
  9. SAP CRM Fiori BP overview page design
  10. SDP协议基本分析(RTSP、WebRTC使用)
  11. C#数据类型02--结构
  12. java的Junit单元测试
  13. 软件工程结对项目:四则运算web
  14. Sofia-SIP辅助文档六 - Sofia-SIP中的SIP和SDP特性
  15. 云承软件和云溪哪个好_多工序开料机的软件对接及注意事项
  16. 创新设计思维---自学报告
  17. 实验|三层交换机配置实例
  18. 医学人工智能读书会与黄智生教授简历(公号回复“医学AI读书会”下载PDF资料,欢迎转发、赞赏、支持科普)
  19. Matlab 之meshgrid interp griddata interp2
  20. 介绍dbt,ETL和ELT Disrupter

热门文章

  1. 集合论——二元关系的定义组成及性质
  2. GB28181流媒体服务LiveGBS启动报错 HTTP Port[10000] In Use
  3. 关于vue中watch和computed
  4. 面试问题记录 2019.3.22(中国铁道科学研究院)
  5. sklearn 数据预处理1: StandardScaler
  6. [机器学习]-[数据预处理]-中心化 缩放 KNN(二)
  7. Atitit 纯java项目的提升进度大法---通用json dsl接口
  8. 网络编程——第一部分
  9. oralcle中的性能统计基本用法statpack
  10. 大数据之-Hadoop3.x_了解一下hive和hbase的区别---大数据之hadoop3.x工作笔记0084