C++socket基础进阶:Select与封装
admin
2024-04-02 05:35:00
0

之前用C#做服务器没搞明白于是从笔者比较熟悉的C++开始入手从头学了一遍,整理一下笔记。
资料来源于《网络多人游戏架构与编程》第三章,这本书讲的很明白,比起网上每篇博客都在介绍的原理,这本书更偏向于代码实现。
代码应该没什么大问题,看懂逻辑的话能自己封装的更好。

上一篇介绍Socket基础函数的在这。

阻塞和非阻塞I/O

开启非阻塞模式

默认情况下,socket操作是阻塞模式,但可以由如下函数转变为非阻塞模式,非阻塞模式的socket被要求执行一个需要阻塞的操作时,它将立刻返回-1,还设置了错误代码errno或WSAGetLastError,分别返回EAGAIN或WSAEWOULDBLOCK。这个代码表示之前的socket行为已经阻塞,没有发生就被终止了。

//Windows系统
int ioctlsocket(SOCKET s,long cmd, //用于控制socket参数,在这种情况下,输入FIONBIOu_long* argp //是这个参数的取值,任意非零值将开启非阻塞模式,0将阻止开启
);
//POSIX兼容系统
int fcntl(int sock,int cmd, //发给socket的命令...
);

在更新的POSIX系统上,必须首先使用F_GETEL获取当前与socket相关的标志,让它们与常数O_NONBLOCK按位或运算之后,使用F_SETEL命令更新socket上的标志。

当socket出于非阻塞模式,调用任何阻塞函数都是安全的,因为我们知道如果它不能在没有阻塞的情况下完成,它会立刻返回。

Select

socket库提供了同时检查多个socket的方式,只要其中有一个socket准备好了就开始执行:

int select(int nfds,fd_set* readfds,fd_set* writefds,fd_set* exceptfds,const timeval* timeout
);

参数:

  1. nfds在POSIX平台,是待检查的编号最大的socket的标识符。在POSIX平台,每一个socket只是一个整数,所以直接将所有socket的最大值传入这个函数。在Windows平台,socket表示为指针,而不是整数,所以这个参数不起作用,可以忽略。
  2. readfds是指向socket集合的指针,称为fd_set,包含要检查可读性的socket。
  3. writefds是指向fd_set的指针,存储待检查可写性的socket。当select函数返回时,保留在writefds中所有socket都保证可写,不会引起调用线程的阻塞。给writefds传入nullptr来跳过任何socket可写性的检查。通常,只有当socket的输出缓冲区有太多数据时,socket才会阻塞写操作。
  4. excptfds是指向fd_set的指针,这个fd_set存储待检查错误的socket。当select函数返回,保留在exceptfds中的所有socket都已经发生了错误。给exceptfds传入nullptr来跳过任何错误的检查。
  5. timeout是指向超时之前可以等待最长时间的指针。如果在readfds中的任意一个socket可读,writefds中的任意一个socket可写,或者exceptfds中的任意一个socket发生错误之前发生超时,清空所有集合,select函数将控制返回给调用线程。给timeout输入nullptr来表名没有超时限制。

select函数返回执行之后保留在readfds、writefds和exceptfds中socket的数量。如果发生超时,这个值是0.

fd_set:

fd_set myReadSet;
FD_ZERO(&myReadSet);	//初始化一个空的fd_set
FD_SET(mySocket, &myReadSet);	//给fd_set添加一个socket
FD_INSET(mySocket, &myReadSet);	//检查在select函数返回值后,一个socket是否在fd_set中

封装

类型安全的SocketAddress类

class SocketAddress {
public:SocketAddress(uint32_t inAddress, uint16_t inPort) {GetAsSockAddrIn()->sin_family = AF_INET;GetAsSockAddrIn()->sin_addr.S_un.S_addr = htonl(inAddress);GetAsSockAddrIn()->sin_port = htons(inPort);}SocketAddress(const sockaddr& inSockAddr) {memcpy(&mSockAddr, &inSockAddr, sizeof(sockaddr));}size_t GetSize() const { return sizeof(sockaddr); }
private:sockaddr mSockAddr;sockaddr_in* GetAsSockAddrIn() {return reinterpret_cast (&mSockAddr);}
};
typedef std::shared_ptr SocketAddressPtr;

使用SocketAddressFactory类的域名解析

class SocketAddressFactory {
public:static SocketAddressPtr CreateIPv4FromString(const std::string& inString) {auto pos = inString.find_last_of(':');std::string host, service;if (pos != std::string::npos) {host = inString.substr(0, pos);service = inString.substr(pos + 1);}else {host = inString;//use default portservice = "0";}addrinfo hint;memset(&hint, 0, sizeof(hint));hint.ai_family = AF_INET;addrinfo* result;int error = getaddrinfo(host.c_str(), service.c_str(), &hint, &result);if (error != 0) {freeaddrinfo(result);return nullptr;}while (!result->ai_addr && result->ai_next)result = result->ai_next;if (!result->ai_addr) {freeaddrinfo(result);return nullptr;}auto toRet = std::make_shared(*result->ai_addr);freeaddrinfo(result);return toRet;}
};

类型安全的UDP Socket

class UDPSocket {
public:~UDPSocket();int Bind(const SocketAddress& inToAddress);int SendTo(const void* inData, int inLen, const SocketAddress& inTo);int ReceiveFrom(void* inBuffer, int inLen, SocketAddress& outFrom);
private:UDPSocket(SOCKET inSocket) : mSocket(inSocket) {}SOCKET mSocket;
};
typedef std::shared_ptr UDPSocketPtr;int UDPSocket::Bind(const SocketAddress& inBindAddress) {int err = bind(mSocket, &inBindAddress.mSockAddr, inBindAddress.GetSize());if (err != 0) {//return error from UDPSocket::Bindreturn -1;}return 0;
}
int UDPSocket::SendTo(const void* inData, int inLen, const SocketAddress& inTo) {int byteSentCount = sendto(mSocket, static_cast(inData), inLen, 0, &inTo.mSockAddr, inTo.GetSize());if (byteSentCount >= 0) return byteSentCount;//return error from UDPSocket::SendToreturn -1;
}
int UDPSocket::ReceiveFrom(void* inBuffer, int inLen, SocketAddress& outFrom) {int fromLength = outFrom.GetSize();int readByteCount = recvfrom(mSocket,static_cast (inBuffer),inLen,0,&outFrom.mSockAddr,&fromLength);if (readByteCount >= 0) return readByteCount;//return error from UDPSocket::ReceiveFromreturn -1;
}
UDPSocket::~UDPSocket() {closesocket(mSocket);
}

类型安全的TCP Socket

class TCPSocket {
public:~TCPSocket();int Connect(const SocketAddress& inAddress);int Bind(const SocketAddress& inToAddress);int Listen(int inBackLog = 32);std::shared_ptr Accept(SocketAddress& inFromAddress);int Send(const void* inData, int inLen);int Receive(void* inBuffer, int inLen);
private:TCPSocket(SOCKET inSocket) : mSocket(inSocket) {}SOCKET mSocket;
};
typedef std::shared_ptr TCPSocketPtr;int TCPSocket::Connect(const SocketAddress& inAddress) {int err = connect(mSocket, &inAddress.mSockAddr, inAddress.GetSize());if (err < 0) return -1;return NO_ERROR;
}
int TCPSocket::Listen(int inBackLog) {int err = listen(mSocket, inBackLog);if (err < 0)return -1;return NO_ERROR;
}
int TCPSocket::Send(const void* inData, int inLen) {int bytesSendCount = send(mSocket,static_cast (inData),inLen, 0);if (bytesSendCount < 0)return -1;return bytesSendCount;
}
int TCPSocket::Receive(void* inBuffer, int inLen) {int bytesReceiveCount = recv(mSocket,static_cast (inBuffer),inLen, 0);if (bytesReceiveCount < 0)return -1;return bytesReceiveCount;
}
TCPSocket::~TCPSocket() {closesocket(mSocket);
}

与类型安全的TCPSocket一起使用的select函数

fd_set* FillSetFromVector(fd_set& outSet,const std::vector* inSockets) {if (inSockets) {FD_ZERO(&outSet);for (const TCPSocketPtr& socket : *inSockets)FD_SET(socket->mSocket, &outSet);return &outSet;}return nullptr;
}void FillVectorFromSet(std::vector* outSockets,const std::vector* inSockets,const fd_set& inSet
) {if (inSockets && outSockets) {outSockets->clear();for (const TCPSocketPtr& socket : *inSockets)if (FD_ISSET(socket->mSocket, &inSet))outSockets->push_back(socket);}
}int Select(const std::vector* inReadSet,std::vector* outReadSet,const std::vector* inWriteSet,std::vector* outWriteSet,const std::vector* inExceptSet,std::vector* outExceptSet
) {fd_set read, write, except;fd_set* readPtr = FillSetFromVector(read, inReadSet);fd_set* writePtr = FillSetFromVector(read, inWriteSet);fd_set* exceptPtr = FillSetFromVector(read, inExceptSet);int toRet = select(0, readPtr, writePtr, exceptPtr, nullptr);if (toRet > 0) {FillVectorFromSet(outReadSet, inReadSet, read);FillVectorFromSet(outWriteSet, inWriteSet, write);FillVectorFromSet(outExceptSet, inExceptSet, except);}return toRet;
}

运行一个TCP服务器循环

void DoTCPLoop() {TCPSocketPtr listenSocket = CreateTCPSocket(INET);SocketAddress receivingAddres(INADDR_ANY, 48000);if (listenSocket->Bind(receivingAddres) != NO_ERROR)return;std::vector readBlockSockets;readBlockSockets.push_back(listenSocket);std::vector readableSockets;while (true) {if (Select(&readBlockSockets, &readableSockets,nullptr, nullptr,nullptr, nullptr)) {for (const TCPSocketPtr& socket : readableSockets) {if (socket == listenSocket) {SocketAddress newCilentAddress;TCPSocketPtr newSocket = listenSocket->Accept(newCilentAddress);readBlockSockets.push_back(newSocket);// 在这了处理新连入 ProcessNewCilent(newSocket, newCilentAddress);}else {char segment[MAX_SEGMENT_SIZE];int dataReceived = socket->Receive(segment, MAX_SEGMENT_SIZE);if (dataReceived > 0) {//在这里处理新数据 ProcessDataFromClient(socket, segment, dataReceived);}}}}}
}

相关内容

热门资讯

原创 今... 今日为5月23日,国际现货黄金价格在4500美元/盎司整数关口附近徘徊不前,日内最低触及4480美元...
三连亏后变为“无主”状态,农尚... 从吴亮手中接盘农尚环境(300536)不足三年后,林峰如今让出了公司控制权,上市公司进入“无主”状态...
55岁湖南女首富出手!豪掷13... 快科技5月24日消息,与马斯克、库克并肩而坐,刚参加完国宴的湖南女首富周群飞就买了家上市企业。 近日...
外资加仓A股,岂是跟风这么简单... 熬过忙碌的交易日,在周末安静时段,理清接下来布局方向。本篇为大家准备了5条要闻,涵盖市场动态、行业变...
原创 俄... 在全球能源的残酷牌桌上,手里攥着石油,腰杆子才能硬气。长期以来,中东的沙漠、俄罗斯的冰原、美国的页岩...
喜力啤酒有产品将涨价,华润啤酒... 来源:红星新闻 红星资本局5月22日消息,今日,红星资本局从雪花啤酒(厦门)有限公司、华润啤酒方面获...
原创 金... 心理预期调整刻不容缓,五月二十二日,黄金价格或将重现十五年前的历史性低迷。 近期若您密切关注着黄金市...
原创 马... 埃隆·马斯克如果能让SpaceX实现“科幻小说”级别的目标,他可能获得1万亿美元的收入。 埃隆·马斯...
涨涨涨!放开限制、可加杠杆!这... 韩国股市站在风口上! 据最新消息,为吸引更多海外资金进入股市,韩国政府计划放开限制,允许境外投资者直...
下周9家上会丨科创板首单IPO... IPO及再融资上会预告 据交易所官网审核动态信息,下周(5.25-5.29)IPO上会审核6家企业,...
富途、老虎市值蒸发1/4!或被... 来源:金融时报 5月22日,中国证监会宣布依法对Tiger Brokers (NZ) Limited...
马爸爸的好兄弟钱多多搞了杀猪盘... *此图由AI生成 作者| 史大郎&猫哥 来源| 是史大郎&大猫财经Pro 上周四,港股经纬天地大崩盘...
原创 壳... 编辑:XL 国际能源圈最近炸开了锅,壳牌这家百年石油巨头在2026年3月与委内瑞拉政府正式签署多项油...
存储热潮愈演愈烈!奖金拿到手软... 财联社5月24日讯(编辑 卞纯)在席卷全球的存储芯片热潮中,韩国“存储芯片双雄”SK海力士和三星无疑...
揽牌、合作、生态,跨境支付头部... 近日,国内头部跨境支付机构密集落地海外重要布局,一方面,连连数字、PingPong两家公司相继在中东...
原创 帮... 老铁们,周末好!我是帮主郑重。刚扫了一眼下周的财经日历,好家伙,事件一个接一个,堪称“消息面轰炸周”...
海南省住建厅与中国石化海南石油... 5月22日,中国石化海南石油分公司代表、党委书记李新强、总经理蔡文东一行赴海南省住建厅拜访交流。省住...
原创 金... 2026年5月22日,国际黄金价格报4536.7美元/盎司,较前期高点5597美元回落约1100美元...
“双标”换卡背后,银行还需多些... 新华社记者 颜之宏、杨深深 持到期银行卡和身份证去银行网点换新卡,却被要求“必须交回旧卡才能取新卡”...
“离境退税2.0”带动“中国购... 【环球时报综合报道】编者的话:5月18日,商务部等6部门联合发布《关于加力优化离境退税措施扩大入境消...