1. 主动socket和被动socket
一般来说,使用socket函数创建的socket默认是主动socket,这意味着一个主动的socket可以调用connect跟一个被动socket建立一个连接,对主动socket来说,这叫主动打开。
被动socket是一个通过调用listen函数监听要发起连接的socket,当被动socket接受一个连接通常称为被动打开。
在大多数网络程序中,服务端会作为被动socket被动接受连接,而客户端会作为主动socket主动发起连接。
服务端通过socket函数创建的socket是主动socket,而listen函数就是把这个还未接受连接的主动socket转换为被动socket
,因为服务端只需要被动接受客户端的连接请求。
2. listen函数
从listen函数的字面意思来看是用来监听客户端的连接,但是从接受连接来看,我们知道listen函数的本质其实是用来将主动socket转换为被动socket,同时限制服务端同一时刻所能接受客户端连接请求的个数。
另外, 如果套接字 sockfd 没有显示调用bind函数绑定指定的套接字地址的话,listen函数会选择本地ip地址,并随机选择一个端口号绑定到 sockfd上,但是一般作为服务器程序,通常会显示调用 bind 函数。
当调用listen函数时,会让套接字从CLOSED状态转换为LISTEN状态。
函数原型:
#include <sys/socket.h>
int listen(int sockfd, int backlog);
参数说明:
sockfd: 表示要设置的服务端套接字文件描述符
backlog: 表示要设置服务端套接字的未决连接队列的大小(注意:这个大小指的是同时建立三次握手连接的个数,包括已完成队列和未完成队列中的连接)。
返回值:成功返回0,失败返回- 1。
3. 关于backlog大小
典型的服务器程序可以同时服务于多个客户端,服务器调用listen函数声明sockfd处于监听状态,并且最多允许有backlog个tcp连接,当有客户端发起连接时,服务器调用的accept()接受这个连接。
(图片来自《linux/unix系统编程手册》)
但是客户端可能会在服务器调用accept之前调用connect,这种情况是有可能发生的,如果此时服务端可能正忙于处理其他客户端,这将产生一个未决连接。系统内核有一个未决连接队列会记录所有未决连接的信息,这样服务器在后面调用accept时就能够处理这些未决连接了,而backlog参数就是用来限制这种未决连接数量的。
如果未决连接队列已经满了,当接收到更多的连接请求就会忽略
,也就是说客户端调用connect函数可能会阻塞,直到未决连接队列中的未决连接被accept为止(注意:当一个连接被accept时就会从未决连接队列删除,此时未决连队列有空位了)。
关于backlog参数和未决连接队列,我们将在后面学习connect函数时再来详细介绍。