首页
技术小册
AIGC
面试刷题
技术文章
MAGENTO
云计算
视频课程
源码下载
PDF书籍
「涨薪秘籍」
登录
注册
Go 中的 = 和 := 有什么区别?
Go 中的指针的意义是什么?
Go 多值返回有什么用?
Go 有异常类型吗?
Go 中的 rune 和 byte 有什么区别?
Go 语言中的深拷贝和浅拷贝?
什么叫字面量和组合字面量?
对象选择器自动解引用怎么用?
map 的值不可寻址,那如何修改值的属性?
有类型常量和无类型常量的区别?
为什么传参使用切片而不使用数组?
Go 语言中 hot path 有什么用呢?
引用类型与指针,有什么不同?
Go 是值传递,还是引用传递、指针传递?
Go中哪些是可寻址,哪些是不可寻址的?
slice 扩容后容量及内存如何计算?
goroutine 存在的意义是什么?
说说 Go 中闭包的底层原理?
defer 的变量快照什么情况会失效?
说说你对 Go 里的抢占式调度的理解。
简述一下 Go 栈空间的扩容/缩容过程?
说一下 GMP 模型的原理。
GMP 模型为什么要有 P ?
不分配内存的指针类型能用吗?
如何让在强制转换类型时不发生内存拷贝?
Go 中的 GC 演变是怎样的?
Go 中哪些动作会触发 runtime 调度?
有哪些情况会导致协程泄露?
局部变量分配在栈上还是堆上?
为什么常量、字符串和字典不可寻址?
为什么 slice 元素是可寻址的?
Go 的默认栈大小是多少?最大值多少?
Go 中的分段栈和连续栈的区别?
内存对齐、内存布局是怎么回事?
Go 里是怎么比较相等与否?
所有的 T 类型都有 *T 类型吗?
数组对比切片有哪些优势?
GMP 偷取 G 为什么不需要加锁?
堆引用栈内存是怎么回收的?
当前位置:
首页>>
技术小册>>
GO面试指南
小册名称:GO面试指南
在 Go 语言中, GMP 模型中的 P(processor) 管理了若干个 Goroutine ,而这些 Goroutine 最终都运行在 M(machine)上。在 GMP 模型中,当一个 P 管理的 Goroutine 阻塞时,为了让其他 Goroutine 继续运行,这个 P 就会从其他 P 管理的 Goroutine 中偷取一些 Goroutine,然后放到自己的队列中运行。 在这个偷取 Goroutine 的过程中,由于 Goroutine 的状态是通过轻量级的原子操作来改变的,而不是通过加锁来保护的,所以偷取 Goroutine 不需要加锁。 下面是一个简单的示例代码,演示了当一个 Goroutine 阻塞时,偷取 Goroutine 的过程: ``` package main import ( "fmt" "runtime" ) func main() { runtime.GOMAXPROCS(2) // 设置 P 的数量为 2 ch := make(chan int, 2) go func() { for i := 0; i < 10; i++ { ch <- i fmt.Println("send: ", i) } }() go func() { for { select { case i := <-ch: fmt.Println("recv: ", i) } } }() select {} } ``` 在这个示例中,我们启动了两个 Goroutine ,其中一个往一个带缓冲的通道中发送数值,另一个从通道中接收数值并打印。由于通道中有缓冲,所以发送操作不会阻塞,但是接收操作会一直阻塞,直到通道中有数据。 在这个过程中,由于只有一个 P ,所以一旦接收 Goroutine 阻塞,整个程序就会阻塞。为了解决这个问题,Go 运行时系统会自动将这个阻塞的 Goroutine 从当前 P 的队列中偷取,并放到另一个 P 的队列中运行,这样就可以让程序继续运行。在这个过程中,并没有加锁的操作,而是使用了轻量级的原子操作来实现。
上一篇:
数组对比切片有哪些优势?
下一篇:
堆引用栈内存是怎么回收的?
该分类下的相关小册推荐:
企业级Go应用开发从零开始
Go开发权威指南(上)
深入浅出Go语言核心编程(二)
深入解析go语言
WebRTC音视频开发实战
深入浅出Go语言核心编程(五)
go编程权威指南(三)
go编程权威指南(二)
Go开发基础入门
深入浅出Go语言核心编程(七)
Go开发权威指南(下)
深入浅出Go语言核心编程(一)