问题描述
我正在编写一个可移植的Socket类,该类支持发送和接收的超时.要实现这些超时,我使用的是select()
....但是,有时我需要知道我在内部被阻塞了多长时间. select()
当然,在Linux上我可以通过在调用select()
之前和之后调用gettimeofday()
然后使用timersub()
计算增量...
I'm writing a portable Socket class that supports timeouts for both sending and receiving... To implement these timeouts I'm using select()
.... But, I sometimes need to know how long I was blocked inside select()
which of course on Linux I would implement by calling gettimeofday()
before and after I call select()
and then using timersub()
to calculate the delta...
鉴于Windows上的select()
接受struct timeval
的超时,我应该使用哪种方法替换Windows上的gettimeofday()?
Given that select()
on Windows accepts struct timeval
for it's timeout, what method should I used to replace gettimeofday() on Windows?
推荐答案
I ended up finding this page: gettimeofday()
function for Windows (now via the Wayback Machine) which has a handy, dandy implementation of gettimeofday()
on Windows. It uses the GetSystemTimeAsFileTime()
method to get an accurate clock.
更新:这是从"Unix到HPC Windows移植词典"的替代活动链接 gettimeofday()
(现在通过Wayback Machine),它指向OP所引用的实现.还要注意,链接的实现中有一个错字:
Update: Here's an alternative active link from the 'Unix to Windows Porting Dictionary for HPC' gettimeofday()
(now via the Wayback Machine) that points to the implementation the OP referred to. Note also that there's a typo in the linked implementation:
#if defined(_MSC_VER) || defined(_MSC_EXTENSIONS)
#define DELTA_EPOCH_IN_MICROSECS 11644473600000000Ui64 // WRONG
#else
#define DELTA_EPOCH_IN_MICROSECS 11644473600000000ULL // WRONG
#endif
显示的值最后缺少一个额外的0
(它们假定为微秒,而不是100纳秒间隔的数量).通过此评论找到了这种错字 Google代码项目页面.正确使用的值如下所示:
The values shown are missing an extra 0
at the end (they assumed microseconds, not the number of 100-nanosecond intervals). This typo was found via this comment on a Google code project page. The correct values to use are shown below:
#if defined(_MSC_VER) || defined(_MSC_EXTENSIONS)
#define DELTA_EPOCH_IN_MICROSECS 116444736000000000Ui64 // CORRECT
#else
#define DELTA_EPOCH_IN_MICROSECS 116444736000000000ULL // CORRECT
#endif
PostgreSQL在Windows上的gettimeofday实现:
/*
* gettimeofday.c
* Win32 gettimeofday() replacement
*
* src/port/gettimeofday.c
*
* Copyright (c) 2003 SRA, Inc.
* Copyright (c) 2003 SKC, Inc.
*
* Permission to use, copy, modify, and distribute this software and
* its documentation for any purpose, without fee, and without a
* written agreement is hereby granted, provided that the above
* copyright notice and this paragraph and the following two
* paragraphs appear in all copies.
*
* IN NO EVENT SHALL THE AUTHOR BE LIABLE TO ANY PARTY FOR DIRECT,
* INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES, INCLUDING
* LOST PROFITS, ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS
* DOCUMENTATION, EVEN IF THE UNIVERSITY OF CALIFORNIA HAS BEEN ADVISED
* OF THE POSSIBILITY OF SUCH DAMAGE.
*
* THE AUTHOR SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE. THE SOFTWARE PROVIDED HEREUNDER IS ON AN "AS
* IS" BASIS, AND THE AUTHOR HAS NO OBLIGATIONS TO PROVIDE MAINTENANCE,
* SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS.
*/
#include "c.h"
#include <sys/time.h>
/* FILETIME of Jan 1 1970 00:00:00. */
static const unsigned __int64 epoch = ((unsigned __int64) 116444736000000000ULL);
/*
* timezone information is stored outside the kernel so tzp isn't used anymore.
*
* Note: this function is not for Win32 high precision timing purpose. See
* elapsed_time().
*/
int
gettimeofday(struct timeval * tp, struct timezone * tzp)
{
FILETIME file_time;
SYSTEMTIME system_time;
ULARGE_INTEGER ularge;
GetSystemTime(&system_time);
SystemTimeToFileTime(&system_time, &file_time);
ularge.LowPart = file_time.dwLowDateTime;
ularge.HighPart = file_time.dwHighDateTime;
tp->tv_sec = (long) ((ularge.QuadPart - epoch) / 10000000L);
tp->tv_usec = (long) (system_time.wMilliseconds * 1000);
return 0;
}
这篇关于在Windows上应该使用什么来替换gettimeofday()?的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持!