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

當(dāng)前位置:首頁(yè) > 網(wǎng)站舊欄目 > 學(xué)習(xí)園地 > 設(shè)計(jì)軟件教程 > gen_server tasting 之超簡(jiǎn)單名稱服務(wù)(續(xù))

gen_server tasting 之超簡(jiǎn)單名稱服務(wù)(續(xù))
2010-01-13 23:16:25  作者:  來(lái)源:
    前幾天寫了篇《gen_server tasting 之超簡(jiǎn)單名稱服務(wù) 》東西,親身體驗(yàn)了 erlang otp 的強(qiáng)悍威力。不過(guò)正所謂“超簡(jiǎn)單”,那個(gè)版本還是很初級(jí)的,所以這兩天邊繼續(xù)研究邊動(dòng)手,開(kāi)發(fā)迭代版本的名稱服務(wù)。

 

在這個(gè)版本中,需要提供如下功能:

 

  1. 使用 otp 的 supervisor 監(jiān)控樹,保證服務(wù)可靠性。
  2. 添加日志功能,通過(guò)定制 sasl alarm_handler 來(lái)記錄警告事件。
  3. 將名稱服務(wù)打包為 application,暫且叫 vsns 吧,very stabilization name server 呵呵。
  4. 開(kāi)放 socket 服務(wù) (使用半阻塞的混合模式),使用 vsns://verb /param 自定義協(xié)議對(duì)外提供訪問(wèn)支持。

最終驗(yàn)證性的功能測(cè)試用例如下,主要的測(cè)試代碼位于 test/0 方法中,其上的幾個(gè)方法都用于 socket 通信:

 

Erlang代碼 復(fù)制代碼
  1. -module(vsns_tcp_client).   
  2.   
  3. -author(lzy).   
  4. -email(lzy.dev@gmail.com).   
  5. -date("2009.02.06").   
  6. -vsn(0.11).   
  7.   
  8. -compile(export_all).   
  9.   
  10. conn() ->   
  11.     {ok, Socket} = gen_tcp:connect("localhost"8304,   
  12.         [binary, {packet, 2}, {reuseaddr, true}, {active, once}]),   
  13.     Socket.   
  14.   
  15. eval(Socket, Args, AssertVal) ->   
  16.     ok = gen_tcp:send(Socket, Args),   
  17.     receive   
  18.         {tcp, _, AssertVal} ->   
  19.             io:format("Ok. ~p = ~p.~n", [Args, AssertVal]);   
  20.         {tcp_closed, _} ->   
  21.             case Args of   
  22.                 <<"vsns://kernel_oops">> ->   
  23.                     io:format("Ok. kernel_oops = tcp_closed.~n");   
  24.                 _Other ->   
  25.                     io:format("Connection abort by server.~n")   
  26.             end;   
  27.         Other  ->   
  28.             io:format("Assert faild. ~p != ~p.~n", [Other, AssertVal])   
  29.     end,   
  30.     inet:setopts(Socket, [{active, once}]).   
  31.   
  32. close(Socket) ->   
  33.     gen_tcp:close(Socket).   
  34.   
  35. test() ->   
  36.     S = conn(),   
  37.   
  38.     eval(S, <<"vsns://remove_all">>, <<"ack">>),   
  39.   
  40.     eval(S, <<"vsns://save/abc/123">>, <<"">>),   
  41.     eval(S, <<"vsns://save/abc/456">>, <<"123">>),   
  42.     eval(S, <<"vsns://save/abc/789">>, <<"456">>),   
  43.   
  44.     eval(S, <<"vsns://load_all">>, <<"ack">>),   
  45.   
  46.     eval(S, <<"vsns://remove/abc">>, <<"789">>),   
  47.     eval(S, <<"vsns://remove/not_value">>, <<"">>),   
  48.   
  49.     eval(S, <<"foo">>, <<"unknow">>),   
  50.   
  51.     eval(S, <<"vsns://kernel_oops">>, <<"">>),   
  52.   
  53.     ok = close(S),   
  54.   
  55.     pass.   
  56.   
  57. %% File end.  

 

          實(shí)際實(shí)現(xiàn) supervisor 監(jiān)控樹、日志和警告事件功能的過(guò)程,也是學(xué)習(xí) 《Erlang 程序設(shè)計(jì)》的過(guò)程。

 

          首先,為名稱服務(wù)添加監(jiān)控進(jìn)程。erlang otp 監(jiān)控樹很簡(jiǎn)單,只需要實(shí)現(xiàn)一個(gè) supervisor behaviour module 提供給 otp supervisor 模塊就可以,前面版本的名稱服務(wù)是通過(guò) erlang shell 啟動(dòng)的,在以后將由這個(gè)監(jiān)控進(jìn)程來(lái)啟動(dòng)她,主要的啟動(dòng)代碼在 init/1 方法中,監(jiān)控模塊代碼如下:

 

Erlang代碼 復(fù)制代碼
  1. -module(name_server_sup).   
  2.   
  3. -author(lzy).   
  4. -email(lzy.dev@gmail.com).   
  5. -date("2009.02.04").   
  6. -vsn(0.1).   
  7.   
  8. -behaviour(supervisor).   
  9.   
  10. %% gen_supervisor behaviour callback functions.   
  11. -export([init/1]).   
  12.   
  13. %% Interface functions.   
  14. -export([start/0, start_in_shell/0, start_link/1]).   
  15.   
  16. start() ->   
  17.     spawn(fun() -> supervisor:start_link({local, ?MODULE}, ?MODULE, _Arg = []) end).   
  18.   
  19. start_in_shell() ->   
  20.     {ok, Pid} = supervisor:start_link({local, ?MODULE}, ?MODULE, _Arg = []),   
  21.     unlink(Pid).   
  22.   
  23. start_link(Args) ->   
  24.     supervisor:start_link({local, ?MODULE}, ?MODULE, Args).   
  25.   
  26. init([]) ->   
  27.     gen_event:swap_handler(alarm_handler, {alarm_handler, swap}, {vsns_alarm_handler, foo}),   
  28.   
  29.     {ok, {   
  30.             {one_for_one, 310},   
  31.             [{   
  32.                 vsns_name_server,   
  33.                 {name_server, start_link, []},   
  34.                 permanent,   
  35.                 1,   
  36.                 worker,   
  37.                 [name_server]   
  38.             }]         
  39.     }}.   
  40.   
  41. %% File end.  

 

          有了這個(gè) name_server_sup 就不怕 name_server 崩潰了,supervisor 進(jìn)程會(huì)負(fù)責(zé)重新啟動(dòng),對(duì)于描述監(jiān)控策略的數(shù)據(jù)結(jié)構(gòu)可參考 erlang doc。其中的 vsns_alarm_handler 是定制的警告事件處理模塊,負(fù)責(zé)將服務(wù)中的報(bào)警記錄到 erlang sasl 日志中,后期可以使用 rb 工具來(lái)查看處理。接下來(lái)就是警告日志處理模塊代碼:

 

Erlang代碼 復(fù)制代碼
  1. -module(vsns_alarm_handler).   
  2.   
  3. -author(lzy).   
  4. -email(lzy.dev@gmail.com).   
  5. -date("2009.02.04").   
  6. -vsn(0.11).   
  7.   
  8. -behaviour(gen_event).   
  9.   
  10. %% gen_event behaviour callback functions.   
  11. -export([init/1, handle_event/2, handle_call/2, handle_info/2, terminate/2, code_change/3]).   
  12.   
  13. init(Args) ->   
  14.     io:format("vsns_alarm_handler init : ~p.~n", [Args]),   
  15.     {ok, Args}.   
  16.   
  17. handle_event({set_alarm, {remove_all, From}}, _State) ->   
  18.     error_logger:error_msg("vsns depot clear by ~p started.~n.", [From]),   
  19.     {ok, _State};   
  20.   
  21. handle_event({clear_alarm, {remove_all, From}}, _State) ->   
  22.     error_logger:error_msg("vsns depot clear by ~p done.~n.", [From]),   
  23.     {ok, _State};   
  24.   
  25. handle_event(Event, State) ->   
  26.     error_logger:error_msg("unmatched event: ~p.~n", [Event, State]),   
  27.     {ok, State}.   
  28.   
  29. handle_call(_Req, State) ->   
  30.     {ok, State, State}.   
  31.        
  32. handle_info(_Info, State) ->   
  33.     {ok, State}.   
  34.   
  35. terminate(_Reason, _State) ->   
  36.     ok.   
  37.   
  38. code_change(_OldVsn, State, _Extra) ->   
  39.     {ok, State}.   
  40.   
  41. %% File end.  
 

          歸根到底,就是通過(guò) error_logger:error_msg 調(diào)用來(lái)記錄日志。當(dāng)然還涉及到 erlang sasl 的配置:

 

Config代碼 復(fù)制代碼
  1. %% file name: sasl_log.config   
  2. %% auther: lzy   
  3. %% email: lzy.dev@gmail.com   
  4. %% date: 2009.02.04  
  5. %% version: 0.1  
  6.   
  7. [{sasl, [   
  8.     {sasl_error_logger, false},    
  9.     {errlog_type, error},   
  10.     {error_logger_mf_dir, "./logs"},   
  11.     %% 10M per log file.   
  12.     {error_logger_mf_maxbytes, 1048760},   
  13.     {error_logger_mf_maxfiles, 5}   
  14. ]}].   
  15.   
  16. %% File end.  
 

          該配置文件可以通過(guò) erlang shell 的 啟動(dòng)啟動(dòng)參數(shù)指定。-boot start_sasl -config .\sasl_log。再接下來(lái)就是打包 vsns application,這需要一個(gè) application 描述文件和一個(gè) application behavior 模塊,很簡(jiǎn)單具體配置參數(shù)語(yǔ)意可參考 erlang doc。

 

App代碼 復(fù)制代碼
  1. %% file name: vsns.app   
  2. %% auther: lzy   
  3. %% email: lzy.dev@gmail.com   
  4. %% date: 2009.02.05  
  5. %% version: 0.1  
  6.   
  7. {   
  8.     application, vsns,   
  9.     [   
  10.         {description, "very stabilization name service."},   
  11.         {vsn, "1.0a"},   
  12.         {modules, [vsns_app, vsns_supervisor, name_server, vsns_alarm_handler]},   
  13.         {registered, [vsns_supervisor, name_server]},   
  14.         {applications, [kernel, stdlib]},   
  15.         {mod, {vsns_app, []}},   
  16.         {start_phases, []}   
  17.     ]   
  18. }.   
  19.   
  20. %% File end.  
 
Erlang代碼 復(fù)制代碼
  1. -module(vsns_app).   
  2.   
  3. -author(lzy).   
  4. -email(lzy.dev@gmail.com).   
  5. -date("2009.02.05").   
  6. -vsn(0.1).   
  7.   
  8. -behavior(application).   
  9.   
  10. -export([start/2, stop/1]).   
  11.   
  12. start(_Type, Args) ->   
  13.     name_server_sup:start_link(Args).   
  14.   
  15. stop(_State) ->   
  16.     void.   
  17.   
  18. %% File end.  

 

          經(jīng)過(guò)這樣的包裝,就可以通過(guò) application:start(vsns) 調(diào)用來(lái)啟動(dòng) vsns 服務(wù)。通過(guò) appmon 工具可以看到如下進(jìn)程樹:

 

vsns 進(jìn)程樹

 

到這里,我們就可以通過(guò) erlang 來(lái)使用 vsns 了。

 

Erlang代碼 復(fù)制代碼
  1. C:\Program Files\erl5.6.4\usr\lzy_app\vsns>..\..\..\bin\erl.exe -sname vsns +P 1  
  2. 02400 -smp enable +S 1 -boot start_sasl -config sasl_log   
  3. Eshell V5.6.4  (abort with ^G)   
  4. (vsns@srclzy)1> application:start(vsns).   
  5. vsns_alarm_handler init : {foo,{alarm_handler,[]}}.   
  6. name_server starting.   
  7. ok   
  8. (vsns@srclzy)2> name_server:save(abc, 123).   
  9. undefined   
  10. (vsns@srclzy)3> name_server:load_all().   
  11. [{abc,123}]  
 

          最后還需要一個(gè) socket tcp 服務(wù)器,來(lái)將 vsns 暴露出來(lái),允許其它 client 來(lái)使用服務(wù)。otp 中沒(méi)有類似的 socket server behavior,但可以通過(guò) gen_server 來(lái)實(shí)現(xiàn),當(dāng)然甚至可以實(shí)現(xiàn)一個(gè)非 otp 相關(guān)的 socket 服務(wù)器。這里 Serge Aleynikov 實(shí)現(xiàn)了一個(gè)很好 tcp 服務(wù)器,基于有限狀態(tài)機(jī)模式來(lái)處理請(qǐng)求,在此做了很好的闡述:Building a Non-blocking TCP server using OTP principles ,不過(guò)恐怕需要代理來(lái)打開(kāi)連接。在他給出的代碼中,我添加了幾行代碼,將 socket server 提供的服務(wù)是做為可配置的,通過(guò) application 環(huán)境來(lái)配置 socket server 使用的 gen_fsm behaviour module,大約位于 tcp_server_app 模塊的 15 和 27 行。

 

Erlang代碼 復(fù)制代碼
  1. -module(tcp_server_app).   
  2.   
  3. ... ...   
  4.   
  5. -define(DEF_SERVICE, tcp_echo_fsm).   
  6.   
  7. ... ...   
  8.   
  9. start(_Type,  _Args) ->   
  10.     ListenPort = get_app_env(listen_port, ?DEF_PORT),    
  11.     ServiceMod = get_app_env(service_mod, ?DEF_SERVICE),    
  12.     supervisor:start_link({local, ?MODULE}, ?MODULE, [ListenPort, ServiceMod]).   
  13.   
  14. ... ...  
 

          在 saleyn_tcp_server 中提供的是 echo 服務(wù)。為了將 saleyn_tcp_server 服務(wù)指定成 vsns,除了上面的修改外,剩下就只需要實(shí)現(xiàn)一個(gè)調(diào)用 vsns 的 gen_fsm behaviour module 了,代碼很簡(jiǎn)單,是基于 tcp_echo_fsm 修改得來(lái)的,呵呵。

 

Erlang代碼 復(fù)制代碼
  1. -module(vsns_tcp_fsm).   
  2.   
  3. -author(lzy).   
  4. -email(lzy.dev@gmail.com).   
  5. -date("2009.02.06").   
  6. -vsn(0.1).   
  7. -remark("vsns_tcp_fsm used by saleyn_tcp_server appliction to support vsns socket server.").   
  8. -remark("It referenced from saleyn_tcp_server/tcp_echo_fsm module.").   
  9.   
  10. -behaviour(gen_fsm).   
  11.   
  12. -export([start_link/0, set_socket/2]).   
  13.   
  14. %% gen_fsm callbacks   
  15. -export([init/1, handle_event/3,   
  16.          handle_sync_event/4, handle_info/3, terminate/3, code_change/4]).   
  17.   
  18. %% FSM States   
  19. -export([   
  20.     'WAIT_FOR_SOCKET'/2,   
  21.     'WAIT_FOR_DATA'/2  
  22. ]).   
  23.   
  24. -record(state, {   
  25.                 socket,    % client socket   
  26.                 addr       % client address   
  27.                }).   
  28.   
  29. -define(TIMEOUT, 120000).   
  30.   
  31. %%%------------------------------------------------------------------------   
  32. %%% API   
  33. %%%------------------------------------------------------------------------   
  34.   
  35. %%-------------------------------------------------------------------------   
  36. %% @spec (Socket) -> {ok,Pid} | ignore | {error,Error}   
  37. %% @doc To be called by the supervisor in order to start the server.   
  38. %%      If init/1 fails with Reason, the function returns {error,Reason}.   
  39. %%      If init/1 returns {stop,Reason} or ignore, the process is   
  40. %%      terminated and the function returns {error,Reason} or ignore,   
  41. %%      respectively.   
  42. %% @end   
  43. %%-------------------------------------------------------------------------   
  44. start_link() ->   
  45.     gen_fsm:start_link(?MODULE, [], []).   
  46.   
  47. set_socket(Pid, Socket) when is_pid(Pid), is_port(Socket) ->   
  48.     gen_fsm:send_event(Pid, {socket_ready, Socket}).   
  49.   
  50. %%%------------------------------------------------------------------------   
  51. %%% Callback functions from gen_server   
  52. %%%------------------------------------------------------------------------   
  53.   
  54. %%-------------------------------------------------------------------------   
  55. %% Func: init/1  
  56. %% Returns: {ok, StateName, StateData}          |   
  57. %%          {ok, StateName, StateData, Timeout} |   
  58. %%          ignore                              |   
  59. %%          {stop, StopReason}   
  60. %% @private   
  61. %%-------------------------------------------------------------------------   
  62. init([]) ->   
  63.     process_flag(trap_exit, true),   
  64.     {ok, 'WAIT_FOR_SOCKET', #state{}}.   
  65.   
  66. %%-------------------------------------------------------------------------   
  67. %% Func: StateName/2  
  68. %% Returns: {next_state, NextStateName, NextStateData}          |   
  69. %%          {next_state, NextStateName, NextStateData, Timeout} |   
  70. %%          {stop, Reason, NewStateData}   
  71. %% @private   
  72. %%-------------------------------------------------------------------------   
  73. 'WAIT_FOR_SOCKET'({socket_ready, Socket}, State) when is_port(Socket) ->   
  74.     % Now we own the socket   
  75.     inet:setopts(Socket, [binary, {packet, 2}, {reuseaddr, true}, {active, once}]),   
  76.     {ok, {IP, _Port}} = inet:peername(Socket),   
  77.     {next_state, 'WAIT_FOR_DATA', State#state{socket=Socket, addr=IP}, ?TIMEOUT};   
  78.   
  79. 'WAIT_FOR_SOCKET'(Other, State) ->   
  80.     error_logger:error_msg("State: 'WAIT_FOR_SOCKET'. Unexpected message: ~p\n", [Other]),   
  81.     %% Allow to receive async messages   
  82.     {next_state, 'WAIT_FOR_SOCKET', State}.   
  83.   
  84. %% Notification event coming from client   
  85. 'WAIT_FOR_DATA'({data, Data}, #state{socket=S} = State) ->   
  86.     ok = handle_data(S, string:tokens(binary_to_list(Data), "/")),   
  87.     inet:setopts(S, [{active, once}]),   
  88.     {next_state, 'WAIT_FOR_DATA', State, ?TIMEOUT};   
  89.   
  90. 'WAIT_FOR_DATA'(timeout, State) ->   
  91.     error_logger:error_msg("~p Client connection timeout - closing.\n", [self()]),   
  92.     {stop, normal, State};   
  93.   
  94. 'WAIT_FOR_DATA'(Data, State) ->   
  95.     io:format("~p Ignoring data: ~p\n", [self(), Data]),   
  96.     {next_state, 'WAIT_FOR_DATA', State, ?TIMEOUT}.   
  97.   
  98. %%-------------------------------------------------------------------------   
  99. %% Func: handle_event/3  
  100. %% Returns: {next_state, NextStateName, NextStateData}          |   
  101. %%          {next_state, NextStateName, NextStateData, Timeout} |   
  102. %%          {stop, Reason, NewStateData}   
  103. %% @private   
  104. %%-------------------------------------------------------------------------   
  105. handle_event(Event, StateName, StateData) ->   
  106.     {stop, {StateName, undefined_event, Event}, StateData}.   
  107.   
  108. %%-------------------------------------------------------------------------   
  109. %% Func: handle_sync_event/4  
  110. %% Returns: {next_state, NextStateName, NextStateData}            |   
  111. %%          {next_state, NextStateName, NextStateData, Timeout}   |   
  112. %%          {reply, Reply, NextStateName, NextStateData}          |   
  113. %%          {reply, Reply, NextStateName, NextStateData, Timeout} |   
  114. %%          {stop, Reason, NewStateData}                          |   
  115. %%          {stop, Reason, Reply, NewStateData}   
  116. %% @private   
  117. %%-------------------------------------------------------------------------   
  118. handle_sync_event(Event, _From, StateName, StateData) ->   
  119.     {stop, {StateName, undefined_event, Event}, StateData}.   
  120.   
  121. %%-------------------------------------------------------------------------   
  122. %% Func: handle_info/3  
  123. %% Returns: {next_state, NextStateName, NextStateData}          |   
  124. %%          {next_state, NextStateName, NextStateData, Timeout} |   
  125. %%          {stop, Reason, NewStateData}   
  126. %% @private   
  127. %%-------------------------------------------------------------------------   
  128. handle_info({tcp, Socket, Bin}, StateName, #state{socket=Socket} = StateData) ->   
  129.     % Flow control: enable forwarding of next TCP message   
  130.     inet:setopts(Socket, [{active, once}]),   
  131.     ?MODULE:StateName({data, Bin}, StateData);   
  132.   
  133. handle_info({tcp_closed, Socket}, _StateName,   
  134.             #state{socket=Socket, addr=Addr} = StateData) ->   
  135.     error_logger:info_msg("~p Client ~p disconnected.\n", [self(), Addr]),   
  136.     {stop, normal, StateData};   
  137.   
  138. handle_info(_Info, StateName, StateData) ->   
  139.     {noreply, StateName, StateData}.   
  140.   
  141. %%-------------------------------------------------------------------------   
  142. %% Func: terminate/3  
  143. %% Purpose: Shutdown the fsm   
  144. %% Returns: any   
  145. %% @private   
  146. %%-------------------------------------------------------------------------   
  147. terminate(_Reason, _StateName, #state{socket=Socket}) ->   
  148.     (catch gen_tcp:close(Socket)),   
  149.     ok.   
  150.   
  151. %%-------------------------------------------------------------------------   
  152. %% Func: code_change/4  
  153. %% Purpose: Convert process state when code is changed   
  154. %% Returns: {ok, NewState, NewStateData}   
  155. %% @private   
  156. %%-------------------------------------------------------------------------   
  157. code_change(_OldVsn, StateName, StateData, _Extra) ->   
  158.     {ok, StateName, StateData}.   
  159.   
  160. handle_data(S, ["vsns:""save", Key, Value]) ->   
  161.     gen_tcp:send(S, list_to_binary(swap_undefined(name_server:save(Key, Value))));   
  162.   
  163. handle_data(S, ["vsns:""load", Key]) ->   
  164.     gen_tcp:send(S, list_to_binary(swap_undefined(name_server:load(Key))));   
  165.   
  166. handle_data(S, ["vsns:""load_all"]) ->   
  167.     name_server:load_all(),   
  168.     gen_tcp:send(S, <<"ack">>); % list_to_binary(name_server:load_all())   
  169.   
  170. handle_data(S, ["vsns:""remove", Key]) ->   
  171.     gen_tcp:send(S, list_to_binary(swap_undefined(name_server:remove(Key))));   
  172.   
  173. handle_data(S, ["vsns:""remove_all"]) ->   
  174.     name_server:remove_all(),   
  175.     gen_tcp:send(S, <<"ack">>); % list_to_binary(name_server:remove_all())   
  176.   
  177. handle_data(S, ["vsns:""kernel_oops"]) ->   
  178.     gen_tcp:send(S, list_to_binary(name_server:kernel_oops()));   
  179.   
  180. handle_data(S, _Data) ->   
  181.     gen_tcp:send(S, <<"unknow">>).   
  182.   
  183. swap_undefined(undefined) ->   
  184.     "";   
  185.   
  186. swap_undefined(Other) ->   
  187.     Other.   
  188.   
  189. %   File end.  
主站蜘蛛池模板: 桁架楼承板_钢筋桁架楼承板厂家-山东新材料科技 | 云南万通汽车学校【官方网站】| 直流屏|青岛直流屏|直流屏电池-世界500强艾默生直流屏标准生产厂家赛里斯能源科技有限公司 | 西安防静电地板_防静电地板厂家_防静电地板价格_OA网络地板_写字楼架空地板_机房墙板安装-红梅防静电地板厂家直销 | 液压尾管悬挂器,机械式尾管悬挂器价格,石油套管扶正器厂家,连续油管悬挂器,高压双塞水泥头,免钻塞注水泥分级箍,单塞套管水泥头价格,弹性套管扶正器,铸铝钢性扶正器,钢性套管扶正器厂家 | 日本国际高中学校招生【立学仕日本留学服务】日语中学学校|美术生|艺术生|高中生|本科|研究生|语言学校留学择校申请 | 配电箱自动生产线-配电箱生产线设备-山东炜桦智能 | 数控滑台,机床滑台,十字滑台,直线滑台,三轴滑台,立柱滑台厂家-泊头市北重机械制造有限公司 | 思达测试|山东思达高科机械设备有限公司| 乌海市腐植酸盐加工,内蒙古腐植酸钠,内蒙古型煤粘合剂生产厂家|创联腐植酸盐加工有限公司 | 指挥调度|调度系统|应急指挥调度|应急指挥|可视化调度|多媒体指挥调度|融合通信|综合调度|应急指挥系统|IP调度系统-北京瑞光极远数码科技有限公司 | 呼吸家官网|肺功能检测仪生产厂家|国产肺功能仪知名品牌|肺功能检测仪|肺功能测试仪|婴幼儿肺功能仪|弥散残气肺功能仪|肺功能测试系统|广州红象医疗科技有限公司|便携式肺功能仪|大肺功能仪|呼吸康复一体机|儿童肺功能仪|肺活量计|医用简易肺功能仪|呼吸康复系统|肺功能仪|弥散肺功能仪(大肺)|便携式肺功能检测仪|肺康复|呼吸肌力测定肺功能仪|肺功能测定仪|呼吸神经肌肉刺激仪|便携式肺功能 | 连云港机械手厂家_全自动焊接机械手_刀轴焊接机_智能轴类焊接机_连云港建博自动化设备有限公司 | 无锡今飞激光技术有限公司-手持激光焊接机_激光打标机_激光清洗机_平台激光焊接机_焊接专机- | 专业液压对辊,双齿辊破碎机,沙子烘干机,制砂洗沙设备生产线厂家 - 巩义市吉宏机械 | 深圳市桃子自动化科技有限公司-点胶机_灌胶机_焊锡机_螺丝机_SCARA机器人 | 新颖科技 - 家电及汽车外饰面板供应商| 硬脂酸钡_硬脂酸镁_硬脂酸镉_硬脂酸铝_硬脂酸锌-石家庄中石恒达矿产品有限公司 | 钠长石-硅铁-硅钙-孕育剂-球化剂-安阳李氏实业有限公司 | 碎料机_拌料机_吸料机_温控箱_烤料斗_模温机_东莞索亿塑料破碎机厂家 | 一体式电磁流量计_分体式电磁流量计_卫生级电磁流量计_卫生型电磁流量计_电池供电电磁流量计_卡箍式电磁流量计_废水电磁流量计_德克森仪表(淮安)有限公司官网 | 模具配件加工厂|东莞模具配件加工|模具配件加工厂|精密塑胶模具配件|东莞市优迪精密模具制品有限公司 | 耐磨涂料_陶瓷涂料_高温涂料_高硬度耐磨涂料-北京耐默科技 | 毛刷_毛刷辊_工业毛刷辊厂家_毛刷加工制造厂【丰汇刷业】 | 收银系统_收银机_pos收款机_门店管理系统-客如云 | 联系我们果博福布斯公司客服电话17787888880[河南河北区] | 移动石料破碎机-颚式锤式反击式破碎机设备厂家_山东.青州富康机械 | 交通标志牌-交通标牌-铝圆牌-铝三角片-铝滑槽-公路警示指示牌-方牌-高速道路反光牌毛坯-交通设施安全警示标识牌-路名指示限速限高牌-厂家加工交通标牌铝板半成品毛坯-上海吕盟铝业有限公司 | 深圳蓝枫印刷_画册印刷_彩页印刷_宣传册印刷_包装盒印刷_彩盒印刷厂_不干胶印刷厂 | 浙江日新电气有限公司 | 削片机|木材破碎机|木材粉碎机|模板破碎机|双轴撕碎机_郑州木工机械制造厂 | 家具板|橱柜衣柜家具板|木饰面定制|榉木胶合板|防水胶合板|胶合板厂家-佛山吉盛唐朝木业-大坂木业 | 长沙设计公司丨品牌策划设计服务丨画册设计-石青|长沙画册设计 | 免费建站_自助建站_网站建设_企业建站_中小企业建网站_免费网站模板-牵牛建站 | 郑州试驾场地_双超专业汽车试驾场地_郑州双超文化传播有限公司官方网站 | 企业头条—优府网企业自媒体 | 户外工程照明|洗墙灯厂家|聚翔(广东)照明科技有限公司 | 上海海外置业展_2024海外置业移民留学展_4月5-7日_企业参展处/免费领门票 | 制砂机锤头_锤式破碎机锤头_粉煤机锤头_巩义市东辰实业 | 山东健泽医疗科技有限公司-官网_深呼吸训练器_肋骨固定板_放疗定位膜耗材生产厂家招商 | 苏州温测仪器有限公司-苏州炉温跟踪仪|苏州隔热箱|苏州炉温测试仪|苏州恒温恒湿箱 |