socket编程之select()
int select(int maxfdp,fd_set *readfds,fd_set *writefds,fd_set *errorfds,struct timeval *timeout);
参数列表:
int maxfdp,是指集合中所有文件描述符的范围,即所有文件描述符的最大值加1,在Windows中这个参数的值无所谓。
fd_set *readfds是指向fd_set结构的指针,这个集合中应该包括文件描述符,我们是要监视这些文件描述符的读变化的,即我们关心是否可以从这些文件中读取数据了,如果这个集合中有一个文件可读,select就会返回一个大于0的值,表示有文件可读,如果没有可读的文件,则根据timeout参数再判断是否超时,若超出timeout的时间,select返回0,若发生错误返回负值。可以传入NULL值,表示不关心任何文件的读变化。
fd_set *writefds是指向fd_set结构的指针,这个集合中应该包括文件描述符,我们是要监视这些文件描述符的写变化的,即我们关心是否可以向这些文件中写入数据了,如果这个集合中有一个文件可写,select就会返回一个大于0的值,表示有文件可写,如果没有可写的文件,则根据timeout参数再判断是否超时,若超出timeout的时间,select返回0,若发生错误返回负值。可以传入NULL值,表示不关心任何文件的写变化。
fd_set *errorfds同上面两个参数的意图,用来监视文件错误异常。
struct timeval* timeout是select的超时时间,这个参数至关重要,它可以使select处于三种状态:
第一,若将NULL以形参传入,即不传入时间结构,就是将select置于阻塞状态,一定等到监视文件描述符集合中某个文件描述符发生变化为止;
第二,若将时间值设为0秒0毫秒,就变成一个纯粹的非阻塞函数,不管文件描述符是否有变化,都立刻返回继续执行,文件无变化返回0,有变化返回一个正值;
第三,timeout的值大于0,这就是等待的超时时间,即 select在timeout时间内阻塞,超时时间之内有事件到来就返回了,否则在超时后不管怎样一定返回,返回值同上述。
返回值:
负值:select错误
正值:某些文件可读写或出错
0:等待超时,没有可读写或错误的文件
readfds:
If listen has been called and a connection is pending, accept will succeed.
Data is available for reading (includes OOB data if SO_OOBINLINE is enabled).
Connection has been closed/reset/terminated.
writefds:
If processing a connect call (nonblocking), connection has succeeded.
Data can be sent.
exceptfds:
If processing a connect call (nonblocking), connection attempt failed.
OOB data is available for reading (only if SO_OOBINLINE is disabled).
FD_CLR(s, *set):从set中删除套接字s。
FD_ISSET(s, *set):检查s是否set集合的一名成员;如答案是肯定的是,则返回TRUE。
FD_SET(s, *set):将套接字s加入集合set。
FD_ZERO( * set):将set初始化成空集合。
在大规模的网络连接方面,还是推荐使用IOCP或EPOLL模型.但是Select模型可以使用在诸如对战类游戏上,比如类似星际这种,因为它小巧易于实现,且对战类游戏的网络连接量并不大. 对于Select模型想要突破Windows 64个限制的话,可以采取分段轮询,一次轮询64个.例如套接字列表为128个,在第一次轮询时,将前64个放入队列中用Select进行状态查询, 待本次操作全部结束后.将后64个再加入轮询队列中进行轮询处理.这样处理需要在非阻塞式下工作.以此类推,Select也能支持无限多个。
简单例子:
#define _WINSOCK_DEPRECATED_NO_WARNINGS
#include <WinSock2.h>
#include <iostream>
#include<Windows.h>
#pragma comment(lib, "ws2_32.lib")
using namespace std;
SOCKET server, clientSet[FD_SETSIZE];
int serverAddLen;//服务器地址长度
sockaddr_in serverAdd;//服务器地址
int clientCount = 0;//记录接入客户端的数量
DWORD WINAPI WorkerThread(LPVOID lpParam);
int main()
{
WSADATA wsaData;
SOCKET client;
sockaddr_in clientAdd;
int clientAddLen = sizeof(clientAdd);
DWORD dwThreadId;
DWORD ret = MAKEWORD(2, 2);
if (ret == WSAStartup(0x0202, &wsaData))
{
cerr << "初始化失败!" << endl;
}
serverAdd.sin_addr.S_un.S_addr = htonl(INADDR_ANY);
serverAdd.sin_family = AF_INET;
serverAdd.sin_port = htons(5000);
serverAddLen = sizeof(serverAdd);
server = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
bind(server, (struct sockaddr*)&serverAdd, serverAddLen);
listen(server, SOMAXCONN);
CreateThread(NULL, 0, WorkerThread, NULL, 0, &dwThreadId);
while (clientCount < FD_SETSIZE)
{
client = accept(server, (struct sockaddr *)&clientAdd, &clientAddLen);
cout << "Accepted client:" << inet_ntoa(clientAdd.sin_addr) << ":" << ntohs(clientAdd.sin_port);
clientSet[clientCount++] = client;
}
WSACleanup();
system("pause");
return 1;
}
DWORD WINAPI WorkerThread(LPVOID lpParam)
{
fd_set fdread;
int ret;
char szMesage[8192];
while (1)
{
int i;
FD_ZERO(&fdread);
for (i = 0; i < clientCount; ++i)
{
if (clientSet[i] != NULL)
FD_SET(clientSet[i], &fdread);
}
timeval timeout = { 3,0 };
ret = select(i + 1, &fdread, NULL, NULL, &timeout);
if (ret == 0) //超时
{
continue;
}
for (i = 0; i < clientCount; ++i)
{
if (FD_ISSET(clientSet[i], &fdread))
{
ret = recv(clientSet[i], szMesage, sizeof(szMesage), 0);
if (ret == 0 || ret == SOCKET_ERROR && WSAGetLastError() == WSAECONNRESET)
{
cerr << "client socket closed" << endl;
FD_CLR(clientSet[i], &fdread);
closesocket(clientSet[i]);
clientSet[i] = NULL;
}
else
{
szMesage[ret] = '\0';
cout << szMesage << endl;
}
}
}
}
return 0;
}
改进:
#define _WINSOCK_DEPRECATED_NO_WARNINGS
#include <WinSock2.h>
#include <iostream>
#include<Windows.h>
#pragma comment(lib, "ws2_32.lib")
using namespace std;
SOCKET server, clientSet[FD_SETSIZE];
int serverAddLen;//服务器地址长度
sockaddr_in serverAdd;//服务器地址
int clientCount = 0;//记录接入客户端的数量
void WorkerThread();
int main()
{
WSADATA wsaData;
DWORD dwThreadId;
DWORD ret = MAKEWORD(2, 2);
if (ret == WSAStartup(0x0202, &wsaData))
{
cerr << "初始化失败!" << endl;
}
serverAdd.sin_addr.S_un.S_addr = htonl(INADDR_ANY);
serverAdd.sin_family = AF_INET;
serverAdd.sin_port = htons(5000);
serverAddLen = sizeof(serverAdd);
server = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
bind(server, (struct sockaddr*)&serverAdd, serverAddLen);
listen(server, SOMAXCONN);
WorkerThread();
/*while (clientCount < FD_SETSIZE)
{
client = accept(server, (struct sockaddr *)&clientAdd, &clientAddLen);
cout << "Accepted client:" << inet_ntoa(clientAdd.sin_addr) << ":" << ntohs(clientAdd.sin_port)<<endl;
clientSet[clientCount++] = client;
}*/
WSACleanup();
system("pause");
return 1;
}
void WorkerThread()
{
fd_set fdread;
int ret;
char szMesage[8192];
SOCKET client;
sockaddr_in clientAdd;
int clientAddLen = sizeof(clientAdd);
while (1)
{
int i;
FD_ZERO(&fdread);
FD_SET(server, &fdread);
for (i = 0; i < clientCount; ++i)
{
if (clientSet[i] != NULL)
FD_SET(clientSet[i], &fdread);
}
timeval timeout = { 3,0 };
ret = select(i + 1, &fdread, NULL, NULL, &timeout);
if (ret == 0) //超时
{
continue;
}
else if(ret== SOCKET_ERROR&&GetLastError()!= WSAEINVAL)//WSAEINVAL参数全为空错误
{
cerr << "select error!" <<GetLastError()<< endl;
break;
}
if (FD_ISSET(server, &fdread))
{
if (clientCount < FD_SETSIZE)
{
cout << "有新的连接请求!" << endl;
client = accept(server, (sockaddr*)&clientAdd, &clientAddLen);
if (client == INVALID_SOCKET)
{
cerr << "无法接受请求" << endl;
}
else
{
cout << "Accepted client:" << inet_ntoa(clientAdd.sin_addr) << ":" << ntohs(clientAdd.sin_port) << endl;
clientSet[clientCount++] = client;
}
}
else
{
cerr << "达到上限无法接受请求!" << endl;
}
}
for (i = 0; i < clientCount; ++i)
{
if (FD_ISSET(clientSet[i], &fdread))
{
ret = recv(clientSet[i], szMesage, sizeof(szMesage), 0);
if (ret == 0 || ret == SOCKET_ERROR && WSAGetLastError() == WSAECONNRESET)
{
cerr << "client socket closed" << endl;
FD_CLR(clientSet[i], &fdread);
closesocket(clientSet[i]);
clientSet[i] = NULL;
}
else
{
szMesage[ret] = '\0';
cout << szMesage << endl;
}
}
}
}
}
socket编程之select()的更多相关文章
- socket编程之 select、poll、kqueue、epoll
原生API select int select(int numfds, fd_set *readfds, fd_set *writefds, fd_set *exceptfds, struct tim ...
- 详述socket编程之select()和poll()函数
转自:http://www.cppblog.com/myjfm/archive/2011/10/26/159093.aspx select()函数和poll()函数均是主要用来处理多路I/O复用的情况 ...
- socket编程之select相关
FD_ZERO,FD_ISSET这些都是套节字结合操作宏 看看MSDN上的select函数, 这是在select io 模型中的核心,用来管理套节字IO的,避免出现无辜锁定. int se ...
- socket编程之select(),poll(),epoll()
socket编程,通信 client端 socket() ----->connect() ------->recv() -----> close(); server端 socket ...
- PHP Socket 编程之9个主要函数的使用之测试案例
php的socket编程算是比较难以理解的东西吧,不过,我们只要理解socket几个函数之间的关系,以及它们所扮演的角色,那么理解起来应该不是很难了,在笔者看来,socket编程,其实就是建立一个网络 ...
- [深入浅出WP8.1(Runtime)]Socket编程之UDP协议
13.3 Socket编程之UDP协议 UDP协议和TCP协议都是Socket编程的协议,但是与TCP协议不同,UDP协议并不提供超时重传,出错重传等功能,也就是说其是不可靠的协议.UDP适用于一次只 ...
- iPhone socket 编程之BSD Socket篇
iPhone socket 编程之BSD Socket篇 收藏在进行iPhone网络通讯程序的开发中,不可避免的要利用Socket套接字.iPhone提供了Socket网络编程的接口CFSocket, ...
- 老雷socket编程之websocket实现
老雷socket编程之websocket实现 我们主要实现私聊和群聊两个功能,要在web端实现想微信QQ那样的即时通讯的功能,我们需要了解一下websocket.websocket是一种可以双向通讯的 ...
- 老雷socket编程之PHP利用socket扩展实现聊天服务
老雷socket编程之PHP利用socket扩展实现聊天服务 socket聊天服务原理 PHP有两个socket的扩展 sockets和streamssockets socket_create(AF_ ...
随机推荐
- 剑指offer--40.翻转单词顺序列
时间限制:1秒 空间限制:32768K 热度指数:276854 本题知识点: 字符串 题目描述 牛客最近来了一个新员工Fish,每天早晨总是会拿着一本英文杂志,写些句子在本子上.同事Cat对Fish写 ...
- MySQL 不开启slave如何完成异地复制
1,分批次通过远程的binlog来进行数据加载 业务新需求,线上数据库数据拉到本次,但是不允许开启slave服务,不建立直接外网的数据库账号,也不能打通数据库对外网的网络,所以我们测试环境无法通过普通 ...
- Faster R-CNN改进篇(一): ION ● HyperNet ● MS CNN
一. 源起于Faster 深度学习于目标检测的里程碑成果,来自于这篇论文: Ren, Shaoqing, et al. "Faster R-CNN: Towards real-time ob ...
- 在tableView中设置cell的图片和文字
// 设置UITableViewCellEditingStyle的 accessoryType UITableViewCellAccessoryNone, // d ...
- PyQt: “AttributeError: 'Form' object has no attribute 'exec_'” when opening second window
# -*- coding: utf-8 -*- import sys from PyQt5.QtWidgets import QApplication , QMainWindow from PyQt5 ...
- 【剑指offer】和为s的两个数字
原创博文,转载请注明出处! # 题目 # 思路 首先定义两个指针,第一个指针p指向数组的第一个数字,第二个指针q指向数组的最后一个数字.如果p+q=s,则找到要找的数字:如果p+q<s,则p向后 ...
- 模块(Modules)
一.引入模块 模块:当编写更大的应用程序时,所有的代码肯定会分成多个文件,这样便于维护,另外已经编写好的函数和对象在被多个程序中使用时,不用把函数和对象拷贝到每个程序中. 模块支持以上功能,在Pyth ...
- 新手向——关于Python3.5在Windows 10 系统下发布模块的终极讲解
博主自己在发布Python模块的时候也是摸索了好久啊,因为跟着书上写的步骤一步一步来终究会跪的节奏有木有啊!!!几经波折终于搞出来了,贴下来与诸君共勉.之前的步骤相信大家都已经知道了,那我们就直接跳过 ...
- 每天一个linux命令:【转载】mv命令
mv命令是move的缩写,可以用来移动文件或者将文件改名(move (rename) files),是Linux系统下常用的命令,经常用来备份文件或者目录. 1.命令格式: mv [选项] 源文件或目 ...
- 《DSP using MATLAB》示例Example 7.13
代码: M = 25; alpha = (M-1)/2; n = [0:1:M-1]; hd = (2/pi) * ( (sin( (pi/2)*(n-alpha) ).^2)./(n-alpha) ...