0

我有以下简单的 UDP 服务器:

  • 只接受二进制 <<0:32>>,

  • 否则它会崩溃

     -module(server).
     -export([listen/1]).
    
     listen(Port) ->
         spawn(fun()->run_it(Port) end).
    
     run_it(Port) ->
     {ok, Skt} = gen_udp:open(Port, [binary]),
     loop(Skt).
    
     loop(Skt) ->
     receive
         {udp, Skt, _, _, Bin} ->
             case Bin of
                 <<0:32>> ->
                     io:fwrite("~p~n", [{"Good Format: ", Bin}]),
             loop(Skt)
             end
     end.
    

现在,我的对等 UDP 客户端将故意发送格式错误的数据。

我可以编写一个 case 子句来匹配任何消息,而忽略任何格式错误的消息。

但是,如果最终出现错误,它将无济于事。

我在某处读过:“不要防御性地编程,让它崩溃,然后修复它”。

    =ERROR REPORT==== 19-Jul-2020::21:15:29.872000 ===
    Error in process <0.93.0> with exit value:
    {{case_clause,<<0>>},[{server,loop,1,[{file,"server.erl"},{line,16}]}]}

酷,它崩溃了,但我希望我的服务器现在自动重启:-)

我读过一个名为“supervisor”的进程可以监视我的服务器并在检测到它死亡时重新启动它。

所以,我使用了“rebar3”,因为当我只用 1 行“rebar3 compile”编译多个文件时,它对我有很大帮助。

它会自动创建一个带有 3 个文件的 /src/ ,但我现在只对 2 个文件感兴趣:

  • server_app.erl
  • server_sup.erl

另外,我已经阅读了文档,但我仍然远未理解。

谁能提供建议,或者将我的 19 行代码server.erl 转换为server_app.erl并由server_sup.erl监督?

注意:我不是在寻找 gen_server,我经常看到它,但我是否也有义务将其转换为 gen_server,或者只有 application+supervisor 可以满足我的要求?

提前致谢,

最好的祝福,

4

1 回答 1

3

监督者是 OTP 监督树的一部分它处理所有重启策略等。尽管可以不将gen_servers用于其模块,但我反对它: gen_servers 为最常见的服务器操作提供了方便的抽象(它们处理名称注册系统消息和其他开箱即用的好东西)。

尽管“让它崩溃”引理在 Erlang 中很常见,但这并不意味着您不必预见代码可能面临的问题,或者您只关心幸福的情况。我不想看到任何系统由于某人的畸形/恶意而崩溃nc -u

请记住,如果达到主管的重启限制,它也会死亡,最终到达应用程序的顶级主管,这会在死亡时使 VM 崩溃。

让我们来看看代码(很少编辑rebar3 new app生成的内容):

该应用程序不需要任何版本:

-module(server_app).

-behaviour(application).
-export([start/2, stop/1]).

start(_StartType, _StartArgs) ->
    server_sup:start_link().

stop(_State) ->
    ok.

主管有更多的配置,但就是这样:

-module(server_sup).

-behaviour(supervisor).
-export([start_link/0]).
-export([init/1]).

start_link() ->
    supervisor:start_link(?MODULE, []).

init([]) ->
    SupFlags = #{strategy => one_for_all,
                 intensity => 0,
                 period => 1},
    ChildSpecs = [#{id => my_server,
                    start => {server, start_link, [12345]},
                    type => worker
                   }],
    {ok, {SupFlags, ChildSpecs}}.

并且server.erl需要在启动函数中进行一些修改:

-module(server).
-export([start_link/1]).

start_link(Port) ->
    {ok, spawn_link(fun()->run_it(Port) end)}.

run_it(Port) ->
    {ok, Skt} = gen_udp:open(Port, [binary]),
    loop(Skt).

loop(Skt) ->
    receive
        {udp, Skt, _, _, Bin} ->
            case Bin of
                <<0:32>> ->
                    io:fwrite("~p~n", [{"Good Format: ", Bin}]),
                    loop(Skt)
            end

server.erlasgen_server会是这样的:

-module(server).

-export([start_link/1]).

-behaviour(gen_server).
-export([
         init/1,
         handle_cast/2,
         handle_call/3,
         handle_info/2
        ]).

start_link(Port) ->
    gen_server:start_link(?MODULE, Port, []).

init(Port) ->
    gen_udp:open(Port, [binary]).

handle_call(_Call, _From, State) ->
    {reply, ok, State}.

handle_cast(_Msg, State) ->
    {noreply, State}.

handle_info({udp, Skt, _, _, Bin}, Skt) ->
    case Bin of
        <<0:32>> -> io:fwrite("~p~n", [{"Good Format: ", Bin}])
    end,
    {noreply, Skt};
handle_info(_Msg, State) -> % Messages that are not {udp, Skt, _, _, _} are discarded
    {noreply, State}.
于 2020-07-19T22:07:24.933 回答