- 相關推薦
Linux-poll函數深入理解
其實,poll函數與select函數差不多,下面就一起來詳細了解一下poll函數吧!更多消息請關注應屆畢業生網!
函數原型:
#include
int poll(struct pollfd fd[], nfds_t nfds, int timeout);
struct pollfd的結構如下:
struct pollfd{
int fd; // 文件描述符
short event;// 請求的事件
short revent;// 返回的事件
}
每個pollfd結構體指定了一個被監視的文件描述符。第一個參數是一個數組,即poll函數可以監視多個文件描述符。每個結構體的events是監視該文件描述符的事件掩碼,由用戶來設置。revents是文件描述符的操作結果事件,內核在調用返回時設置。events中請求的任何事件都可能在revents中返回。合法的事件如下:
后三個只能作為描述字的返回結果存儲在revents中,而不能作為測試條件用于events中。
這些事件在events域中無意義,因為它們在合適的時候總是會從revents中返回。使用poll()和select()不一樣,你不需要顯式地請求異常情況報告。
POLLIN | POLLPRI等價于select()的讀事件,POLLOUT |POLLWRBAND等價于select()的寫事件。POLLIN等價于POLLRDNORM |POLLRDBAND,而POLLOUT則等價于POLLWRNORM。
例如,要同時監視一個文件描述符是否可讀和可寫,我們可以設置 events為POLLIN |POLLOUT。在poll返回時,我們可以檢查revents中的標志,對應于文件描述符請求的events結構體。如果POLLIN事件被設置,則文件描述符可以被讀取而不阻塞。如果POLLOUT被設置,則文件描述符可以寫入而不導致阻塞。這些標志并不是互斥的:它們可能被同時設置,表示這個文件描述符的讀取和寫入操作都會正常返回而不阻塞。
第二個參數nfds:要監視的描述符的數目。
timeout參數指定等待的毫秒數,無論I/O是否準備好,poll都會返回。timeout指定為負數值表示無限超時;timeout為0指示poll調用立即返回并列出準備好I/O的文件描述符,但并不等待其它的事件。這種情況下,poll()就像它的名字那樣,一旦選舉出來,立即返回。
成功時,poll()返回結構體中revents域不為0的文件描述符個數;如果在超時前沒有任何事件發生,poll()返回0;失敗時,poll()返回-1,并設置errno為下列值之一:
EBADF:一個或多個結構體中指定的文件描述符無效。
EFAULT:fds指針指向的地址超出進程的地址空間。
EINTR:請求的事件之前產生一個信號,調用可以重新發起。
EINVAL:nfds參數超出PLIMIT_NOFILE值。
ENOMEM:可用內存不足,無法完成請求。
demo:
代碼與上一篇文章中" 利用select實現IO多路復用TCP服務端 "中代碼差不多
#include
#include
#include
#include
#include
#include
#include
#include
#define MAX_BUFFER_SIZE 1024
#define IN_FILES 3
#define TIME_DELAY 60*5
#define MAX(a,b) ((a>b)?(a):(b))
int main(int argc ,char **argv)
{
struct pollfd fds[IN_FILES];
char buf[MAX_BUFFER_SIZE];
int i,res,real_read, maxfd;
fds[0].fd = 0;
if((fds[1].fd=open("data1",O_RDONLY|O_NONBLOCK)) < 0)
{
fprintf(stderr,"open data1 error:%s",strerror(errno));
return 1;
}
if((fds[2].fd=open("data2",O_RDONLY|O_NONBLOCK)) < 0)
{
fprintf(stderr,"open data2 error:%s",strerror(errno));
return 1;
}
for (i = 0; i < IN_FILES; i++)
{
fds[i].events = POLLIN;
}
for(i=0;i
{
fds[i].events = POLLIN;
}
while(fds[0].events || fds[1].events || fds[2].events)
{
if (poll(fds, IN_FILES, TIME_DELAY) <= 0)
{
printf("Poll error ");
return 1;
}
for (i = 0; i< IN_FILES; i++)
{
if (fds[i].revents)
{
memset(buf, 0, MAX_BUFFER_SIZE);
real_read = read(fds[i].fd, buf, MAX_BUFFER_SIZE);
if (real_read < 0)
{
if (errno != EAGAIN)
{
return 1;
}
}
else if (!real_read)
{
close(fds[i].fd);
fds[i].events = 0;
}
else
{
if (i == 0)
{
if ((buf[0] == 'q') || (buf[0] == 'Q'))
{
return 1;
}
}
else
{
buf[real_read] = '';
printf("%s", buf);
}
}
}
}
}
exit(0);
}
【Linux-poll函數深入理解】相關文章:
深入理解java的反射07-16
深入理解PHP的.htaccess文件08-11
如何深入理解photoshop通道09-14
深入理解Java事物原理與應用09-18
最新的Java容器類的深入理解09-05
淺談Java線程中斷的本質深入理解08-23
Java Tomcat和激活MyEclips的深入理解10-20
對Java中HashMap和TreeMap的區別的深入理解06-09
C語言指針函數和函數指針詳解09-29