如何在读取函数调用中实现超时?
我想使用串行通信端口,我想实现一个超时每次我调用读取函数调用。
int filedesc = open( "dev/ttyS0", O_RDWR ); read( filedesc, buff, len );
编辑:
我正在使用Linux操作系统。 如何实现使用select函数调用?
select()需要5个参数,首先是最高文件描述符+ 1,然后是一个fd_set用于读取,一个用于写入,一个用于exception。 最后一个参数是一个结构timeval,用于超时。 它在错误时返回-1,在超时时返回0,或者设置的集合中的文件描述符的数量。
#include <stdio.h> #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h> #include <sys/select.h> int main(void) { fd_set set; struct timeval timeout; int rv; char buff[100]; int len = 100; int filedesc = open( "dev/ttyS0", O_RDWR ); FD_ZERO(&set); /* clear the set */ FD_SET(filedesc, &set); /* add our file descriptor to the set */ timeout.tv_sec = 0; timeout.tv_usec = 10000; rv = select(filedesc + 1, &set, NULL, NULL, &timeout); if(rv == -1) perror("select"); /* an error accured */ else if(rv == 0) printf("timeout"); /* a timeout occured */ else read( filedesc, buff, len ); /* there was data to read */ }
作为select()
的替代方法,对于串口(terminal)的特定情况,可以使用tcsetattr()
将文件描述符置于非规范模式,并具有读取超时。
为此,请取消ICANON
标志,并设置VTIME
控制字符:
struct termios termios; tcgetattr(filedesc, &termios); termios.c_lflag &= ~ICANON; /* Set non-canonical mode */ termios.c_cc[VTIME] = 100; /* Set timeout of 10.0 seconds */ tcsetattr(filedesc, TCSANOW, &termios);
注意VTIME
是以十分之一秒为VTIME
来测量的,用于它的types通常是一个unsigned char
,这意味着最大超时值是25.5秒。
如果你设置的套接字在非阻塞模式下运行,每次读取的调用将只读取当前可用的数据(如果有的话)。 所以这实际上等于立即超时。
您可以使用如下函数在套接字上设置非阻塞模式:
int setnonblock(int sock) { int flags; flags = fcntl(sock, F_GETFL, 0); if (-1 == flags) return -1; return fcntl(sock, F_SETFL, flags | O_NONBLOCK); }
(有关从非阻塞套接字读取的更多信息,请参阅read
手册页)
你不说什么操作系统,但如果你在Linux下运行,你可以使用select调用。 如果文件描述符中有某些内容需要读取,或者可以设置它,以便在没有任何内容需要读取的情况下会超时。 返回码指示哪个。
Linux考虑两种types的设备。 “慢”(如networking)和“快”(如磁盘)。 非阻塞模式适用于“慢速”设备,而不是“快速”模式。 块设备应该是“快”,所以select/轮询总是告诉他们已经准备好I / O。 对于一些闪存设备来说,这是一个谎言,我看到一些I / O调用,即使select / poll指示设备已准备好,几个字节也只用几个字节。 这就是说,即使对于慢速的设备,读取延迟本身也没有很强的保证。 如果您提供的缓冲区足够大并且数据可用(您已经使用select或poll来确保这一点),则读取调用本身可能会花费大量时间(networking上的几百ms)。