erlang套接字

Erlang的套接字可以有三种打开模式:主动(active)、单次主动(active once)或被动(passive)。
这是通过在gen_tcp:connect(Address, Port, Options)或gen_tcp:listen(Port, Options)
的Options参数里加入{active, true | false | once}选项实现的。

如果指定{active, true}就会创建一个主动套接字,指定{active, false}则是被动套接
字。{active, once}创建的套接字只会主动接收一个消息,接收完之后必须重新启用才能接收
下一个消息。

一、主动型

当一个主动套接字被创建后,它会在收到数据时向控制进程发送{tcp, Socket, Data}
消息。控制进程无法控制这些消息流。恶意的客户端可以向系统发送成千上万的消息,
而它们都会被发往控制进程。控制进程无法阻止这些消息流。

进程无法控制通往服务器循环的消息流。如果客户端生成数据的速度快于服务器处理数
据的速度,系统就会遭受数据洪流的冲击:消息缓冲区会被塞满,系统可能会崩溃或表现异常。

{ok,Listen} = gen_tcp:listen(Port,[...,{active,true},...]),
{ok,Socket} = gen_tcp:accapt(Listen),
loop(Socket).
loop(Socket) ->
  receive
      {tcp,Socket,Data} ->
           对数据进行操作
       {tcp_closed,Socket} ->
       ....
    end.

 

二、被动型

如果一个套接字是用被动模式打开的,控制进程就必须调用gen_tcp:recv(Socket, N)
来从这个套接字接收数据。然后它会尝试从套接字接收N个字节。如果N = 0,套接字就
会返回所有可用的字节。在这个案例里,服务器可以通过选择何时调用gen_tcp:recv来
控制客户端所发的消息流

{ok,Listen} = gen_tcp:listen(Port,[...,{active,true},...]),
{ok,Socket} = gen_tcp:accapt(Listen),
loop(Socket).
loop(Socket) ->
  case gen_tcp:recv(Socket,N) of
    {Ok, B} ->
               ...对数据进行操作...
                loop(Socket);
            {error,closed}
                ...
    end.

 

三、混合型

既不是阻塞也不是非阻塞。用{active, once}选项打开套接字。套接字在这个模式下虽然是主动的,但只针对一个消息。当控制进程收到一个消
息后,必须显式调用inet:setopts才能重启下一个消息的接收,在此之前系统会处于阻塞状态。这种方法集合了前两种模式的优点

{ok,Listen} = gen_tcp:listen(Port,[...,{active,once},...]),
{ok,Socket} = gen_tcp:accapt(Listen),
loop(Socket).
loop(Socket) ->
  receive
      {tcp,Socket,Data} ->
              对数据进行操作
              准备接收下一个消息
              inet:setopts(Socket,[{active,once}]),
              loop(Socket);
       {tcp_closed,Socket} ->
       ....
    end.

 

posted @ 2015-08-07 12:53  wywdahai  阅读(493)  评论(0编辑  收藏  举报