回顾select:
下面的事件支持selece可读:
1.socket内核接收缓存区中的字节数大于或等于其低水位标记SO_RCVLOWAT(0x1004)。我们可以无阻塞地读取该socket,并且读操作返回字节数为0。
2.socket通信的对方关闭连接。此时对该socket的读操作将返回0。
3.监听socket上有新的连接请求。
4.socket上有未处理的错误。此时我们可以使用getsockopt来读取和清除错误。
下面的事件支持select可写:
1.socket内核发送缓存区中的字节数大于或等于其低水位标记SO_SNDLOWAT ( 0x1003) 。我们可以无阻塞写该socket,并且写操作返回字节数大于0。
2.socket写操作被关闭。对写操作被关闭的socket执行之写操作将触发SIGPIPE信号,这个信号默认程序退出。
3.socket使用非阻塞connet连接成功或者失败之后触发写操作。
4.socket上有为处理的错误,我们用getsockopt读取和清除该错误。
网络程序中,select能处理的异常情况只有一种:socket上接收到带外数据。
带外数据:
有些传输层协议具有带外(Out Of Band,OOB)数据的概念,用于迅速通告对方本端发生的重要事件。因此,带外数据比普通数据(也称为带内数据)有更高的优先级,它应该总是立即被发送,而不论发送缓冲区中是否有排队等待发送的普通数据。带外数据的传输可以使用一条独立的传输层连接,也可以映射到传输普通数据的连接中。实际应用中,带外数据的使用很少见,已知的仅有telnet、ftp等远程非活跃程序。
UDP没有实现带外数据传输,TCP也没有真正的带外数据。不过TCP利用其头部中的紧急指针标志和紧急指针两个字段,给应用程序提供了一种紧急方式。TCP的紧急方式利用传输普通数据的连接来传输紧急数据。这种紧急数据的含义和带外数据类似,因此TCP紧急数据称为带外数据。
用select处理带外数据:
select上接收到普通数据或者带外数据都会使select返回,但是返回以后的fds处于不同的就绪状态,普通数据触发可读状态,带外数据触发异常状态。
服务端select处理带外数据代码:
/*************************************************************************
> File Name: 1.c
> Author: 朱紫钰
> Mail: zhuziyu1157817544@gmail.com
> Created Time: 2017年06月01日 星期四 18时38分23秒
************************************************************************/
#include<stdio.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<assert.h>
#include<errno.h>
#include<string.h>
#include<fcntl.h>
#include<stdlib.h>
#include<unistd.h>
#include<arpa/inet.h>
int main(int argc,char **argv)
{
if(argc <= 2) {
printf("argument is too less\n");
return 1;
}
const char *ip = argv[1];
int port = atoi(argv[2]);
int ret = 0;
struct sockaddr_in address;
bzero(&address,sizeof(address));
address.sin_family = AF_INET;
inet_pton(AF_INET,ip,&address.sin_addr);
int listenfd = socket(PF_INET,SOCK_STREAM,0);
assert(listenfd >= 0);
ret = bind(listenfd,(struct sockaddr*)&address,sizeof(address));
assert(ret != -1);
ret = listen(listenfd,5);
assert(ret != -1);
struct sockaddr_in client_address;
socklen_t client_length = sizeof(client_address);
int connfd = accept(listenfd,(struct sockaddr*)&client_address,&client_length);
if(connfd < 0){
printf("The errno is %d\n",connfd);
close(connfd);
}
char buf[1024];
fd_set read_fds;
fd_set except_fds;
FD_ZERO(&read_fds);
FD_ZERO(&except_fds);
/*监控的描述符已经定了*/
while(1){
memset(buf,'0',sizeof(buf));
FD_SET(connfd,&read_fds);
FD_SET(connfd,&except_fds);
ret = select(connfd+1,&read_fds,NULL,&except_fds,NULL);
if(ret < 0){
printf("selection failure\n");
break;
}
if(FD_ISSET(connfd,&read_fds)){
ret = recv(connfd,buf,sizeof(buf-1),0);
if(ret <= 0){
break;
}
printf("get %d bytes of normal data:%s\n",ret,buf);
}
else if(FD_ISSET(connfd,&except_fds)){
ret = recv(connfd,buf,sizeof(buf-1),0);
if(ret <= 0){
break;
}
printf("get %d bytes of obb data%s\n",ret,buf);
}
}
close(connfd);
close(listenfd);
return 0;
}