Socket编程C++中的选择()
我有一个socket程序,它有发送和接收两部分,但是当我使用select时,它总是返回时间错误。这是代码Socket编程C++中的选择(),c++,sockets,select,C++,Sockets,Select,我有一个socket程序,它有发送和接收两部分,但是当我使用select时,它总是返回时间错误。这是代码 void server::run(){ int RetVal; struct timeval *tp=new timeval; tp->tv_sec=300; tp->tv_usec=500000; fd_set readfds; FD_ZERO(&readfds); FD_SET(sock,&readfds);
void server::run(){
int RetVal;
struct timeval *tp=new timeval;
tp->tv_sec=300;
tp->tv_usec=500000;
fd_set readfds;
FD_ZERO(&readfds);
FD_SET(sock,&readfds);
FD_SET(sock1,&readfds);
cout<<"run() is working fine";
if((RetVal=select(1,&readfds,NULL,NULL,tp))==SOCKET_ERROR) //check for incoming packets.
{
cout<< "Timer error!";
Sleep(50);
}
else if(RetVal>0) //There are incoming packets.
{
if(FD_ISSET(sock1, &readfds)) //incoming packet from peer host 1
{
send();
}
else if(FD_ISSET(sock, &readfds)) //incoming packet from peer host 1
{
send1();
}
else{
cout<<"no port is open to communicate";
}
}
}
server::server(){
WSADATA wsadata;
try{
if (WSAStartup(0x0202,&wsadata)!=0)
throw"Starting WSAStartup() error\n";
//Display name of local host
if(gethostname(servername,HOSTNAME_LENGTH)!=0) //get the hostname
throw"Get the host name error,exit";
}
catch(char *str){
cerr<<str<<":"<<WSAGetLastError()<<endl;
}
printf("Server: %s waiting to be contacted for time/size request...\n",servername);
}
server::~server(){
WSACleanup();
}
void server::send(){
try{
if ((sock= socket(AF_INET, SOCK_STREAM, 0)) == SOCKET_ERROR)
{
cout<< "SERVER: socket unable to initialize\n";
}
servers.sin_family = AF_INET;
servers.sin_port = htons(port);
servers.sin_addr.s_addr = INADDR_ANY;
if (bind(sock, (sockaddr*)&servers, sizeof(servers)) == SOCKET_ERROR)
{
cout<<"SERVER: Cant Bind the Socket";
closesocket(sock);
}
if (listen(sock, MAXPENDING) == SOCKET_ERROR)
//if((value=listen(sock, MAXPENDING))==SOCKET_ERROR)
{
cout<<"SERVER: listening socket not open";
closesocket(sock);
}
if ((clientSock= accept(sock, 0, 0)) == SOCKET_ERROR)
{
throw"SERVER: connection not accepted";
}
if ((nBytes= send(clientSock, "Hello", 20, 0)) == SOCKET_ERROR)
{
throw"SERVER: data sending failed ";
}
}
catch(char *str){
cerr<<str<<":"<<WSAGetLastError()<<endl;
}
closesocket(clientSock);
closesocket(sock);
}
void server::send1(){
try{
if ((sock1= socket(AF_INET, SOCK_STREAM, 0)) == SOCKET_ERROR)
{
throw "SERVER: socket unable to initialize\n";
}
servers.sin_family = AF_INET;
servers.sin_port = htons(port);
servers.sin_addr.s_addr = INADDR_ANY;
if (bind(sock1, (sockaddr*)&servers, sizeof(servers)) == SOCKET_ERROR)
{
throw"SERVER: Cant Bind the Socket";
closesocket(sock1);
}
if (listen(sock1, MAXPENDING) == SOCKET_ERROR)
{
throw "SERVER: listening socket not open";
closesocket(sock1);
}
cout << "SERVER: Ready to run now...";
if ((clientSock= accept(sock1, 0, 0)) == SOCKET_ERROR)
{
throw"SERVER: connection not accepted";
}
if ((nBytes= send(clientSock, "Hello", 20, 0)) == SOCKET_ERROR)
{
throw"SERVER: data sending failed ";
}
}
catch(char *str){
cerr<<str<<":"<<WSAGetLastError()<<endl;
}
closesocket(clientSock);
closesocket(sock1);
}
我尝试了很多东西,但我的选择总是返回相同的东西。
我的程序在没有运行的情况下工作正常,如果我直接调用send和recv,请选择它的工作正常。但是我想select不起作用。select函数返回SOCKET\u ERROR,或者在出错时返回-1,超时时返回0。当您得到一个错误而不是超时时,您应该使用WSAGetLastError获取实际错误,并在例如中查找
您还应该将第一个参数设置为可移植的最高套接字编号+1
但是您有一个更大的问题,似乎在调用select之后才创建实际的套接字。这意味着select调用中使用的套接字将无效,这就是出现错误的原因
通常首先创建套接字,将它们绑定到本地地址,并将它们标记为侦听套接字。然后在套接字上进行选择,当可读时,它们具有可以接受的传入连接
我认为你需要重新考虑你的设计
你还应该考虑在你接受了一个连接后会发生什么。现在,您只需接受一个连接,发送一些文本,然后再次关闭它。但是,如果您想继续在已接受的套接字上通信,并且可能不止一个连接,那么您还有很多工作要做
您还尝试创建两个绑定到同一本地地址的套接字,但这将不起作用。请记住,被动侦听套接字实际上可以接受任意数量的传入连接,而不仅仅是单个连接
要检查哪些套接字可以接收数据,请使用例如select(选择)和包含所有已接受连接的读取集,以及侦听套接字。select函数返回套接字错误或实际错误时返回-1,超时时返回0。当您得到一个错误而不是超时时,您应该使用WSAGetLastError获取实际错误,并在例如中查找
您还应该将第一个参数设置为可移植的最高套接字编号+1
但是您有一个更大的问题,似乎在调用select之后才创建实际的套接字。这意味着select调用中使用的套接字将无效,这就是出现错误的原因
通常首先创建套接字,将它们绑定到本地地址,并将它们标记为侦听套接字。然后在套接字上进行选择,当可读时,它们具有可以接受的传入连接
我认为你需要重新考虑你的设计
你还应该考虑在你接受了一个连接后会发生什么。现在,您只需接受一个连接,发送一些文本,然后再次关闭它。但是,如果您想继续在已接受的套接字上通信,并且可能不止一个连接,那么您还有很多工作要做
您还尝试创建两个绑定到同一本地地址的套接字,但这将不起作用。请记住,被动侦听套接字实际上可以接受任意数量的传入连接,而不仅仅是单个连接
要检查哪些套接字可以接收数据,请使用例如select(带有包含所有已接受连接的读取集)和侦听套接字。如果存在内存泄漏,则每个循环分配一个新的timeval结构,但从未释放它。select的第一个参数错误。让它成为1+1。此外,在堆上分配struct timeval完全没有意义。使用局部变量并传递其地址进行选择。errno的值是多少?这是Windows套接字库,因此不会设置errno。您存在内存泄漏,每个循环分配一个新的timeval结构,但从未释放它。select的第一个参数错误。让它成为1+1。此外,在堆上分配struct timeval完全没有意义。使用局部变量并传递其地址进行选择。errno的值是多少?这是Windows套接字库,因此不会设置errno。这会导致错误WSAENOTSOCK 10038 nonsocket上的套接字操作。尝试对非套接字的对象执行操作。套接字句柄参数未引用有效的套接字,或者对于select,fd_集的成员无效。但我的其他两个套接字都在工作。@networks刚刚更新了我的答案,说明了为什么select不起作用。这给了我错误WSAENOTSOCK 10038 nonsocket上的套接字操作。尝试对非套接字的对象执行操作。套接字句柄参数未引用有效的套接字,或者对于select,fd_集的成员无效。但是我的其他两个套接字都工作了。@networks刚刚更新了我的答案,解释了为什么select不工作。