我對(duì)Erlang和Gevent的感受

當(dāng)然它們是很不一樣的東西妥泉。

上一篇文章中我也表述對(duì)這種的喜愛掐禁。

Erlang

Erlang大概是2012年接觸的。然后就被震撼了:

  • 簡(jiǎn)單的語(yǔ)法
  • 隨意的spawn erlang process
  • 基于消息傳遞數(shù)據(jù)
  • 模式匹配
  • 遞歸
  • 優(yōu)雅的二進(jìn)制處理
  • 強(qiáng)大的OTP框架
  • 分布式支持
  • 自帶NOSQL

在學(xué)習(xí)了Erlang后,總是想著用Erlang搞點(diǎn)事情出來(lái)讲竿,
于是先后有了:

  • make-proxy 你懂的
  • code-battle 當(dāng)時(shí)Erlang練手的一個(gè)個(gè)人項(xiàng)目

在不斷的學(xué)習(xí)使用 Erlang, Gevent的過(guò)程中拟杉,我也有了自己的感受和看法庄涡,我就依次講講幾個(gè)印象深刻的地方:

簡(jiǎn)單的語(yǔ)法

Erlang沒(méi)有多少語(yǔ)法糖,直接了當(dāng)搬设,看別人代碼很容易穴店。
(特別是基于OTP的項(xiàng)目)

雖然Python基礎(chǔ)語(yǔ)法也很簡(jiǎn)單,并且哲學(xué)就是干一件事情只有最好的一個(gè)辦法拿穴。但是Python的各種語(yǔ)法糖會(huì)讓你看別人代碼不是那么行云流水般流暢泣洞。

比如Python用了三年多了,前幾個(gè)月才知道還可以這么寫:

with A() as a, B() as b, C() as c:
    pass

基于消息傳遞數(shù)據(jù)

Pid ! MyData

如此簡(jiǎn)潔的語(yǔ)法默色, 可以在一個(gè)Erlang Node內(nèi)球凰,跨Node,跨機(jī)器發(fā)送消息腿宰。 MyData 是任何Erlang term呕诉。 也就是任意Erlang合法數(shù)據(jù)。

模式匹配 & 遞歸

-module(fibo).
-export([start/1]).
-vsn(1.1).
 
start(Num) ->
    ResultList = start(Num, 0, []),
    Result = lists:sum(ResultList),
    io:format("~p~n", [ResultList]),
    io:format("~p~n", [Result]),
    ok.
 
start(0, _, Result) ->
    [0 | Result];
 
start(1, _, Result) ->
    [1 | Result];
 
start(Num, 0, Result) ->
    start(Num, 1, [0 | Result]);
 
start(Num, 1, Result) ->
    start(Num, 2, [1 | Result]);
 
start(Num, Acc, [H1, H2 | _Tail] = Result) when Acc =< Num ->
    start(Num, Acc + 1, [H1 + H2 | Result]);
 
start(_, _, Result) ->
    lists:reverse(Result).

這個(gè)簡(jiǎn)單的例子就說(shuō)明了模式匹配和遞歸是多么的符合人的自然思維酗失。而且在模式匹配期間就可以從列表元組中提取自己所需的數(shù)據(jù)义钉。

模式匹配的痛苦

Erlang程序內(nèi)部自己的匹配真的很爽,但是如果遇到需要去 使用
一些 通用數(shù)據(jù)格式協(xié)議 (比如 protobuf, msgpack) 要和其他語(yǔ)言交互的時(shí)候规肴,那真是遇到惡魔

比如一個(gè) protobuf 的定義如下:

message MyMessage {
    required int32 id = 1;
    required string name = 2;
    optional int32 age = 3;
}

Python中是這樣來(lái)打包/解包數(shù)據(jù)的:

# 打包
msg = MyMessage()
msg.id = 1
msg.name = "hi"
data = msg.SerializeToString()

# 解包
msg = MyMessage()
msg.ParseFromString(data)

print msg.id
print msg.name
print msg.age

Erlang是這樣的:

%% 打包
Data = proto_pb:encode_message({
    mymessage,
     1,
     <<"hi">>,
    undefined,
    }).
%% 解包
{mymessage, Id, Name, Age} = proto_pb:decode_cmd(Data).

好捶闸,我們現(xiàn)在在 MyMessage 中加一個(gè)域:

message MyMessage {
    required int32 id = 1;
    required string name = 2;
    optional int32 age = 3;
    optional int32 gender = 4;
}

Python 程序不用修改夜畴, Erlang 必須修改。因?yàn)橐?strong>匹配啊删壮。

在與他人合作的時(shí)候贪绘,如果采用protobuf, msgpack,這些數(shù)據(jù)格式,那么修改消息定義央碟,對(duì)Erlang就是個(gè)災(zāi)難税灌!

雖然我在工作中有著很大的自主權(quán),選擇自己喜歡的技術(shù)亿虽,但是正是由于這個(gè)原因我沒(méi)在大項(xiàng)目中用過(guò)Erlang菱涤。

當(dāng)然,還是自己Erlang技術(shù)不過(guò)硬洛勉。

優(yōu)雅的二進(jìn)制處理

我就舉一個(gè)簡(jiǎn)單的例子粘秆。獲取以大端序表示數(shù)字5的 4bytes 長(zhǎng)的binary
Python中這么做:

import struct
fmt = struct.Struct('>i')
binary = fmt.pack(5)

Erlang:

<<5:32>>.

當(dāng)然你也可以明確的寫成

<<5:32/integer-big>>

這不過(guò)這是默認(rèn)行為而已。

是不是覺得太方便了收毫,Python 的 struct那一大堆格式誰(shuí)TM記得住啊攻走。你心里是不是竊喜 用erlang寫socket發(fā)包收包處理包頭方便多呢? 騷年此再, Erlang在建立socket如果指定了 {packet, N} 參數(shù)昔搂,如果N是 1,2,3,4這寫數(shù)字,那么發(fā)數(shù)據(jù)添加包頭输拇,收數(shù)據(jù)去除包頭已經(jīng)幫你搞定了摘符!

得益與 模式匹配, 你可以這樣:

<<A:16, B:16, C/binary>> = <<D>>
%% A 是 D 的第一和第二字節(jié)
%% B 是 D 的第三和第四字節(jié)
%% C 是 D 的第五到最后剩下的字節(jié)
%% 當(dāng)然D的長(zhǎng)度得》=4 字節(jié)。否則會(huì)報(bào)錯(cuò)淳附。

Python 里有 列表解析议慰, Erlang里也有,并且更強(qiáng)大奴曙。
而且 Erlang 還有 二進(jìn)制列表解析 别凹!

強(qiáng)大的OTP框架

gen_server, gen_fsm, gen_event, supervisor 這些模式組成了OTP的分層架構(gòu)。它讓你寫的程序思路清晰洽糟,各個(gè)組件分工明確炉菲。


Gevent

然后在回來(lái)看看Gevent。

我先學(xué)的Erlang坤溃,然后才看的Gevent這個(gè)Python的庫(kù)拍霜。
剛開始學(xué)Gevent的時(shí)候總是要和Erlang去對(duì)比,覺得這么庫(kù)很難理解啊薪介。比如:

  • 我在Python shell中 gevent.spawn一個(gè)函數(shù)祠饺,它怎么不運(yùn)行? 按照自然的思維就是spawn后就應(yīng)該要運(yùn)行啊汁政。Erlang就是這樣的

  • 為什么spawn出的協(xié)程要join道偷,一調(diào)用join就會(huì)把整個(gè)python process阻塞住缀旁。絲毫感覺不到異步啊。

在學(xué)習(xí)兩次失敗后勺鸦,后來(lái)終于想明白了:

  • shell 中spawn不運(yùn)行并巍,是因?yàn)間event 的event loop沒(méi)有跑起來(lái),無(wú)法去調(diào)度greenlet换途。

  • join就是在啟動(dòng)gevent 的 gevent loop懊渡,一旦gevent loop通過(guò)其他方式啟動(dòng)起來(lái)了, 那么就可以在程序中自然的spawn進(jìn)程军拟。新spawn的 greenlet 會(huì)被調(diào)度執(zhí)行剃执。

當(dāng)跨過(guò)這個(gè)障礙后,學(xué)習(xí)和使用Gevent容易了很多吻谋。

得易于Python這么有好的語(yǔ)言忠蝗,大量的庫(kù),還有Pycharm這么優(yōu)秀的IDE漓拾。 (我以前是VIM死忠。戒祠。骇两。),再加上gevent這么給力的庫(kù)姜盈,寫并發(fā)真是輕松愜意啊低千。

然后就有了基于Gevent的項(xiàng)目:

最近對(duì)Gevent又有了新的感悟。

所以有了這篇文章馏颂,以及用了一天時(shí)間將 codebattle 的服務(wù)器寫了個(gè)Gevent的版本 codebattle-server-python

結(jié)論

Erlang很好示血,天生就是為了服務(wù)器編程而生。
但從我上面的 模式匹配 的描述救拉,以及Python的易用性难审,
以后會(huì)更多的用Gevent


最后附上一個(gè) 激情視頻

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市亿絮,隨后出現(xiàn)的幾起案子告喊,更是在濱河造成了極大的恐慌,老刑警劉巖派昧,帶你破解...
    沈念sama閱讀 211,496評(píng)論 6 491
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件黔姜,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡蒂萎,警方通過(guò)查閱死者的電腦和手機(jī)秆吵,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,187評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)五慈,“玉大人纳寂,你說(shuō)我怎么就攤上這事主穗。” “怎么了烈疚?”我有些...
    開封第一講書人閱讀 157,091評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵黔牵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我爷肝,道長(zhǎng)猾浦,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,458評(píng)論 1 283
  • 正文 為了忘掉前任灯抛,我火速辦了婚禮金赦,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘对嚼。我一直安慰自己夹抗,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,542評(píng)論 6 385
  • 文/花漫 我一把揭開白布纵竖。 她就那樣靜靜地躺著漠烧,像睡著了一般。 火紅的嫁衣襯著肌膚如雪靡砌。 梳的紋絲不亂的頭發(fā)上已脓,一...
    開封第一講書人閱讀 49,802評(píng)論 1 290
  • 那天,我揣著相機(jī)與錄音通殃,去河邊找鬼度液。 笑死,一個(gè)胖子當(dāng)著我的面吹牛画舌,可吹牛的內(nèi)容都是我干的堕担。 我是一名探鬼主播,決...
    沈念sama閱讀 38,945評(píng)論 3 407
  • 文/蒼蘭香墨 我猛地睜開眼曲聂,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼霹购!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起句葵,我...
    開封第一講書人閱讀 37,709評(píng)論 0 266
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤厕鹃,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后乍丈,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體剂碴,經(jīng)...
    沈念sama閱讀 44,158評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,502評(píng)論 2 327
  • 正文 我和宋清朗相戀三年轻专,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了忆矛。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,637評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖催训,靈堂內(nèi)的尸體忽然破棺而出洽议,到底是詐尸還是另有隱情,我是刑警寧澤漫拭,帶...
    沈念sama閱讀 34,300評(píng)論 4 329
  • 正文 年R本政府宣布亚兄,位于F島的核電站,受9級(jí)特大地震影響采驻,放射性物質(zhì)發(fā)生泄漏审胚。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,911評(píng)論 3 313
  • 文/蒙蒙 一礼旅、第九天 我趴在偏房一處隱蔽的房頂上張望膳叨。 院中可真熱鬧,春花似錦痘系、人聲如沸菲嘴。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,744評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)龄坪。三九已至,卻和暖如春复唤,著一層夾襖步出監(jiān)牢的瞬間悉默,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,982評(píng)論 1 266
  • 我被黑心中介騙來(lái)泰國(guó)打工苟穆, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人唱星。 一個(gè)月前我還...
    沈念sama閱讀 46,344評(píng)論 2 360
  • 正文 我出身青樓雳旅,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親间聊。 傳聞我的和親對(duì)象是個(gè)殘疾皇子攒盈,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,500評(píng)論 2 348

推薦閱讀更多精彩內(nèi)容