我有一个程序(混合C和Fortran,虽然似乎没有相关性)nanosleep.但是,如果我timespec的tv_sec值为0,则它根本不会睡眠.该tv_nsec值可以是一秒钟的微秒,但它不会睡眠.(如果tv_sec是的话1,它可以暂时休眠一下.)为什么会这样?
为了让事情更加混乱,usleep使用适当的值(即995000usec)按预期睡眠一秒钟.
我在RHEL 5.8和RHEL 6.4框中看到了这个问题.两者都在使用gcc.
这是调用nanosleep的函数:
void msleep(int *milliseconds)
{
long usec;
struct timespec sleep;
usec = (*milliseconds) % 1000;
sleep.tv_sec = (*milliseconds) / 1000;
sleep.tv_nsec = 1000*usec;
nanosleep(&sleep, NULL);
}
Run Code Online (Sandbox Code Playgroud)
显然,我实际上并不需要纳秒精度!
我还测试了一个版本,我检查了返回值; 它总是0(成功),因此rem输出参数(如果中断则剩余时间)从未设置.
你缺少一个因子1000.
试试这个:
#define _POSIX_C_SOURCE 199309L /* shall be >= 199309L */
#include <time.h>
void msleep(int *milliseconds)
{
int ms_remaining = (*milliseconds) % 1000;
long usec = ms_remaining * 1000;
struct timespec ts_sleep;
ts_sleep.tv_sec = (*milliseconds) / 1000;
ts_sleep.tv_nsec = 1000*usec;
nanosleep(&ts_sleep, NULL);
}
Run Code Online (Sandbox Code Playgroud)
更紧凑:
#define _POSIX_C_SOURCE 199309L /* shall be >= 199309L */
#include <time.h>
void msleep(int * pmilliseconds)
{
struct timespec ts_sleep =
{
*pmilliseconds / 1000,
(*pmilliseconds % 1000) * 1000000L
};
nanosleep(&ts_sleep, NULL);
}
Run Code Online (Sandbox Code Playgroud)
最后是一个完整的实现,包括错误处理和nanosleep()早期中断的情况:
#define _POSIX_C_SOURCE 199309L
#include <time.h>
#include <errno.h>
#include <stdio.h>
int ms_sleep(unsigned int ms)
{
int result = 0;
{
struct timespec ts_remaining =
{
ms / 1000,
(ms % 1000) * 1000000L
};
do
{
struct timespec ts_sleep = ts_remaining;
result = nanosleep(&ts_sleep, &ts_remaining);
}
while ((EINTR == errno) && (-1 == result));
}
if (-1 == result)
{
perror("nanosleep() failed");
}
return result;
}
Run Code Online (Sandbox Code Playgroud)
遵循包装器以满足OP的要求:
#include <errno.h>
#include <stdio.h>
int ms_sleep(unsigned int);
void msleep(int * pms)
{
int result = 0;
if ((NULL == pms) || (0 > *pms)) /* Check for valid input. */
{
errno = EINVAL;
result = -1;
}
else
{
result = ms_sleep(*pms));
}
if (-1 == result)
{
perror("ms_sleep() failed");
/* Exit and/or log error here. */
}
}
Run Code Online (Sandbox Code Playgroud)
更新(参考下面的chux评论):
假设至少C99,这部分上面的代码
struct timespec ts_sleep =
{
*pmilliseconds / 1000,
(*pmilliseconds % 1000) * 1000000L
};
Run Code Online (Sandbox Code Playgroud)
可能最好像这样写
struct timespec ts_sleep =
{
.tv_sec = *pmilliseconds / 1000,
.tv_nsec = (*pmilliseconds % 1000) * 1000000L
};
Run Code Online (Sandbox Code Playgroud)
不依赖于struct timespec会员的顺序.