升级!!!运用select实现一个简单的TCP通信
多路转接模型
多路转接IO:对大量的描述符进行就绪事件监控–让进程能够仅仅对就绪的描述符执行操作 不仅仅提高效率,而且避免阻塞。 只要是存在对描述符进行监控的需求,都可以使用多路转接模型进行事件的监控
多路转接模型使用场景:只要对描述符有(可读,可写,异常)事件的监控需求都可以使用多路转接模型。也使用与对大量描述符进行监控,但是同一时间只有少量的描述符活跃的场景(因为三种模型都是并发处理的)。
当然udp可以使用多路转接模型,udp只有一个描述符,阻塞操作也可以,所以可以用也可以不用,使用了select模型或者epoll模型也没有差别。
select模型
select模型的操作流程简介
- 定义某个事件的描述符集合(可读事件描述符集合,可写事件描述符集合,异常事件描述符集合),初始化清空集合。对哪个描述符关心什么事件,就把这个描述符添加到相应事件的描述符集合中
- 将集合拷贝到内核中进行监控(只有内核才能知道你的文件描述符是否就绪),监控的原理是轮询遍历判断。(若超时等待,有描述符就绪事件则调用返回)
- 监控调用的返回,表示监控出错/ 有描述符就绪 / 监控等待超时了 并且调用返回的时候,将事件监控的描述符集合中未就绪的描述符集合从集合中移除(只保留就绪的描述符)
注意:因为调用返回的时候修改了集合,因此下次监控的时候,就需要重新向集合中添加描述符。 - 轮询判断哪个描述符仍然在哪个集合中,就确定了这个描述符是否就绪了某个事件,然后进行对应事件的操作即可。
注意:select并不会直接返回给用户就绪的描述符直接操作,而是返回了就绪的描述符集合,因此需要我们进行轮询判断
TCP的实现
封装一个TCP服务端
#pragma once
#include <iostream>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <unistd.h>
#include <stdio.h>
#include <string>
#include <string.h>
using namespace std;
//创建TCP类,提供监听listen,accept,recv,send等功能
class TcpSvr{
public:
//构造函数
TcpSvr()
:sock_fd_(-1)
{
}
~TcpSvr()
{
}
//创建套接字
int CreateSocket()
{
sock_fd_ = socket(AF_INET, SOCK_STREAM, 0);
if(sock_fd_ < 0)
{
//创建套接字失败
perror("socket\n");
return -1;
}
printf("套接字创建成功!\n");
return 0;
}
//绑定地址信息
int Bind(const string& IP, const uint16_t port)
{
sockaddr_in addr;
//协议
addr.sin_family = AF_INET;
//IP地址
addr.sin_addr.s_addr = inet_addr("0.0.0.0");
//端口
addr.sin_port = htons(19999);
int ret = bind(sock_fd_, (struct sockaddr*)&addr, sizeof(addr));
if(ret < 0)
{
perror("bind");
return -1;
}
printf("地址绑定成功!\n");
return 0;
}
//listen侦听套接字