poll

12_30 I/O复用poll

1.如果一个一个字符读取,有几个字符就读去几次,因为接收缓冲区内还有数据

2.如果直接结束客户端,select就会收到这个描述符上有读事件,recv就不会受到数据0,就会知道已经关闭,然后就关闭它。

int poll(struct pollfd* fds,nfds_t nfds,int timeout);

struct pollfd
{
	int fd;//文件描述符
	short events;//注册的事件
	short revents;//实际发生的事件,由内核填充
}

poll的文件描述符没有上限

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <assert.h>
#include <sys/select.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>

#define MAX 10
int socket_init();

void poll_fds_init(struct pollfd fds[])
{
	for(int i=0;i<MAX;i++)
	{
		fds[i].fd=-1;
		fds[i].events=0;
		fds[i].revents=0;
	}
}

void poll_fds_add(int fd,struct pollfd fds[])
{
	for(int i=0;i<MAX;i++)
	{
		if(fds[i].fd==-1)
		{
			fds[i].fd=fd;
			fds[i],events=POLLIN;
			fds[i].revents=0;
			break;
		}
	}
}

void poll_fds_del(int fd,struct pollfd fds[])
{
	for(int i=0;i<MAX;i++)
	{
		if(fds[i].fd==fd)
		{
			fds[i].fd==-1;
			fds[i].events=0;
			fds[i].revents=0;
			break;
		}
	}
}


int main()
{
	int sockfd=socket_init();
	assert(sockfd!=-1);
	
	struct pollfd fds[MAX];
	poll_fds_init(poll_fds);
	poll_fds_add(sockfd,poll_fds);
	
	while(1)
	{
		int n=poll(poll_fds,MAX,5000);
		if(n<0)
		{
			printf("poll error\n");
		}
		else if(n==0)
		{
			printf("time out\n");
		}
		else
		{
			for(int i=0;i<MAX;i++)
			{
				if(poll_fds[i].fd==-1)
				{
					continue;
				}
				
				if(poll_fds[i]_revents&POLLIN)
				{
					if(poll_fds[i].fd===sockfd)
					{
						struct sockaddr_in caddr;
						int len=sizeof(caddr);
						int c=accept(sockfd,(struct sockaddr*)&caddr,&len);
						if(c<0)
						{
							continue;
						}
						printf("accept:%d\n",c);
						
						poll_fds_add(poll_fds);
					}
					else
					{
						char buff[128]={0};
						int num=recv(poll_fds[i].fd,buff,127,0);
						if(num<=0)
						{
							close(poll_fds[i].fd);
							poll_fds_del(poll_fds[i].fd,poll_fds);
							printf("client close\n");
						}
						else
						{
							printf("recv(%d)=%s\n",poll_fds[i].fd,buff);
							send(poll_fds[i].fd,"ok",2,0);
						}
					}
				}
				
			}
		}
	}
	
}

int socket_init()
{
	int sockfd=socket(AF_INET,SOCK_STREAM,0);
	if(sockfd!=-1)
	{
		return -1;
	}
	
	struct sockaddr_in saddr;
	memset(saddr,0,sizeof(saddr));
	saddr.sin_family=AF_INET;
	saddr.sin_port=htons(6000);
	saddr.sin_saddr.s_addr=inet_addr("127.0.0,1");
	
	int res=bind(sockfd,(struct sockaddr*)&saddr,sizeof(saddr));
	if(res==-1)
	{
		return -1;
	}
	
	res=listen(sockfd,5);
	if(res==-1)
	{
		return -1;
	}
	return sockfd;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值