golang、c++,并发、并行(一)

 

 

正式接触golang大半个月,写了几个练习项目,让人有些感概。

 

 

Golang这样的自动化内存管理水平和并发调度能力,让我印象很深。

 

单是那样的内存管理水平,通过加入特定的并发优化c++内存池,我可以做到。

那种并发调度能力,通过引入复杂的调度算法,也勉强可以做到。

 

但是这两个加在一起,配合出色的跨平台标准库,组成这样的一个并发友好跨平台生态,这我五年之内可能都做不到。

 

没有用c++写过高并发代码的人,可能很难理解这种复杂的情绪。

这是一种,久渴的人,突然在沙漠中看到一片绿洲的感觉,惊喜万分;而后又发现原来这个绿洲一直就在你身旁。但你却不停错过的悔恨。

 

c++的高并发编程,一个听的人感觉很有逼格,写的人累死的事情。

这是在11版前,甚至连跨平台线程模型都没有的环境。

这是连标准库一个小小的random生成器都没有实现并发安全性的环境。

这是一个需要面对海量第三方库,然后基本上没有文档,和一堆编译问题和运行bug的环境。海量到,最后已经完全麻木,只感觉眼前不停地飞过一个又一个库名,一个又一个错误。等你费了吃奶的劲,终于都正常运行了。才又发现,只是串行情况下运行正常。那并发运行呢?

这是一个在沙子上建立的楼房。或许一声咳嗽后,就会顷刻倒塌。

 

就是这么一片糟糕的环境。扑面而来的蛮荒与原始,让你怀疑还活在上个世纪,活在穿孔卡片式计算机的时代。

 

 

06年,计算机正式进入多核元年。

以前只在大型机、小型机上,使用的昂贵玩意儿,开始大规模出现在桌面pc和廉价服务器上。

 

一个让硬件工程师,大松一口气,“终于他妈的,从功耗、主频陷阱解脱了。”他们成功的甩掉了一个沉重的包袱。

而软件工程师,开始戴上了沉重的枷锁。心里一万头。。。奔过。“免费午餐”正式结束。

 

 

并发、并行两个坑爹货正式进入大多数程序员的视野。

 

 

线程级的并发、并行

线程级并发,是一种使用多线程进行程序流程设计的方式。(和单核的指令级并行不同)

在单核时代,主要是应用于gui,网络通信等要求低延迟高响应的场景。

线程级并行,是一部分并发在多核cpu上的一种表现。

 

。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。

简单举例:

 

程序a,包含100斤肉、50斤淀粉、10斤香辛料。

单核cpu,一座火腿肠工厂。

 

串行:

 

100斤肉-->50斤淀粉-->10斤香辛料-->火腿肠工厂-->火腿肠

把肉、淀粉、香辛料运入工厂,一分钟(60秒)后,工厂生产出160斤火腿肠。

 

现在客户说一分钟太久了,要一边运进原料一边生产出火腿肠。降低出货时间。

并发:

 

10斤肉-->5斤淀粉-->1斤香辛料-->火腿肠工厂-->火腿肠

(然后重复)

10斤肉-->5斤淀粉-->1斤香辛料-->火腿肠工厂-->火腿肠

。。。

 

现在可以做到0.1分钟(6秒钟)就生产出16斤火腿肠。

 

 

现在老板说火腿肠生产的太慢,要扩大工厂规模。于是找来了工程师,工程师说,继续扩大单个工厂成本太高,划不来。不如,再建一座新工厂。

 

程序a,包含100斤肉、50斤淀粉、10斤香辛料。

双核cpu,两座火腿肠工厂,工厂a、工厂b。

 

继续用上面并发生产方式:

(两座工厂同时生产)

 

 

10斤肉-->5斤淀粉-->1斤香辛料-->火腿肠工厂a-->火腿肠

 

10斤肉-->5斤淀粉-->1斤香辛料-->火腿肠工厂b-->火腿肠

 

 

现在0.1分钟(6秒钟)运入20斤肉、10斤淀粉、2斤香辛料,

(10*2斤肉、5*2斤淀粉、1*2斤香辛料)

生产出32(16*2)斤火腿肠。

 

这就是并行。

。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。

 

下一篇结合语言继续扯。。。