c-如何在读取函数调用中实现超时?
我想使用串行com端口进行通信,并且每次调用read函数调用时都想实现超时。
int filedesc = open( "dev/ttyS0", O_RDWR );
read( filedesc, buff, len );
编辑:
我正在使用Linux OS。 如何使用选择函数调用实现?
domlao asked 2020-02-22T07:17:28Z
5个解决方案
76 votes
选择() 需要5个参数,首先是最高的文件描述符+ 1,然后是fd_set用于读取,一个用于写入,一个用于异常。 最后一个参数是struct timeval,用于超时。 错误时返回-1,超时时返回0或设置的集合中文件描述符的数量。
#include
#include
#include
#include
#include
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 */
}
Puppe answered 2020-02-22T07:17:43Z
39 votes
作为VTIME的替代方法,对于串行端口(终端)的特定情况,您可以使用unsigned char将文件描述符置于非规范模式,并具有读取超时。
为此,请取消设置VTIME标志,并设置unsigned char控制字符:
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以十分之一秒为单位进行测量,并且其使用的类型通常是unsigned char,这意味着最大超时为25.5秒。
caf answered 2020-02-22T07:18:13Z
8 votes
如果将套接字设置为在非阻塞模式下运行,则每次读取调用将仅读取当前可用的数据(如果有)。 因此,这实际上等于立即超时。
您可以使用以下功能在套接字上设置非阻塞模式:
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手册页)
sth answered 2020-02-22T07:18:42Z
4 votes
您没有说操作系统是什么,但是如果您在Linux下运行,则可以使用select调用。 如果文件描述符上有要读取的内容,它会返回,或者您可以对其进行设置,以便在没有内容可读取的情况下超时。 返回码指示哪个。
sizzzzlerz answered 2020-02-22T07:19:03Z
0 votes
下面的代码对每个字符使用Millisec超时。我在我的项目之一中使用它来从COM端口读取。
size_t TimeoutRead (int port, void*buf, size_t size, int mlsec_timeout)
{
struct pollfd fd = { .fd = port, .events = POLLIN };
size_t bytesread = 0;
while (poll (&fd, 1, mlsec_timeout) == 1)
{
int chunksize = read (port, buf + bytesread, size);
if (chunksize == -1)
return -1;
bytesread += chunksize;
size -= chunksize;
if (size == 0)
return bytesread;
}
// TODO: IsTimeout = true;
return bytesread;
}
rick-rick-rick answered 2020-02-22T07:19:23Z