当handle_调用返回noreply时,erlang otp gen_服务器将断开连接

当handle_调用返回noreply时,erlang otp gen_服务器将断开连接,erlang,otp,Erlang,Otp,使用以下命令启动服务器: erlc server.erl ; erl -eval 'server:start()' 在另一个终端: telnet localhost 3547 这可能会成功建立连接,但几秒钟后,由于我之外的原因,服务器关闭了连接。也允许读取,{noreply,NewState}的文档 有人能解释一下吗?我感到非常困惑 源代码 -module(server). -mode(compile). -behavior(gen_server). -compile(export_all

使用以下命令启动服务器:

erlc server.erl ; erl -eval 'server:start()'
在另一个终端:

telnet localhost 3547
这可能会成功建立连接,但几秒钟后,由于我之外的原因,服务器关闭了连接。也允许读取,
{noreply,NewState}
的文档

有人能解释一下吗?我感到非常困惑

源代码

-module(server).
-mode(compile).
-behavior(gen_server).

-compile(export_all).

-export([   main/1
          , start/0
          , stop/0
          , stop_and_wait/0
        ]).
-export([init/1, handle_call/3, handle_cast/2, handle_info/2,
        terminate/2, code_change/3]).

-define(SERVER, ?MODULE).
-define(DEFAULT_PORT, 3547).

-record(state, {port, lsock}).

start_link(Port) ->
  gen_server:start_link({local, ?SERVER}, ?MODULE, [Port], []).

start() ->
  start_link(?DEFAULT_PORT).

stop() ->
  gen_server:cast(?SERVER, stop).

stop_and_wait() ->
  gen_server:call(?SERVER, stop, infinity).

init([Port]) ->
  {ok, LSock} = gen_tcp:listen(Port, [{active, false}, {reuseaddr, true}]),
  {ok, #state{port = Port, lsock = LSock}, 0}.

handle_call({do_stuff, Arg}, _From, State) ->
  io:format("do_stuff is called with ~p~n", [Arg]),
  % {reply, ok, State};
  {noreply, State};

handle_call(stop, _From, State) ->
  {stop, normal, ok, State}.

handle_cast(stop, State) ->
  {stop, normal, State}.

handle_info(timeout, #state{lsock = LSock} = State) ->
  Server = self(),
  Listener = spawn(fun() -> listen(Server, LSock) end),
  {noreply, State}.

terminate(_Reason, _State) ->
  ok.

code_change(_Oldvsn, State, _Extra) ->
  {ok, State}.

listen(Server, LSock) ->
  {ok, Socket} = gen_tcp:accept(LSock),
  gen_server:call(?SERVER, {do_stuff, 1}),
  listen(Server, LSock).

main(_) ->
  io:format("~p~n", [ok]),
  ok.

genu服务器返回
{noreply,NewState}
:handle\u call/3
实现是允许的,但这并不意味着
genu服务器不必回复调用。相反,在这种情况下,假定
genu服务器
将在稍后使用
genu服务器:reply/2
调用进行应答


genu server:call/2,3
的默认超时为5秒。代码中发生的事情是,
listen/2
函数在接受套接字的进程中运行,因此它是该套接字的所有者,然后调用
gen\u server:call(?server,{do\u stuff,1})
。由于您的
gen_服务器
没有回复该呼叫,因此
gen_服务器:call
在5秒后超时,终止进程并关闭套接字。

非常感谢。我找到了相应的文件。