安徽新华电脑专修学院_安徽电脑培训_安徽电脑培训学校_合肥电脑培训

當前位置:首頁 > 網站舊欄目 > 學習園地 > 設計軟件教程 > 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!
安徽新華電腦學校專業職業規劃師為你提供更多幫助【在線咨詢
主站蜘蛛池模板: 游离二氧化硅处理仪-恒温恒湿称重系统-智能蒸馏仪-硫化物酸化吹气仪-萃取仪-COD消解仪 | 网带式等温正火生产线_燃气式铝合金加热炉_燃气式烘干窑炉-湖州中科炉业科技有限公司 | 进销存软件|仓库管理软件|库存物资出入库|ERP生产|MRP|易特软件官方网站 | 湖北大洋塑胶有限公司|AGR|PPR|RTP|HDPE|e-PSP钢塑复合压力管道生产厂家 | 互动投影_全息投影_提供一站式互动投影解决方案_水滴石科技 | 四川蜀易控科技有限公司-酒店客房控制系统-智慧酒店智能化客房控制系统生产厂家 | 江苏华海诚科新材料有限公司、连云港华海诚科新材料有限公司、连云港新材料 | 连续式回转炉_间歇式回转炉_燃气式回转炉_电加热回转炉-长兴博达机械配件有限公司 | 上海眸社设计-上海专业的VI设计,宣传册设计,画册设计,折页设计公司 | 郫都人才网_郫都招聘网_求职找工作平台 | 深圳蓝枫印刷_画册印刷_彩页印刷_宣传册印刷_包装盒印刷_彩盒印刷厂_不干胶印刷厂 | 西安网站建设,西安网站设计制作,西安短视频拍摄_短视频运营就选动力无限网络推广公司 | 汽车漆品牌|家具漆代理|涂料加盟厂家|家具漆|汽车漆-邦派漆官网 汽车漆|汽车油漆|工业油漆涂料|汽车漆加盟-佛山市科涂涂料有限公司 | 南通出国劳务公司-如东海外经济技术合作有限公司-启东,海门,如皋,海安出国劳务 | 重庆自考网-重庆自学考试 | 装盒机|全自动封盒机|纸盒子包装机|高速装盒机定制-温州胜泰机械有限公司 | 中山四海家具制造有限公司| 组合式空调机组-吊顶式新风换气机-消防高温排烟风机-德州宏楚空调设备有限公司 | 江门市金环电器有限公司| 南京展览公司|南京会展制作|南京展台搭建|南京展厅设计|企业展览馆 | 启东华立石油化工机械设备有限公司|过滤器|混合机|消声器|混合器|管道过滤器|空气过滤器|精细过滤器 | 太原重卡叔叔运输有限公司-山西太原大件运输、太原物流公司、太原货运物流、太原大件运输、太原货运信息、长治物流公司、长治大件运输、晋城物流公司、晋城大件运输、忻州大件运输、朔州大件运输、阳泉大件运输、大同大件运输、吕梁大件运输、临汾大件运输、运城大件运城 | 中科盛世酒窖 - 酒窖设计_酒窖工程_酒窖空调设备_恒温酒柜定制_私人别墅家庭酒窖_不锈钢红酒柜_实木雪茄柜 | 微孔板恒温振荡器-超声波探伤试块-微孔板迷你离心机-南京互川电子有限公司 | 主轴-电主轴-高速电机-高速电主轴厂家|瑞德沃斯品牌 | 呼吸家官网|肺功能检测仪生产厂家|国产肺功能仪知名品牌|肺功能检测仪|肺功能测试仪|婴幼儿肺功能仪|弥散残气肺功能仪|肺功能测试系统|广州红象医疗科技有限公司|便携式肺功能仪|大肺功能仪|呼吸康复一体机|儿童肺功能仪|肺活量计|医用简易肺功能仪|呼吸康复系统|肺功能仪|弥散肺功能仪(大肺)|便携式肺功能检测仪|肺康复|呼吸肌力测定肺功能仪|肺功能测定仪|呼吸神经肌肉刺激仪|便携式肺功能 | 搜药网-中药材中成药大全网普及中医中药功效归经知识,中药学知识天地 | 湖南净声源环保科技有限公司是一家专业从事噪声治理和建筑声学设计生态环境综合治理服务的企业,专业从事株洲电梯隔音治理,湘潭中央空调降噪处理,衡阳邵阳冷却塔噪音治理,岳阳常德大型风机噪声隔音降噪,张家界空压机噪声治理,益阳配电房变压器噪声治理,专业郴州永州工厂企业车间噪声治理,怀化娄底专业机械设备减振降治理,武汉噪音治理隔音降噪公司,孝感噪音治理,立式球磨机的噪声控制,专业隔音降噪公司,、以及各类机械动力设备减振降噪噪声治理的公司,同时为客户提供咨询与解决方案 | 牡丹江网络公司,牡丹江网站建设专家|网络推广|网络营销|黑龙江艺通网络技术开发有限公司 | 阴_阳离子聚丙烯酰胺价格_聚合氯化铝厂家_聚合硫酸铁-巩义市亿洋水处理材料有限公司 | 欧氏运动木地板,体育木地板厂家,篮球木地板价格_欧氏体育木地板 欧派板材官网 | 全屋定制板材 专业供应商 | 育婴师_催乳师证_月嫂证怎么考_育婴师证报考需要什么条件-家政培训网 | 手动叉车|电动搬运车|电动升降平台-牛力机械制造有限公司官网 | 欧派板材官网 | 全屋定制板材 专业供应商| 山东岱新起重机械有限公司,单梁桥式起重机,双梁桥式起重机,通用式门式起重机,欧式起重机系列 | 江西佛像厂 江西法器厂 江西抚州东乡江弘法器有限公司 东乡江弘法器厂 佛像厂 法器厂 | 网站建设|外贸网站建设|做网站公司-济南超越互联-推荐 | 河南反渗透设备,河南纯净水设备,河南软化水设备,郑州EDI超纯水设备,郑州水处理设备厂家_河南江宇环保科技有限公司 | 潍坊劲昊磁电科技有限公司-电磁除铁器,永磁除铁器,管道式除铁器,金属探测仪,磁滚筒,输送设备,给料设备,破碎设备 | 台车炉厂家_台车式退火炉_台车式回火炉—安徽大新工业炉有限公司 | 山西华盛筑景装饰,山西专业公装公司,太原公装装修公司,包括:办公室,酒店宾馆,商铺店铺,学校幼儿园,会所会馆饭店餐馆等装修设计 |