参数1:
要监听的描述符的个数,一般取需要监听的描述符中最大值加1;
参数2~4:
三个数组,分别表示可读,可写,和异常事件对应的描述符集合,数组大小也限制了所能监听的最大描述符个数,一般为1024个;以下四个函数用来操作这三个数组;
参数5:超时时间,不需要可以设置为NULL,表示如果没有活动的事件产生,将一直阻塞(不考虑信号);
服务端利用select()
监听客户端套接字上可读事件:
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <assert.h>
#include <stdio.h>
#include <unistd.h>
#include <errno.h>
#include <string.h>
#include <fcntl.h>
#include <stdlib.h>
int main(int argc,char* argv[])
{
struct sockaddr_in address;
bzero(&address,sizeof(address));
address.sin_family=AF_INET;
address.sin_port=htons(12346);
address.sin_addr.s_addr=htonl(INADDR_ANY);
int sock=socket(PF_INET,SOCK_STREAM,0);
assert(sock>=0);
int ret=bind(sock,(struct sockaddr*)&address,sizeof(address));
assert(ret!=-1);
ret=listen(sock,5);
assert(ret!=-1);
struct sockaddr_in client;
socklen_t length=sizeof(client);
int confd=accept(sock,(struct sockaddr*)&client,&length);
if ( confd < 0 )
{
printf( "errno is: %d\n", errno );
close( sock );
}
char remote_addr[INET_ADDRSTRLEN];
printf("ip:%s , port: %d\n",inet_ntop(AF_INET,&client.sin_addr,remote_addr,16),ntohs(client.sin_port));
char buf[1024];
fd_set rd_fds,ex_fds;
FD_ZERO(&rd_fds);
FD_ZERO(&ex_fds);
FD_SET(confd,&rd_fds);
FD_SET(confd,&ex_fds);
while(1)
{
memset(buf,'\0',sizeof(buf));
FD_SET(confd,&rd_fds);
FD_SET(confd,&ex_fds);
ret=select(confd+1,&rd_fds,nullptr,&ex_fds,nullptr);
if(ret<0){printf("select failed\n");}
if(FD_ISSET(confd,&rd_fds)){
ret=recv(confd,buf,sizeof(buf)-1,0);
if(ret<=0)break;
printf( "get %d bytes of normal data: %s\n", ret, buf );
}
}
close(confd);
close(sock);
return 0;
}
客户端:
服务端: