go语言最大线程 go语言线程模型( 三 )


同时,正因为如此,Go语言的interface可以用来表示任何generic的东西 , 比如一个空的interface , 可以是string可以是int , 可以是任何数据类型 , 因为这些数据类型都不需要实现任何函数 , 自然就满足空interface的定义了 。加上Go语言的type assertion , 可以提供一般动态语言才有的duck typing特性, 而仍然能在compile中捕捉明显的错误 。
5. OO
Go语言本质上不是面向对象语言,它还是过程化的 。但是,在Go语言中, 你可以很轻易的做大部分你在别的OO语言中能做的事 , 用更简单清晰的逻辑 。是的,在这里 , 不需要class , 仍然可以继承,仍然可以多态,但是速度却快得多 。因为本质上,OO在Go语言中,就是普通的struct操作 。
6. Goroutine
这个几乎算是Go语言的招牌特性之一了,我也不想多提 。如果你完全不了解Goroutine,那么你只需要知道 , 这玩意是超级轻量级的类似线程的东西,但通过它,你不需要复杂的线程操作锁操作,不需要care调度,就能玩转基本的并行程序 。在Go语言里,触发一个routine和erlang spawn一样简单 。基本上要掌握Go语言 , 以Goroutine和channel为核心的内存模型是必须要懂的 。不过请放心,真的非常简单 。
7. 更多现代的特性
和C比较,Go语言完全就是一门现代化语言,原生支持的Unicode, garbage collection, Closures(是的,和functional programming language类似), function是first class object,等等等等 。
看到这里,你可能会发现,我用了很多轻易,简单 , 快速之类的形容词来形容Go语言的特点 。我想说的是 , 一点都不夸张,连Go语言的入门学习到提高,都比别的语言门槛低太多太多 。在大部分人都有C的背景的时代,对于Go语言,从入门到能够上手做项目,最多不过半个月 。Go语言给人的感觉就是太直接了 , 什么都直接,读源代码直接,写自己的代码也直接 。
Golang 线程和协程的区别线程:
多线程是为了解决CPU利用率的问题,线程则是为了减少上下文切换时的开销,进程和线程在Linux中没有本质区别,最大的不同就是进程有自己独立的内存空间 , 而线程是共享内存空间 。
在进程切换时需要转换内存地址空间,而线程切换没有这个动作,所以线程切换比进程切换代价要小得多 。
协程:
想要简单,又要性能高,协程就可以达到我们的目的 , 它是用户视角的一种抽象,操作系统并没有这个概念,主要思想是在用户态实现调度算法 , 用少量线程完成大量任务的调度 。
Goroutine是GO语言实现的协程,其特点是在语言层面就支持,使用起来十分方便,它的核心是MPG调度模型:M即内核线程;P即处理器,用来执行Goroutine,它维护了本地可运行队列;G即Goroutine,代码和数据结构;S及调度器,维护M和P的信息 。
Go语言——goroutine并发模型参考:
Goroutine并发调度模型深度解析手撸一个协程池
Golang 的 goroutine 是如何实现的?
Golang - 调度剖析【第二部分】
OS线程初始栈为2MB 。Go语言中,每个goroutine采用动态扩容方式,初始2KB,按需增长 , 最大1G 。此外GC会收缩栈空间 。
BTW,增长扩容都是有代价的,需要copy数据到新的stack,所以初始2KB可能有些性能问题 。
更多关于stack的内容,可以参见大佬的文章 。聊一聊goroutine stack
用户线程的调度以及生命周期管理都是用户层面,Go语言自己实现的,不借助OS系统调用,减少系统资源消耗 。

推荐阅读