亚洲人成网站在线播放2019 _日韩国产欧美精品_久久夜色精品国产欧美乱_在线视频福利一区

當前位置:首頁 > 網站舊欄目 > 學習園地 > 設計軟件教程 > gen_server tasting 之超簡單名稱服務

gen_server tasting 之超簡單名稱服務
2010-01-13 23:16:59  作者:  來源:
     年假不能白休,時間不能浪費,看了 erlang 程序設計的 gen_server 章節,為了更好的理解、掌握于是上手寫一個名稱(鍵值)服務器。這個 lzy_name_svc 服務器是基于 otp gen_server 寫成的,在底層鍵值被保存在了 erlang 的進程字典里,并且用于存儲字典的進程是可以替換的,可以通過 lzy_name_svc:start/1 啟動服務時指定,缺省情況保存在“當前” erlang 進程中。閑話少敘,代碼貼上。

 

Erlang代碼 復制代碼
  1. -module(lzy_name_svc).   
  2.   
  3. -behaviour(gen_server).   
  4.   
  5. -export([init/0, init/1, handle_call/3, handle_cast/2, handle_info/2, terminate/2, code_change/3]).   
  6. -export([start/0, start/1, stop/0, save/2, load/1, load_all/0, remove/1, remove_all/0]).   
  7.   
  8. %% Interface functions.   
  9.   
  10. start() ->   
  11.     gen_server:start_link({local, ?MODULE}, ?MODULE, [], []).   
  12.   
  13. start(Args) ->   
  14.     gen_server:start_link({local, ?MODULE}, ?MODULE, Args, []).   
  15.   
  16. stop() ->   
  17.     gen_server:call(?MODULE, stop).   
  18.   
  19.   
  20. %% @spec save(Key, Value) -> OldValue.   
  21. save(Key, Value) ->   
  22.     gen_server:call(?MODULE, {save, Key, Value}).   
  23.   
  24. %% @spec load(Key) -> Value.   
  25. load(Key) ->   
  26.     gen_server:call(?MODULE, {load, Key}).   
  27.   
  28. %% @spec load_all() -> [{Key, Value}].   
  29. load_all() ->   
  30.     gen_server:call(?MODULE, {load_all}).   
  31.   
  32. %% @spec remove(Key) -> Value.   
  33. remove(Key) ->   
  34.     gen_server:call(?MODULE, {remove, Key}).   
  35.   
  36. %% @spec remove_all() -> [{Key, Value}].   
  37. remove_all() ->   
  38.     gen_server:call(?MODULE, {remove_all}).   
  39.   
  40.   
  41. %%  Callback functions.   
  42.   
  43. init([]) ->   
  44.     {ok, local};   
  45.   
  46. init([{isolation, NameServer}]) ->   
  47.     {ok, {isolation, NameServer}}.   
  48.   
  49. handle_call({save, Key, Value}, _From, NameServer) ->   
  50.     {reply, do_save(Key, Value, NameServer), NameServer};   
  51.   
  52. handle_call({load, Key}, _From, NameServer) ->   
  53.     {reply, do_load(Key, NameServer), NameServer};   
  54.   
  55. handle_call({load_all}, _From, NameServer) ->   
  56.     {reply, do_load_all(NameServer), NameServer};   
  57.   
  58. handle_call({remove, Key}, _From, NameServer) ->   
  59.     {reply, do_remove(Key, NameServer), NameServer};   
  60.   
  61. handle_call({remove_all}, _From, NameServer) ->   
  62.     {reply, do_remove_all(NameServer), NameServer};   
  63.   
  64. handle_call({stop}, _From, NameServer) ->   
  65.     {stop, normal, stopped, NameServer}.   
  66.   
  67. %% Default implement.   
  68.   
  69. handle_cast(_Msg, State) ->   
  70.     {noreply, State}.   
  71.        
  72. handle_info(_Info, State) ->   
  73.     {noreply, State}.   
  74.   
  75. terminate(_Reason, _State) ->   
  76.     ok.   
  77.   
  78. code_change(_OldVsn, State, _Extra) ->   
  79.     {ok, State}.   
  80.   
  81. %% Private functions.   
  82.   
  83. do_save(Key, Value, {isolation, NameServer}) ->   
  84.     NameServer ! {self(), save, Key, Value},   
  85.     receive   
  86.         Msg -> Msg   
  87.     end;   
  88.   
  89. do_save(Key, Value, _) ->   
  90.     erlang:put(Key, Value).   
  91.   
  92. do_load(Key, {isolation, NameServer}) ->   
  93.     NameServer ! {self(), load, Key},   
  94.     receive   
  95.         Msg -> Msg   
  96.     end;   
  97.   
  98. do_load(Key, _) ->   
  99.     erlang:get(Key).   
  100.   
  101. do_load_all({isolation, NameServer}) ->   
  102.     NameServer ! {self(), load_all},   
  103.     receive   
  104.         Msg -> Msg   
  105.     end;   
  106.   
  107. do_load_all(_) ->   
  108.     erlang:get().   
  109.   
  110. do_remove(Key, {isolation, NameServer}) ->   
  111.     NameServer ! {self(), remove, Key},   
  112.     receive   
  113.         Msg -> Msg   
  114.     end;   
  115.   
  116. do_remove(Key, _) ->   
  117.     erlang:erase(Key).   
  118.   
  119. do_remove_all({isolation, NameServer}) ->   
  120.     NameServer ! {self(), remove_all},   
  121.     receive   
  122.         Msg -> Msg   
  123.     end;   
  124.   
  125. do_remove_all(_) ->   
  126.     erlang:erase().  

 

上面這段代碼就是 lzy_name_svc 名稱服務了,有些地方寫得有點冗余,呵呵。

 

          為了能夠替換字典進程來測試驗證名稱服務功能,還寫了一個超簡單的 foo_svc 服務,用來和 lzy_name_svc 通信完成進程字典存取。

 

Erlang代碼 復制代碼
  1. -module(foo_svc).   
  2.   
  3. -export([start/0, load_all/0, server_pid/0]).   
  4.   
  5. start() ->   
  6.     register(fs, spawn(fun() -> loop() end)).   
  7.   
  8. load_all() ->   
  9.     fs !  {self(), load_all},   
  10.     receive   
  11.         Msg -> Msg   
  12.     end.   
  13.        
  14. server_pid() ->   
  15.     fs ! { self(), server_pid},   
  16.     receive   
  17.         Msg -> Msg   
  18.     end.   
  19.        
  20. loop() ->   
  21.     receive   
  22.         {From, save, Key, Value} ->   
  23.             From ! erlang:put(Key, Value),   
  24.             loop();   
  25.         {From, load, Key} ->   
  26.             From ! erlang:get(Key),   
  27.             loop();   
  28.         {From, load_all} ->   
  29.             From ! erlang:get(),   
  30.             loop();   
  31.         {From, remove, Key} ->   
  32.             From ! erlang:erase(Key),   
  33.             loop();   
  34.         {From, remove_all} ->   
  35.             From ! erlang:erase(),   
  36.             loop();   
  37.         {From, server_pid} ->   
  38.             From ! self(),   
  39.             loop()   
  40.     end.  

 

          下面的代碼就是創建和調用服務的相關代碼了,一起貼上來。第一段是以缺省方式啟動了 lzy_name_svc 服務,并向存取 abc -> 123 名稱。

 

Erlang代碼 復制代碼
  1. C:\Program Files\erl5.6.4\usr>..\bin\erl -sname server   
  2. Eshell V5.6.4  (abort with ^G)   
  3. (server@lzy)1> c(lzy_name_svc).   
  4. {ok,lzy_name_svc}   
  5. (server@lzy)2> c(foo_svc.erl).   
  6. {ok,foo_svc}   
  7. (server@lzy)3> lzy_name_svc:start().   
  8. {ok,<0.47.0>}   
  9. (server@lzy)4> lzy_name_svc:save(abc, 123).   
  10. undefined   
  11. (server@lzy)5> lzy_name_svc:load(abc).   
  12. 123  
  13. (server@lzy)6> lzy_name_svc:load(efg).   
  14. undefined   
  15. (server@lzy)7> lzy_name_svc:load_all().   
  16. [{abc,123},   
  17.  {'$ancestors',[<0.35.0>]},   
  18.  {'$initial_call',{gen,init_it,   
  19.                        [gen_server,<0.35.0>,<0.35.0>,   
  20.                         {local,lzy_name_svc},   
  21.                         lzy_name_svc,[],[]]}}]   
  22. (server@lzy)8> lzy_name_svc:remove(abc).   
  23. 123  
  24. (server@lzy)9> lzy_name_svc:load(abc).   
  25. undefined  

 

下面這段是啟動 foo_svc 服務,用它創建的進程來專門存儲名稱數據,是通過 lzy_name_svc:start/1 傳入的 PID。

 

Erlang代碼 復制代碼
  1. C:\Program Files\erl5.6.4\usr>..\bin\erl -sname server   
  2. Eshell V5.6.4  (abort with ^G)   
  3. (server@lzy)1> foo_svc:start().   
  4. true   
  5. (server@lzy)2> NameSvcPid = foo_svc:server_pid().   
  6. <0.37.0>   
  7. (server@lzy)3> lzy_name_svc:start([{isolation, NameSvcPid}]).   
  8. {ok,<0.40.0>}   
  9. (server@lzy)4> lzy_name_svc:save(abc, 123).   
  10. undefined   
  11. (server@lzy)5> lzy_name_svc:load(abc).   
  12. 123  
  13. (server@lzy)6> foo_svc:load_all().   
  14. [{abc,123}]   
  15. (server@lzy)7> lzy_name_svc:remove_all().   
  16. [{abc,123}]   
  17. (server@lzy)8> foo_svc:load_all().   
  18. []  

 

          上邊的兩段都是在同一機器上的同一 erlang 節點上完成服務調用的,下面這段代碼是 lzy_name_svc 服務基于上邊狀態時,在同一機器的另外了個 erlang 節點上通過 rpc 庫完成服務調用的。

 

Erlang代碼 復制代碼
  1. C:\Program Files\erl5.6.4\usr>..\bin\erl -sname client1   
  2. Eshell V5.6.4  (abort with ^G)   
  3. (client1@lzy)1> rpc:call(server@lzy, lzy_name_svc, save, [abc, 123]).   
  4. undefined   
  5. (client1@lzy)2> rpc:call(server@lzy, foo_svc, load_all, []).   
  6. [{abc,123}]  
 

呵呵,挺入門的,就當做為學習過程的記錄吧。看好 erlang。

 

          在學習的過程中,有一個事情比較不解,就是對于 字典進程的 “熱替換” 我想本應該是可以通過 gen_server behaviour 用于“熱代碼替換”的 code_change 方法完成的,但試了幾次都達不到目的,服務倒是跑的正常,可是字典進程就是不能熱替換,code_change 正常返回,可是名稱數據卻還是原有字典進程的。測試驗證代碼如下:

 

Erlang代碼 復制代碼
  1. C:\Program Files\erl5.6.4\usr>..\bin\erl -sname server   
  2. Eshell V5.6.4  (abort with ^G)   
  3. (server@lzy)1> lzy_name_svc:start().   
  4. {ok,<0.37.0>}   
  5. (server@lzy)2> lzy_name_svc:save(abc, 123).   
  6. undefined   
  7. (server@lzy)3> lzy_name_svc:load_all().   
  8. [{abc,123},   
  9.  {'$ancestors',[<0.35.0>]},   
  10.  {'$initial_call',{gen,init_it,   
  11.                        [gen_server,<0.35.0>,<0.35.0>,   
  12.                         {local,lzy_name_svc},   
  13.                         lzy_name_svc,[],[]]}}]   
  14. (server@lzy)4> foo_svc:start().   
  15. true   
  16. (server@lzy)5> NameSvcPid = foo_svc:server_pid().   
  17. <0.41.0>   
  18. (server@lzy)6> foo_svc:load_all().   
  19. []   
  20. (server@lzy)7> lzy_name_svc:code_change(foo, NameSvcPid, foo).   
  21. {ok,<0.41.0>}   
  22. (server@lzy)8> lzy_name_svc:load_all().   
  23. [{abc,123},   
  24.  {'$ancestors',[<0.35.0>]},   
  25.  {'$initial_call',{gen,init_it,   
  26.                        [gen_server,<0.35.0>,<0.35.0>,   
  27.                         {local,lzy_name_svc},   
  28.                         lzy_name_svc,[],[]]}}]  

 

還請哪位 erlang guru 指點~

 

// 2009.02.07 22:52 添加 ////

 

這里提供了該名稱服務的新迭代版本。

gen_server tasting 之超簡單名稱服務(續)

 

添加了如下功能:

 

  1. 使用 otp 監控樹保證服務可靠性。
  2. 添加日志功能,記錄警告事件。
  3. 將名稱服務打包為 application。
  4. 開放 socket 服務,使用 vsns://verb /param 自定義協議對外提供訪問支持。

 


安徽新華電腦學校專業職業規劃師為你提供更多幫助【在線咨詢
亚洲人成网站在线播放2019 _日韩国产欧美精品_久久夜色精品国产欧美乱_在线视频福利一区
欧美污视频久久久| 久久久久久亚洲| 日韩毛片在线免费看| 懂色av粉嫩av蜜臀av| 亚洲精品一区二区三区樱花| 一区二区精品在线| 亚洲人成网站在线观看播放| 亚洲一区二区在线| 懂色av粉嫩av蜜臀av| 日韩av影视| 欧美中文字幕在线观看| 欧美最猛性xxxx| 国内外免费激情视频| 国产又粗又爽又黄的视频| 国产欧美久久久久久| 99在线免费观看视频| 国产精品99一区| 久久久久久久久久久福利| 国产精品美女久久久久久免费| 久久成人免费视频| 亚洲一区二区三区精品动漫| 无码人妻精品一区二区蜜桃网站 | 久久九九免费视频| 欧美成人第一页| 午夜精品在线观看| 日韩精品一区在线视频| 麻豆av一区二区三区| 97精品伊人久久久大香线蕉| www.色综合| 欧美激情a∨在线视频播放| 色综合666| 蜜臀av无码一区二区三区| 99国产在线视频| 国产成人精品视频在线| 久久国产精品久久精品| 日日噜噜夜夜狠狠久久丁香五月| 欧美中文字幕在线观看视频| 国产日韩在线看片| 国产第一页视频| 欧美成人精品在线播放| 日产精品久久久一区二区| 麻豆一区区三区四区产品精品蜜桃| 99视频日韩| 久久精品视频网站| 亚洲国产欧洲综合997久久| 欧美 日韩 国产一区| 久在线观看视频| 久久国产精品影视| 欧洲日本亚洲国产区| 不卡一卡2卡3卡4卡精品在| 国产精品日韩欧美一区二区三区 | 日韩av123| 国产香蕉一区二区三区| 国产成人精品视频ⅴa片软件竹菊| 国产精品爽爽ⅴa在线观看| 亚洲激情一区二区三区| 男女猛烈激情xx00免费视频| 91九色蝌蚪国产| 久久成人免费视频| 欧美精品在线一区| 国产成人亚洲综合无码| 在线观看一区二区三区三州| 蜜桃久久精品乱码一区二区| 色777狠狠综合秋免鲁丝| 久99久在线视频| 欧美 国产 精品| 久久精品免费一区二区| 亚洲一区二区高清视频| 狠狠干一区二区| 日韩在线视频中文字幕| 视频一区二区视频| 99精品一区二区三区的区别| 色综合天天狠天天透天天伊人| 欧美高清性xxxxhdvideosex| 久久国产精品视频在线观看| 亚洲va欧美va国产综合久久| y111111国产精品久久婷婷| 欧美精品免费播放| 精品网站在线看| 久久久久久国产精品一区| 亚洲a∨一区二区三区| www.浪潮av.com| 一区二区日本伦理| 丰满少妇久久久| 一区二区三区视频在线播放| 国产欧亚日韩视频| 久久亚洲成人精品| 国产亚洲欧美一区二区三区| 国产精品精品一区二区三区午夜版| 日本欧美色综合网站免费| 久久久天堂国产精品女人| 无码中文字幕色专区| 91精品国产综合久久久久久丝袜| 亚洲最大福利网站| 国产精品稀缺呦系列在线| 欧美xxxx14xxxxx性爽| 国产欧美一区二区三区久久人妖| 蜜臀久久99精品久久久久久宅男| 国产深夜男女无套内射| 精品国产一二| 国产精品一区专区欧美日韩| 一区二区日本伦理| 91精品国产91久久| 日韩欧美视频一区二区 | 亚洲日本理论电影| 久久精品第九区免费观看| 欧美一级大片视频| 久久精品在线视频| 国产在线精品成人一区二区三区| 精品国产第一页| www.久久草| 日韩av影视| 国产精品露脸自拍| 国产老熟妇精品观看| 亚洲一区二区在线观| 国产成人精品免高潮费视频 | 国产精品一区在线免费观看| 欧美一区二区视频97| 国产不卡av在线免费观看| 含羞草久久爱69一区| 久久夜色精品国产| 成人免费毛片网| 日本精品久久电影| 国产精品初高中精品久久| 99久久99| 欧美中日韩在线| 萌白酱国产一区二区| 久久伊人一区二区| 精品一区二区国产| 亚洲一区二区在线看| 精品国产一区久久久| 国产日韩欧美自拍| 日本女人高潮视频| 欧美精品一区二区免费| 久久天天狠狠| 黄色免费高清视频| 丁香色欲久久久久久综合网| 国产精品久久..4399| 久久婷婷开心| 国产日韩欧美在线观看| 日本电影一区二区三区| 欧美成人精品一区| 久久久噜噜噜久噜久久| 国产精品一区二区三区久久| 日韩精品一区二区免费| 亚洲影院色在线观看免费| 国产精品嫩草视频| 久久久国内精品| 国产精品夜间视频香蕉| 精品日本一区二区| 亚洲精品日产aⅴ| 国产精品极品在线| 久久久久免费精品| 国产精品2018| 成人欧美一区二区三区黑人免费| 欧美日韩国产不卡在线看| 午夜精品一区二区三区四区| 精品中文字幕在线2019| 久久综合伊人77777蜜臀| 久久艳妇乳肉豪妇荡乳av| 国产精品一区视频| 精品人妻少妇一区二区| 日韩久久一级片| 日本午夜人人精品| 都市激情久久久久久久久久久 | 久久深夜福利免费观看| 国产成人高潮免费观看精品 | 2019日韩中文字幕mv| 国产剧情久久久久久| 免费99视频| 欧美日韩大片一区二区三区| 婷婷久久青草热一区二区| 中文字幕一区二区三区四区五区 | 国产成人精品网站| 久草一区二区| 国产二区视频在线播放| 91精品国产自产在线老师啪| www.男人天堂网| 国产精品一区二区三区在线| 国产久一一精品| 蜜臀久久99精品久久久酒店新书| 日韩欧美视频网站| 亚洲精品偷拍视频| 亚洲高清视频一区| 亚洲电影一二三区| 亚洲v国产v| 少妇人妻互换不带套| 午夜精品区一区二区三| 亚洲欧洲三级| 天堂av在线中文| 亚洲aa中文字幕| 日韩在线电影一区| 日韩国产精品一区二区| 欧洲精品久久久| 欧美成人精品免费| 国内精品久久久久久中文字幕| 黄色网zhan| 国模吧一区二区三区| 国产无限制自拍| 国产另类第一区|