goroutine传参panic,原因在源码中找到了

简介: goroutine传参panic,原因在源码中找到了

goroutine传参 有点可怕



1.背景


我昨天写一个需求,老版本继续跑它的逻辑,新版本我Go出去,看看新老版本对比结果,之所以Go出去是因为担心影响线上服务,比如不用Go出去,如果代码异常或者阻塞,就会影响老版本逻辑的执行,因此这里采用Goroutine的方式去处理。


2.代码


640.png


3.上线跑起来

640.png


goroutine报错了。


4.上网解决


上网搜索了之后才明白:


新开goroutine的可用堆栈空间默认分配2k的内存
故传入的参数不宜过大,否则导致程序panic


5. 解决办法


  1. 改为指针类型传递
  2. 减少传递参数的大小,只传入需要使用的参数
  3. 去除协程


我采用的就是第一种,改为指针类型传递。


6. 源码


// The minimum size of stack used by Go code
 _StackMin = 2048
//go:systemstack
func newproc1(fn *funcval, argp unsafe.Pointer, narg int32, callergp *g, callerpc uintptr) *g {
 _g_ := getg()
 if fn == nil {
  _g_.m.throwing = -1 // do not dump full stacks
  throw("go of nil func value")
 }
 acquirem() // disable preemption because it can be holding p in a local var
 siz := narg
 siz = (siz + 7) &^ 7
 // We could allocate a larger initial stack if necessary.
 // Not worth it: this is almost always an error.
 // 4*sizeof(uintreg): extra space added below
 // sizeof(uintreg): caller's LR (arm) or return address (x86, in gostartcall).
 // 这里有判断 大于2048即2K就panic
 if siz >= _StackMin-4*sys.RegSize-sys.RegSize {
  throw("newproc: function arguments too large for new goroutine")
 }
  ...
}


7. 小结


用goroutine一定要小心,参数不易过大,否则上线之后头就大了。另外就是看源码,写的很明白,避坑指南要收藏好!

相关文章
|
8月前
|
编译器 Go
Golang底层原理剖析之函数调用栈-传参和返回值
Golang底层原理剖析之函数调用栈-传参和返回值
72 0
|
Go C++ 索引
太狠了!原来panic可以直接干掉程序
太狠了!原来panic可以直接干掉程序
136 0
|
8月前
|
存储 编译器 Go
Golang底层原理剖析之函数调用栈-栈帧布局与函数跳转
Golang底层原理剖析之函数调用栈-栈帧布局与函数跳转
130 0
|
8月前
|
Java Go
浅谈defer、panic、recover 三者的用法
浅谈defer、panic、recover 三者的用法
49 0
Mgo
|
Shell Go
go调用shell命令两种方式实现(有无返回值)
go调用shell命令两种方式实现(有无返回值)
Mgo
1632 1
普通函数中的this指向问题解决方案call
普通函数中的this指向问题解决方案call
61 0
|
关系型数据库 MySQL Go
[Golang]存在调用同一个包内某个结构体的方法出错?可能是未初始化结构体
[Golang]存在调用同一个包内某个结构体的方法出错?可能是未初始化结构体
|
Unix Linux
C 程序来演示 fork() 和 pipe()
fork() 用于创建子进程。此子进程是原始(父)进程的副本。它是在类Unix操作系统上创建进程的主要方法。
99 0