Erlang gen_server eaddr 在 16358 gen_tcp:connect 次调用后无效
Erlang gen_server eaddrnotavail after 16358 gen_tcp:connect calls
我正在用Erlang写一个服务器,想打开大量的连接。问题是我在 16358 gen_tcp:connect/3
次调用后收到 {error, eaddrnotavail}。如下所示,服务器非常简单。
我修改了我的 MacOS Yosemite 限制; ulimit -n
输出为 6553600。
我开始于:
erl +Q 134217727 +P 1000000 -env ERL_MAX_PORTS 40960000 -env ERTS_MAX_PORTS 40960000
服务器:
-module(naive_server).
-author("Stefan Stan").
-compile(export_all).
start_server(Port) ->
Pid = spawn_link(fun() ->
{ok, Listen} = gen_tcp:listen(Port, [binary, {backlog, 6553600}]),
spawn(fun() -> acceptor(Listen, 0) end),
timer:sleep(infinity)
end),
{ok, Pid}.
acceptor(ListenSocket, Nr) ->
{ok, Socket} = gen_tcp:accept(ListenSocket),
io:format("Client nr ~p connected~n", [Nr]),
spawn(fun() -> acceptor(ListenSocket,Nr+1) end),
handle(Socket).
handle(Socket) ->
receive
{tcp, Socket, <<"quit", _/binary>>} ->
gen_tcp:close(Socket);
{tcp, Socket, Msg} ->
gen_tcp:send(Socket, Msg),
handle(Socket)
end.
客户:
-module(naive_client).
-author("Stefan Stan").
%% API
-export([connect_clients/3, connect/1]).
connect(Port) ->
gen_tcp:connect({127,0,0,1}, Port, []).
connect_clients(Number, Port, List) when is_number(Number), Number >= 0, is_number(Port), Port>0 ->
F =
fun() ->
case Number of
0 -> {ok, lists:reverse(List)};
_ ->
{ok, Pid} = connect(Port),
NewList = [Pid|List],
connect_clients(Number-1, Port, NewList)
end
end,
spawn(F).
我认为,您可能有 运行 个临时传出端口(由您的客户使用)。你能运行:
macosx$ sysctl net.inet.ip.portrange.first net.inet.ip.portrange.last
.
在我的机器上它给我:
net.inet.ip.portrange.first: 49152
net.inet.ip.portrange.last: 65535
相差16383,和你的数字很接近
我正在用Erlang写一个服务器,想打开大量的连接。问题是我在 16358 gen_tcp:connect/3
次调用后收到 {error, eaddrnotavail}。如下所示,服务器非常简单。
我修改了我的 MacOS Yosemite 限制; ulimit -n
输出为 6553600。
我开始于:
erl +Q 134217727 +P 1000000 -env ERL_MAX_PORTS 40960000 -env ERTS_MAX_PORTS 40960000
服务器:
-module(naive_server).
-author("Stefan Stan").
-compile(export_all).
start_server(Port) ->
Pid = spawn_link(fun() ->
{ok, Listen} = gen_tcp:listen(Port, [binary, {backlog, 6553600}]),
spawn(fun() -> acceptor(Listen, 0) end),
timer:sleep(infinity)
end),
{ok, Pid}.
acceptor(ListenSocket, Nr) ->
{ok, Socket} = gen_tcp:accept(ListenSocket),
io:format("Client nr ~p connected~n", [Nr]),
spawn(fun() -> acceptor(ListenSocket,Nr+1) end),
handle(Socket).
handle(Socket) ->
receive
{tcp, Socket, <<"quit", _/binary>>} ->
gen_tcp:close(Socket);
{tcp, Socket, Msg} ->
gen_tcp:send(Socket, Msg),
handle(Socket)
end.
客户:
-module(naive_client).
-author("Stefan Stan").
%% API
-export([connect_clients/3, connect/1]).
connect(Port) ->
gen_tcp:connect({127,0,0,1}, Port, []).
connect_clients(Number, Port, List) when is_number(Number), Number >= 0, is_number(Port), Port>0 ->
F =
fun() ->
case Number of
0 -> {ok, lists:reverse(List)};
_ ->
{ok, Pid} = connect(Port),
NewList = [Pid|List],
connect_clients(Number-1, Port, NewList)
end
end,
spawn(F).
我认为,您可能有 运行 个临时传出端口(由您的客户使用)。你能运行:
macosx$ sysctl net.inet.ip.portrange.first net.inet.ip.portrange.last
.
在我的机器上它给我:
net.inet.ip.portrange.first: 49152
net.inet.ip.portrange.last: 65535
相差16383,和你的数字很接近