创建socket

  UNIX/linux的一个哲学是:所有东西都是文件。socket就是可读、可写、可控制、可关闭的文件描述符。

  创建socket的API:

android ServerSocket监控是否断开 socket停止监听_服务器


  domain:告诉系统使用哪个底层协议。例如对于TCP/IP协议族而言,为AF_INET或AF_INET6。

  type:指定服务类型。主要有SOCK_STREAM服务(流服务,对TCP/IP协议族而言表示传输层使用TCP协议)和SOCK_UGRAM(数据报,对TCP/IP协议族而言表示传输层使用UDP协议)服务。但自Linux内核2.6.17版本起,type参数可以接受服务类型与下面俩个重要的标志相与的值:SOCK_NONBLOCK和SOCK_CLOEXEC。分别表示将新创建的socket设为非阻塞的,以及用fork调用创建子进程时在子进程中关闭该socket。在此之前需要使用额外的系统调用。

  protocol:是在前两个参数构成的协议族集合下,再选一个具体的协议。不过通常前两个参数已经完全决定了所用的协议。所以,该值设置为0。表示使用默认协议。

  socket系统调用成功时返回一个socket文件描述符,失败返回-1并设置errno。

命名socket

  创建socket时,指定了地址族,但是并未指定使用该地址族中的那个具体socket地址。将一个socket与socket地址绑定称为给socket命名。在服务器程序中,我们通常要命名socket,因为只有命名后客户端才能知道该如何连接它。客户端通常不需要命名socket,而是采用匿名方式,即使用操作系统自动分配的socket地址。

  命名的socket的系统调用是bind,定义如下:

android ServerSocket监控是否断开 socket停止监听_系统调用_02


  bind将my_addr所指的socket地址分配给未命名的sockfd文件描述符,addrlen参数指定该socket地址的长度。

  bind成功时返回0,失败则返回-1并设置errno:(常见的两种errno)

android ServerSocket监控是否断开 socket停止监听_客户端_03

监听socket

  socket被命名之后,还不能马上接收客户连接,我们需要使用如下系统调用来创建一个监听队列以存放待处理的客户连接:

android ServerSocket监控是否断开 socket停止监听_客户端_04


  sockfd参数指定被监听的socket。backlog参数提示内核监听队列的最大长度。监听队列的长度如果超过backlog,服务器将不受理新的客户连接,客户端也将受到ECONNREFUSED错误信息。(在内核版本2.2之前的Linux中,backlog参数是指所有处于半连接状态和完全连接状态的socket的上限。在此之后,他只表示处于完全连接状态的socket上限,处于半连接状态的socket的上限则由/proc/sys/net/ipv4/tcp_max_syn_backlog内核参数定义。典型值为5)。

  listen成功时返回0,失败则返回-1,并设置errno。

接受socket连接

  下面的系统调用从listen监听队列中接受一个连接:

android ServerSocket监控是否断开 socket停止监听_客户端_05


  sockfd参数是执行过listen系统调用的监听socket。addr参数用来获取被接受端socket地址,该socket地址的长度由addrlen参数指出。accept成功时返回一个新的连接socket,该socket唯一地标识了被接受的这个连接,服务器可通过读写该socket来与被接受连接对应的客户端通信。accept失败返回-1并设置errno。

  accetp只是从监听队列中取出连接,而不论连接处于何种状态,更不关心任何网络状况的变化。

socket连接和关闭

发起连接

  服务器通过listen调用来被动接受连接,客户端通过connect系统调用来主动连接服务器:

android ServerSocket监控是否断开 socket停止监听_服务器_06


  sockfd参数由socket系统调用返回一个socket。serv_addr参数是服务器监听的socket地址,addrlen参数则指定这个地址的长度。

  connect调用成功返回0。一旦成功建立连接,sockfd就唯一地标识了这个连接,客户端就可以通过读写sockfd来与服务器通信。connect失败则返回-1并设置errno,常见的两种是ECONNREFUSED和ETIMEDOUT:

  ECONNREFUSED:目标端口不存在,连接被拒绝。

  ETIMEDOUT:连接超时。

关闭连接

  关闭连接实际就是关闭该连接对应的socket,可通过close系统调用来完成:

android ServerSocket监控是否断开 socket停止监听_客户端_07


  fd参数是待关闭的socket。不过,close系统调用并非总是立即关闭一个连接,而是将fd的引用计数减一。只有当fd的引用计数为0时,才真正关闭连接。多进程程序中,一次fork系统调用默认将父进程中打开的socket的引用计数加一,因此,我们必须在父进程和子进程中都对该socket执行close调用才能将连接关闭。

  如果无论如何都要立即终止连接(而不是将socket的引用计数减一),可以使用如下的shutdown系统调用:

android ServerSocket监控是否断开 socket停止监听_客户端_08