poll模型
poll和select本质上没有区别,弃用了bitmap,采用数组表示法
每次调用poll()要把数组从用户态拷贝到内核,调用poll()之后把数组从内核态拷贝到用户态
poll()返回后,需要遍历数组,效率比较低
man 3 poll
poll(struct pollfd fds[], nfds_t nfds, int timeout);
pollfd这个结构体在man poll里
ppoll只是多了一个信号屏蔽参数
poll(fd[], maxfd, -1)
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <poll.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <sys/fcntl.h>
#define MAXNFDS 1024
int initserver(int port);
int main(int argc,char *argv[])
{
if (argc != 2) { printf("usage: ./tcppoll port\n"); return -1; }
int listensock = initserver(atoi(argv[1]));
printf("listensock=%d\n",listensock);
if (listensock < 0) { printf("initserver() failed.\n"); return -1; }
struct pollfd fds[MAXNFDS];
for (int ii=0;ii<MAXNFDS;ii++) fds[ii].fd=-1;
fds[listensock].fd=listensock;
fds[listensock].events=POLLIN;
int maxfd=listensock;
while (true)
{
int infds=poll(fds,maxfd+1,5000);
if (infds < 0)
{
perror("poll() failed"); break;
}
if (infds == 0)
{
printf("poll() timeout.\n"); continue;
}
for (int eventfd=0;eventfd<=maxfd;eventfd++)
{
if (fds[eventfd].fd<0) continue;
if ((fds[eventfd].revents&POLLIN)==0) continue;
fds[eventfd].revents=0;
if (eventfd==listensock)
{
struct sockaddr_in client;
socklen_t len = sizeof(client);
int clientsock = accept(listensock,(struct sockaddr*)&client,&len);
if (clientsock < 0) { perror("accept() failed"); continue; }
printf ("accept client(socket=%d) ok.\n",clientsock);
fds[clientsock].fd=clientsock;
fds[clientsock].events=POLLIN;
fds[clientsock].revents=0;
if (maxfd<clientsock) maxfd=clientsock;
}
else
{
char buffer[1024];
memset(buffer,0,sizeof(buffer));
if (recv(eventfd,buffer,sizeof(buffer),0)<=0)
{
printf("client(eventfd=%d) disconnected.\n",eventfd);
close(eventfd);
fds[eventfd].fd=-1;
if (eventfd == maxfd)
{
for (int ii=maxfd;ii>0;ii--)
{
if (fds[ii].fd!=-1)
{
maxfd = ii; break;
}
}
}
}
else
{
printf("recv(eventfd=%d):%s\n",eventfd,buffer);
send(eventfd,buffer,strlen(buffer),0);
}
}
}
}
return 0;
}
int initserver(int port)
{
int sock = socket(AF_INET,SOCK_STREAM,0);
if (sock < 0)
{
perror("socket() failed"); return -1;
}
int opt = 1; unsigned int len = sizeof(opt);
setsockopt(sock,SOL_SOCKET,SO_REUSEADDR,&opt,len);
struct sockaddr_in servaddr;
servaddr.sin_family = AF_INET;
servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
servaddr.sin_port = htons(port);
if (bind(sock,(struct sockaddr *)&servaddr,sizeof(servaddr)) < 0 )
{
perror("bind() failed"); close(sock); return -1;
}
if (listen(sock,5) != 0 )
{
perror("listen() failed"); close(sock); return -1;
}
return sock;
}