初夏小谈:基于TCP协议的网络通信(线程池实现)

一、TCP通信网络中可能用到的接口:

1.创建套接字

int socket(int domain, int type, int protocol);

2.绑定地址信息和端口

int bind(int sockfd, const struct sockaddr *addr,socklen_t addrlen);

3.开始监听:告诉操作系统可以开始与用户进行三次握手连接;每一个客户端到来都会为它创建一个新的socket与它单独通信,为了防止而已连接攻击服务器设置backlog来限制最大并发连接数。在内核中已经创建连接队列和未完成的对列,通过最大节点数量来限制。

int listen(int sockfd, int backlog);

4.获取上一步已完成连接的客户端新建socket。

int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen);

sockfd:监听socket。仅用于接收第一次请求。在后续通信中它不再使用。与客户端进行通信。

5.收发数据

ssize_t recv(int sockfd, void *buf, size_t len, int flags);

recv返回值为0,不是没有数据,而是断开连接。

ssize_t send(int sockfd, const void *buf, size_t len, int flags);

6.客户端连接请求:

int connect(int sockfd, const struct sockaddr *addr,socklen_t addrlen);

TCP通信具体流程:

(1)服务端通信步骤:创建套接字---》绑定地址信息---》开始监听---》获取已完成连接的客户端新建socket---》收发数据---》关闭套接字。

(2)客户端通信步骤:创建套接字---》不推荐手动绑定地址信息---》向服务器发起连接请求---》收发数据---》关闭套接字。

二、TCP协议通信注意事项:

由于基本的TCP服务端程序同时只能与一个客户端进行通信:因为服务器并不知道客户端数据什么时候到来,因此服务端程序流程只能写死,而写死可能会导致阻塞

阻塞分为两点:

1.阻塞到accept的情况:当服务器一接收连接一个客户端,并与之通信,通信完成后,返回上来重新获取新的新建socket,而可能没有接收到,默认接收类型是阻塞,所以会卡在这块。 

2.阻塞到recv的情况:当服务器先接收连接到一个客户端,但客户端并未与之通信,便卡在接收的地方,此时又来了一个客户端,服务器无法与之连接,第二个客户端给服务器发送数据,便无法获取,因为它卡在第一个客户端的recv处。

所以就使用多进程/多线程来处理。

在多进程tcp服务端程序:一个子进程处理一个客户端,实现多个客户端同时处理。父进程必须关闭socket,处理僵尸子进程问题。设置信号。

在多线程tcp服务端程序:一个子线程处理一个客户端,实现多个客户端的同时处理。主线程不能退出即不能关闭socket:线程间共享代码段,数据段。同一线程共享文件描述符表。

三、代码实现:

1.实现TCP通信所需的所有函数的接口类:

2.实现线程池版本的服务端:

3.客户端程序:

4.运行结果:

4.1服务端显示:

4.2客户端jack:

4.3客户端小明:

4.4客户端peter:

四.tcp面向连接特性:

1.TCP有自己的保活探测机制,在操作系统中会进行保活探测。当Linux下双方在7200s内无交互时,就会开始服务端就发保活探测包,每隔75s发一次,共发9次。若都没有回复,则认为连接断开。sysctl -a | grep keep

2.在代码中,若连接断开/对端关闭连接,则recv返回0;send会触发异常—SIGPIPE进程退出。

                                                                                                                                                           珍&源码

猜你喜欢

转载自blog.csdn.net/IT_xiaoQ/article/details/90607592