Czego powinienem użyć, aby zastąpić gettimeofday () w systemie Windows?

Piszę przenośną klasę gniazd, która obsługuje timeouty zarówno dla wysyłania, jak i odbierania... Aby zaimplementować te timeouty używam select().... Ale czasami muszę wiedzieć, jak długo byłem zablokowany wewnątrz select() co oczywiście na Linuksie zaimplementowałbym wywołując gettimeofday() przed i po wywołaniu select(), a następnie używając timersub() do obliczenia delta...

Biorąc pod uwagę, że select() w systemie Windows akceptuje struct timeval jako timeout, jakiej metody powinienem użyć, aby zastąpić gettimeofday() w systemie Windows?

Author: dicroce, 2009-11-04

4 answers

Znalazłem tę stronę: gettimeofday () w systemie windows [edit: link usunięty, ponieważ teraz wskazuje na stronę reklamową]. Który ma poręczną, elegancką implementację gettimeofday () w systemie Windows. Używa metody GetSystemTimeAsFileTime(), aby uzyskać dokładny zegar.

Update: Oto aktywny link [edit: link usunięty, ponieważ teraz wskazuje na stronę reklamową], który wskazuje na implementację, o której mowa w OP. Zauważ również, że w linkowanych jest literówka realizacja:

#if defined(_MSC_VER) || defined(_MSC_EXTENSIONS)
  #define DELTA_EPOCH_IN_MICROSECS  11644473600000000Ui64 // WRONG
#else
  #define DELTA_EPOCH_IN_MICROSECS  11644473600000000ULL // WRONG
#endif

Pokazanym wartościom brakuje dodatkowego 0 na końcu (zakładano mikrosekundy, a nie liczbę 100-nanosekundowych interwałów). Ta literówka została znaleziona przez ten komentarz {[13] } na stronie projektu Google code. Prawidłowe wartości są pokazane poniżej:

#if defined(_MSC_VER) || defined(_MSC_EXTENSIONS)
  #define DELTA_EPOCH_IN_MICROSECS  116444736000000000Ui64 // CORRECT
#else
  #define DELTA_EPOCH_IN_MICROSECS  116444736000000000ULL // CORRECT
#endif

Implementacja gettimeofday w PostgreSQL dla windows:

/*
 * 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;
}
 20
Author: dicroce,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2017-09-28 23:54:08

A może:

unsigned long start = GetTickCount();
// stuff that needs to be timed
unsigned long delta = GetTickCount() - start;

GetTickCount() nie jest zbyt precyzyjny, ale prawdopodobnie zadziała dobrze. Jeśli widzisz wiele interwałów 0, 16 lub 31 milisekund, spróbuj pomiaru czasu w dłuższych interwałach lub użyj bardziej precyzyjnej funkcji, takiej jak timeGetTime.

To co zwykle robię to:

unsigned long deltastack;
int samples = 0;
float average;

unsigned long start = GetTickCount();
// stuff that needs to be timed
unsigned long delta = GetTickCount() - start;

deltastack += delta;
if (samples++ == 10)
{
   // total time divided by amount of samples
   average = (float)deltastack / 10.f;
   deltastack = 0;
   samples = 0;  
}
 4
Author: knight666,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2009-11-04 20:30:19

W Twoim przypadku użyłbym platformy niezależnej std:: clock

 1
Author: Shay Erlichmen,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2009-11-04 21:08:45

Możesz sprawdzić QueryPerformanceCounter i QueryPerformanceFrequency. Są to Bardzo Wysoka rozdzielczość-do jednego ticka na dziesięć cykli w niektórych timerach sprzętowych.

 1
Author: Puppy,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2012-06-05 22:34:33