好男人天堂网,久久精品国产这里是免费,国产精品成人一区二区,男人天堂网2021,男人的天堂在线观看,丁香六月综合激情

當前位置:首頁 > 網站舊欄目 > 學習園地 > 設計軟件教程 > Erlang:一個通用的網絡服務器

Erlang:一個通用的網絡服務器
2010-01-13 23:12:05  作者:  來源:
前面幾篇文章里談到了Erlang的gen_tcp網絡編程和Erlang/OPT的gen_server模塊,現在讓我們將它們兩者綁定在一起

大多數人認為“服務器”意味著網絡服務器,但Erlang使用這個術語時表達的是更抽象的意義
gen_serer在Erlang里是基于它的消息傳遞協議來操作的服務器,我們可以在此基礎上嫁接一個TCP服務器,但這需要一些工作

網絡服務器的結構
大部分網絡服務器有相似的架構
首先它們創建一個監聽socket來監聽接收的連接
然后它們進入一個接收狀態,在這里一直循環接收新的連接,直到結束(結束表示連接已經到達并開始真正的client/server工作)

先看看前面網絡編程里的echo server的例子:
Java代碼 復制代碼
  1. -module(echo).   
  2. -author('Jesse E.I. Farmer <jesse@20bits.com>').   
  3. -export([listen/1]).   
  4.   
  5. -define(TCP_OPTIONS, [binary, {packet, 0}, {active, false}, {reuseaddr, true}]).   
  6.   
  7. % Call echo:listen(Port) to start the service.   
  8. listen(Port) ->   
  9.     {ok, LSocket} = gen_tcp:listen(Port, ?TCP_OPTIONS),   
  10.     accept(LSocket).   
  11.   
  12. % Wait for incoming connections and spawn the echo loop when we get one.   
  13. accept(LSocket) ->   
  14.     {ok, Socket} = gen_tcp:accept(LSocket),   
  15.     spawn(fun() -> loop(Socket) end),   
  16.     accept(LSocket).   
  17.   
  18. % Echo back whatever data we receive on Socket.   
  19. loop(Socket) ->   
  20.     case gen_tcp:recv(Socket, 0) of   
  21.         {ok, Data} ->   
  22.             gen_tcp:send(Socket, Data),   
  23.             loop(Socket);   
  24.         {error, closed} ->   
  25.             ok   
  26.     end.  

你可以看到,listen會創建一個監聽socket并馬上調用accept
accept會等待進來的連接,創建一個新的worker(loop)來處理真正的工作,然后等待下一個連接

在這部分代碼里,父進程擁有listen socket和accept loop兩者
后面我們會看到,如果我們集成accept/listen loop和gen_server的話這樣做并不好

抽象網絡服務器
網絡服務器有兩部分:連接處理和業務邏輯
上面講到,連接處理對每個網絡服務器都是幾乎一樣的
理想狀態下我們可以這樣做:
Java代碼 復制代碼
  1. -module(my_server).   
  2. start(Port) ->   
  3.   connection_handler:start(my_server, Port, businees_logic).   
  4.   
  5. business_logic(Socket) ->   
  6.   % Read data from the network socket and do our thang!  

讓我們繼續完成它

實現一個通用網絡服務器
使用gen_server來實現一個網絡服務器的問題是,gen_tcp:accept調用是堵塞的
如果我們在服務器的初始化例程里調用它,那么整個gen_server機制都會堵塞,直到客戶端建立連接

有兩種方式來繞過這個問題
一種方式為使用低級連接機制來支持非堵塞(或異步)accept
有許多方法來支持這樣做,最值得注意的是gen_tcp:controlling_process,它幫你管理當客戶端建立連接時誰接受了什么消息

我認為另一種比較簡單而更優雅的方式是,一個單獨的進程來監聽socket
該進程做兩件事:監聽“接收連接”消息以及分配新的接收器
當它接收一條新的“接收連接”的消息時,就知道該分配新的接收器了

接收器可以任意調用堵塞的gen_tcp:accept,因為它允許在自己的進程里
當它接受一個連接后,它發出一條異步消息傳回給父進程,并且立即調用業務邏輯方法

這里是代碼,我加了一些注釋,希望可讀性還可以:
Java代碼 復制代碼
  1. -module(socket_server).   
  2. -author('Jesse E.I. Farmer <jesse@20bits.com>').   
  3. -behavior(gen_server).   
  4.   
  5. -export([init/1, code_change/3, handle_call/3, handle_cast/2, handle_info/2, terminate/2]).   
  6. -export([accept_loop/1]).   
  7. -export([start/3]).   
  8.   
  9. -define(TCP_OPTIONS, [binary, {packet, 0}, {active, false}, {reuseaddr, true}]).   
  10.   
  11. -record(server_state, {   
  12.         port,   
  13.         loop,   
  14.         ip=any,   
  15.         lsocket=null}).   
  16.   
  17. start(Name, Port, Loop) ->   
  18.     State = #server_state{port = Port, loop = Loop},   
  19.     gen_server:start_link({local, Name}, ?MODULE, State, []).   
  20.   
  21. init(State = #server_state{port=Port}) ->   
  22.     case gen_tcp:listen(Port, ?TCP_OPTIONS) of   
  23.         {ok, LSocket} ->   
  24.             NewState = State#server_state{lsocket = LSocket},   
  25.             {ok, accept(NewState)};   
  26.         {error, Reason} ->   
  27.             {stop, Reason}   
  28.     end.   
  29.   
  30. handle_cast({accepted, _Pid}, State=#server_state{}) ->   
  31.     {noreply, accept(State)}.   
  32.   
  33. accept_loop({Server, LSocket, {M, F}}) ->   
  34.     {ok, Socket} = gen_tcp:accept(LSocket),   
  35.     % Let the server spawn a new process and replace this loop   
  36.     % with the echo loop, to avoid blocking   
  37.     gen_server:cast(Server, {accepted, self()}),   
  38.     M:F(Socket).   
  39.       
  40. % To be more robust we should be using spawn_link and trapping exits   
  41. accept(State = #server_state{lsocket=LSocket, loop = Loop}) ->   
  42.     proc_lib:spawn(?MODULE, accept_loop, [{self(), LSocket, Loop}]),   
  43.     State.   
  44.   
  45. % These are just here to suppress warnings.   
  46. handle_call(_Msg, _Caller, State) -> {noreply, State}.   
  47. handle_info(_Msg, Library) -> {noreply, Library}.   
  48. terminate(_Reason, _Library) -> ok.   
  49. code_change(_OldVersion, Library, _Extra) -> {ok, Library}.  

我們使用gen_server:cast來傳遞異步消息給監聽進程,當監聽進程接受accepted消息后,它分配一個新的接收器

目前,這個服務器不是很健壯,因為如果無論什么原因活動的接收器失敗以后,服務器會停止接收新的連接
為了讓它變得更像OTP,我們因該捕獲異常退出并且在連接失敗時分配新的接收器

一個通用的echo服務器
echo服務器是最簡單的服務器,讓我們使用我們新的抽象socket服務器來寫它:
Java代碼 復制代碼
  1. -module(echo_server).   
  2. -author('Jesse E.I. Farmer <jesse@20bits.com>').   
  3.   
  4. -export([start/0, loop/1]).   
  5.   
  6. % echo_server specific code   
  7. start() ->   
  8.     socket_server:start(?MODULE, 7000, {?MODULE, loop}).   
  9. loop(Socket) ->   
  10.     case gen_tcp:recv(Socket, 0) of   
  11.         {ok, Data} ->   
  12.             gen_tcp:send(Socket, Data),   
  13.             loop(Socket);   
  14.         {error, closed} ->   
  15.             ok   
  16.     end.  

你可以看到,服務器只含有自己的業務邏輯
連接處理被封裝到socket_server里面
而這里的loop方法也和最初的echo服務器一樣

希望你可以從中學到點什么,我覺得我開始理解Erlang了

歡迎回復,特別關于是如何改進我的代碼,cheers!
安徽新華電腦學校專業職業規劃師為你提供更多幫助【在線咨詢
国产不卡在线观看视频| 麻豆网站在线看| 久草免费在线色站| 精品国产一区二区三区国产馆| 国产亚洲精品aaa大片| 国产一区二区精品久久91| 成人免费一级毛片在线播放视频| 日韩中文字幕一区| 欧美1区2区3区| 日本久久久久久久 97久久精品一区二区三区 狠狠色噜噜狠狠狠狠97 日日干综合 五月天婷婷在线观看高清 九色福利视频 | 国产不卡在线观看| 九九免费精品视频| 成人av在线播放| 99色视频在线观看| 国产美女在线一区二区三区| 999久久狠狠免费精品| 欧美一区二区三区在线观看| 国产成人啪精品| 久久精品大片| 一级毛片视频播放| 999精品视频在线| 99久久精品国产麻豆| 日本在线播放一区| 欧美a级片免费看| 国产成人精品影视| 日本久久久久久久 97久久精品一区二区三区 狠狠色噜噜狠狠狠狠97 日日干综合 五月天婷婷在线观看高清 九色福利视频 | 日本久久久久久久 97久久精品一区二区三区 狠狠色噜噜狠狠狠狠97 日日干综合 五月天婷婷在线观看高清 九色福利视频 | 国产亚洲男人的天堂在线观看| 成人影院一区二区三区| 99久久精品费精品国产一区二区| 美女免费毛片| 日本在线不卡免费视频一区| 国产亚洲精品aaa大片| 成人影视在线观看| a级精品九九九大片免费看| 欧美α片无限看在线观看免费| 国产不卡在线观看| 精品国产一区二区三区久久久蜜臀| 久久99中文字幕久久| 精品在线视频播放| 国产一区二区精品尤物| 欧美国产日韩一区二区三区| 久久精品成人一区二区三区| 四虎久久影院| 欧美一区二区三区在线观看| 欧美日本免费| 九九久久国产精品| 天天做日日爱| 国产欧美精品午夜在线播放| 天天做人人爱夜夜爽2020 | 国产伦久视频免费观看视频| 青青青草影院| 成人在免费观看视频国产| 亚洲精品久久久中文字| 国产不卡在线观看视频| 国产a一级| 国产亚洲男人的天堂在线观看| 四虎影视库| 午夜久久网| 成人高清护士在线播放| 日韩中文字幕一区二区不卡| 国产成人啪精品| 成人在免费观看视频国产| 国产麻豆精品高清在线播放| 免费毛片基地| 日韩专区第一页| 免费一级片在线观看| 韩国三级视频在线观看| 欧美一区二区三区在线观看| 精品视频在线看| 成人影院一区二区三区| 国产麻豆精品| 久久久久久久男人的天堂| 精品国产一区二区三区免费 | 黄视频网站免费观看| 尤物视频网站在线| 亚洲爆爽| 午夜家庭影院| 99热精品一区| 精品国产一区二区三区久久久狼| 国产一区二区福利久久| 国产一区二区精品久久91| 免费的黄视频| 国产伦久视频免费观看视频| 韩国三级视频在线观看| 超级乱淫伦动漫| 精品国产一区二区三区免费 | 精品国产一区二区三区久| 精品国产一区二区三区国产馆| 亚洲精品影院| 精品在线观看一区| 国产一区国产二区国产三区| 国产精品免费精品自在线观看| 亚洲 欧美 91| 日韩在线观看视频黄| 999精品视频在线| 九九免费高清在线观看视频| 天天做日日爱夜夜爽| 日韩中文字幕一区| 成人免费网站视频ww| 亚欧乱色一区二区三区| 精品视频免费在线| 深夜做爰性大片中文| 91麻豆精品国产自产在线| 天天做日日爱夜夜爽| 四虎影视库| 四虎影视久久久| 欧美日本免费| 黄视频网站在线免费观看| a级黄色毛片免费播放视频| 久久精品成人一区二区三区| 色综合久久天天综合| 精品久久久久久中文| 欧美激情影院| 可以免费在线看黄的网站| 精品国产一区二区三区精东影业 | 亚欧视频在线| 四虎影视精品永久免费网站 | 国产成+人+综合+亚洲不卡| 国产国语对白一级毛片| 国产91素人搭讪系列天堂| 日韩综合| 国产视频久久久久| 成人高清视频免费观看| 国产精品自拍亚洲| 九九九国产| 可以免费看毛片的网站| 国产麻豆精品hdvideoss| a级黄色毛片免费播放视频| 国产不卡高清在线观看视频| 国产不卡在线观看| 一本高清在线| 午夜久久网| 亚洲 男人 天堂| 久久国产影院| 国产综合91天堂亚洲国产| 欧美国产日韩精品| 999精品在线| 香蕉视频三级| 二级片在线观看| 一级毛片看真人在线视频| 日本特黄特色aaa大片免费| 国产精品123| 国产a视频精品免费观看| 天天做日日干| 午夜欧美成人久久久久久| 精品国产亚洲一区二区三区| 日本伦理黄色大片在线观看网站| 精品国产香蕉伊思人在线又爽又黄| 日韩在线观看免费完整版视频| 国产一区二区精品| 精品毛片视频| 国产极品精频在线观看| 青草国产在线| 成人高清视频在线观看| 一级女性大黄生活片免费| 韩国三级一区| 青青久久国产成人免费网站| 国产成人女人在线视频观看| 国产国语对白一级毛片| 99色播| 91麻豆国产级在线| 国产精品1024永久免费视频| 久久精品店| 国产成人啪精品| 美女免费精品视频在线观看| 欧美激情一区二区三区在线| 欧美激情在线精品video| 日韩在线观看视频网站| 日本久久久久久久 97久久精品一区二区三区 狠狠色噜噜狠狠狠狠97 日日干综合 五月天婷婷在线观看高清 九色福利视频 | 国产精品123| 久草免费在线色站| 欧美夜夜骑 青草视频在线观看完整版 久久精品99无色码中文字幕 欧美日韩一区二区在线观看视频 欧美中文字幕在线视频 www.99精品 香蕉视频久久 | 精品在线观看一区| 黄视频网站在线看| 你懂的福利视频| 午夜激情视频在线观看| 亚洲天堂免费观看| 国产成人啪精品| 精品国产三级a| 欧美国产日韩久久久| 免费一级生活片| 久久精品免视看国产成人2021| 日韩专区在线播放| 久久成人亚洲| 成人影视在线播放| 成人高清视频在线观看| 国产网站免费| 国产麻豆精品hdvideoss| 国产伦精品一区二区三区在线观看| 香蕉视频久久| 成人影院一区二区三区| 亚洲 男人 天堂| 超级乱淫黄漫画免费| 成人a大片高清在线观看| 日韩专区亚洲综合久久| 久久成人亚洲| 九九久久99| 国产精品自拍一区| 国产福利免费视频|