那些程:进程、线程、协程

导读

要理解进程与线程,首先得了解并发与并行。

并发与并行

  • 并发
    单核CPU时间分片,多个程序切换执行,这就是并发。并发是共享内存的,所以需要加锁。
    因为并发共享内存,并且会几个程序互相切换执行,所以在一个CPU执行的并发必须处理上下文切换的问题。 进程就是在这种背景下,被提出的。进程就是一些相关线程的统称,是一些相关线程的集合。进程搭配虚拟内存、进行表等,可以管理独立程序的运行、切换。
    并发也就是多线程。

由上面这段话可见,程序在运行过程中,对计算机资源的分配是个很重大的问题,于是出现了操作系统专门干这个活。操作系统核心的操作是陷入内核(Kernel),切换到操作系统,让内核来做。

  • 并行
    多核CPU几个程序同时执行,这就是并行。并行也就是我们所说的多进程。

进程和线程

  • 进程
    进程是一个容器,也就是一个程序。
    进程的切换:
  1. 切换页全局目录(Page Global Directory)来加载新的地址空间,实际上会加载新进程的cr3寄存器值。
  2. 切换内核堆栈和硬件上下文,这些包含了内核执行一个新进程的所有信息,包含了CPU寄存器。
  • 线程
    线程是容器里的工作单元。上面说并发的时候讲到,并发是共享内存的,也就说这些并发的线程会共享一个地址空间。线程的切换,不需要重新加载地址空间,页面缓冲区,需要切换寄存器上下文和栈。开销相对较小。

    线程在切换任务的时候,切换寄存器上下文和栈是抢占式的(Preeemptive multitasking),谁抢了是谁的,这就导致了线程之间的执行顺序是无法保证的,所以使用线程时需要小心操作同步问题。

    进程和线程的相同点

    线程和进程的切换,都需要陷入系统调用,即CPU先跑操作系统的调度程序,然后再由调度程序决定该炮哪一个进程(线程)

同步和异步

首先要明确一点,同步/异步这个讨论对于IO密集型才有意义。
因为对于计算密集型程序来说,
你等或者不等,
计算任务都在那里,
不多不少。
对于IO密集型程序来说,
你等就是占着茅坑不拉屎。你的肚子还没有应答,你蹲在那有什么用?反而不如腾出地方来,让程序别的地方先用着。等你确实要拉了,再来。这个坑位,就是CPU啊!

协程

协程,coroutine,也叫纤程(Fiber),或绿色线程。
协程是用户态的轻量级线程。这句话是什么鬼?

协程拥有自己的寄存器上下文和栈。协程调度切换时,将寄存器上下文和栈保存到其他地方,切回来的时候,挥发原来保存的寄存器上下文和栈。

由此可见,协程能保留上一次调用时的状态。协程的任务切换是由用户自己控制的,不存在抢占,所以这种叫做协作式多任务

协程也是单线程的,本质也是异步+回调,但是它是经过包装的,写出的代码看着是同步的代码。写过Node的应该深有体会。

Go研究出来了一套协程的调度算法,所以Go的协程叫做Goroutine。
Kotlin也有,C#也有,不是什么新鲜玩意。但是在语言的层面上的支持,还是第一个。

事件驱动

事件驱动是事先编写一个事件循环,这个事件循环程序不断地检查目前要处理的信息,多用在GUI框架、页面上的JS事件。如果这个事件收到了要处理的事件的信号,就异步去执行。

然后这个信号是推拉结合的。

基于事件驱动的编程是单线程思维,特点是异步+回调。

协程的优劣

协程的优点

  • 跨平台/跨体系架构
  • 与进程/线程相比,上下文切换的开销
  • 与线程相比,不需要原子操作锁定及同步
  • 与事件驱动相比,方便切换控制流,编程模型简单
  • 高并发,一个CPU支持上万的协程

协程的缺点

  • 无法利用多核资源,事件驱动,本质是个单线程的循环而已。需要配合进程来执行。

演进轨迹

IO密集型程序:多进程 -> 多线程 -> 事件驱动 -> 协程
CPU密集型程序:多进程 -> 多线程

  1. 为什么从多进程到多线程是一种进步?
    因为线程比进行性能开销小,而且多线程之间数据通信与同步更加方便。想同步的时候就访问,修改的时候加把锁就可以了。

    这个也是符合我们直觉的。比如要需要一个程序某个部分的几个不同运算值,是在后台多起几个进程分别跑呢?还是写成多线程处理呢?实践中,我们很少见到开多进程的吧?

  2. 那为什么CPU密集型任务没有发展到事件驱动、协程呢?
    因为CPU密集型的任务,一般都是需要计算的,计算是需要CPU的。CPU的多少是和系统的硬件相关,异步是改变不了这一点的,等待的时候不能计算,等待完了该算的还得接着算,所以说没什么卵用。

多进程(并行)/多线程(并发)/协程各自的适用场景

为什么程序多了电脑卡

进程多了,操作系统会频繁切换进程。从上面可知,进程一切换,页全局目录、内核堆栈、硬件上下文都会变。一旦进程变多之后,频繁切换会挤占大部分资源。

相关知识补充

image.png

啥是“陷入内核”?看到这个词的第一反应是写错了。其实不然。
要说请首先需明确,特权级的概念。就行linux系统里面,喜欢划分管理员和用户等不同权限角色一样。在系统中,为了让资源调配集中管理,便有了特权级这个概念。特权级一共有0-3这四个级别,0最高,可以管理CPU。
linux系统里面只有两个级别,0级和3级。操作系统的内核当然是最高级0级,应用程序是3级。当应用程序需要访问系统资源时,CPU就进入了内核,从图上看好像陷进入一样。

所谓的上下文,是指程序在运行过程中的一些中间状态,比如会运算出来一些值,这些值在后面的计算中也会用到。当然这些值必须保存在内存中。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,186评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,858评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,620评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,888评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,009评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,149评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,204评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,956评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,385评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,698评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,863评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,544评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,185评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,899评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,141评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,684评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,750评论 2 351