linux网络编程常用函数说明
connect函数
int connect (int sockfd,struct sockaddr * serv_addr,int addrlen);
(1)connect之前调用bind不是必须的,内核会分配ip以及临时port;
(2)connect调用后,会发送SYN分节,如果没有收到SYN-ACK分节,则返回ETIMEOUT;
底层细节说明:发送SYN,6s后未收到SYN-ACK,则再发送一个SYN,24s后未收到SYN-ACK,则再发一个SYN。共等待75s后仍未收到响应,则ETIMEOUT;(如connect一个子网内未分配的ip)
对工程的启示:阻塞connect,在后端挂掉的情况下,75s才返回,这个过程中,有可能阻塞所有工作线程或IO线程;
(3)connect调用后,如果得到的响应是RST,则马上返回ECONNECTREFUSED;(如connect路由器的一个port)
(4)connect调用后,如果路由n跳后找不到主机,则返回EHOSTUNREACH;(如connect一个不存在的ip)
listen函数
int listen(int sockfd, int backlog);
(1)调用时机:本函数在socket与bind之后,accept之前server端调用;
(2)backlog为未完成连接队列,已完成连接队列长度之和;
底层细节说明:client端发出connect之后,SYN报文到达server时,server会在未完成队列中新增一项,标明client有连接发起,且未完成,并回SYN-ACK报文;client收到SYN-ACK后,继续发送ACK以完成三次握手,server端收到ACK后将未完成队列中的新增项已知已完成队列;
如果server调用了accept,accept会从已完成队列中取出新增项,以完成后续工作。
(3)如果server未收到SYN-ACK,会在一个超时时间后(75s),将新增项移出未完成队列;
(4)如果SYN到达,且队列都是满的,则抛弃SYN包,且不发送RST(说明只是暂时繁忙,期待client不久后重新connect可以成功)
accept函数
int accept(int listen_fd, struct sockaddr addr, socklen_t addr_len);
(1)accept从已完成队列队首取出fd并返回;
(2)已完成队列为空时则阻塞;
(3)如果不关心client的地址信息,第二个参数与第三个参数可置空。
close函数
int close(int fd);
close一个套接口,默认行为是,tcp会尝试发送“已排队”的数据,发送完毕后终止tcp连接。
一般来说,会由客户端首先发起close,向服务器发送一个FIN,服务器回ACK,此时服务器套接口进入CLOSE_WAIT,客户端进入FIN_WAIT_2;
后续服务器发送FIN,客户端回ACK,完成四步挥手,客户端进入TIME_WAIT状态。(该流程是正常终止)
设置SO_LINGER可以改变以上默认行为,直接关闭tcp连接,丢弃发送缓冲区中“已排队”的数据,并向对端发送RST,而不是通过正常的四步握手终止,从而避免TIME_WAIT状态。