FILETIME в __int64

Как правильно преобразовать структуру FILETIME в __int64? Не могли бы вы рассказать мне?


person akif    schedule 14.10.2009    source источник


Ответы (6)


Я не думаю, что вы должны: «Не приводить указатель на структуру FILETIME к значению ULARGE_INTEGER* или __int64*, потому что это может вызвать ошибки выравнивания в 64-разрядной Windows».

Источник.

Если бы вы действительно хотели, это было бы что-то вроде:

__int64 to_int64(FILETIME ft)
{
    return static_cast<__int64>(ft.dwHighDateTime) << 32 | ft.dwLowDateTime;
}

FILETIME ft = // ...
__int64 t = to_int64(ft);

Но что-то вроде:

FILETIME ft = // ...
__int64 t = *reinterpet_cast<__int64*>(&ft);

Плохо.

person GManNickG    schedule 14.10.2009
comment
Как насчет memcpy(‹__int64›, &FILETIME, 8); ? Это безопасно? - person akif; 14.10.2009
comment
Мне было бы memcpy сложнее понять, но я уверен, что это сработает. Компилятор, вероятно, может оптимизировать сдвиг или лучше. - person GManNickG; 14.10.2009
comment
Действительно, понять было сложнее, но пока получилось :) Спасибо - person akif; 15.10.2009

Нет необходимости возвращаться к загадочным конструкциям с использованием побитового ИЛИ. В Windows API есть все необходимое для этого.

unsigned __int64    convert( const FILETIME & ac_FileTime )
{
  ULARGE_INTEGER    lv_Large ;

  lv_Large.LowPart  = ac_FileTime.dwLowDateTime   ;
  lv_Large.HighPart = ac_FileTime.dwHighDateTime  ;

  return lv_Large.QuadPart ;
}  

Или, если вы хотите напрямую перейти к __int64.

__int64 convert_to_int64( const FILETIME & ac_FileTime )
{
  return static_cast< __int64 > ( convert( ac_FileTime ) ) ;
}
person QBziZ    schedule 12.12.2011
comment
В наши дни (начиная с C++11) его можно просто записать как ULARGEINTEGER{ ft.dwLowDateTime, ft.dwHighDateTime }.QuadPart. Возможно, не более читабельно, однако вам не нужны дополнительные функции/макросы. - person klaus triendl; 08.11.2018

Пытаться

(__int64(filetime.dwHighDateTime)<<32) | __int64(filetime.dwLowDateTime)
person Martin B    schedule 14.10.2009

Конечно, вы можете просто передать __int64, приведенный к файловому времени, следующим образом *(FILETIME*)&int64Val. Это будет нормально работать под Visual C++.

ie

__int64 createTime = 0;
__int64 accessTime = 0;
__int64 writeTime = 0;
GetFileTime( hFile, *(FILETIME*)&createTime, *(FILETIME*)&accessTime, *(FILETIME*)&writeTime );
person Goz    schedule 14.10.2009
comment
Не приводите указатель на структуру FILETIME к значению ULARGE_INTEGER* или __int64*, поскольку это может привести к ошибкам выравнивания в 64-разрядной версии Windows. docs.microsoft.com/en-us/ окна/рабочий стол/api/minwinbase/ - person MrTux; 09.03.2019
comment
@MrTux: в ссылке сказано, что вы не должны выполнять приведение FROM к структуре FILETIME. 2 значения очень разные. По сути, FILETIME может иметь выравнивание по 4 байтам (или, может быть, даже по 2), поэтому приведение к __int64 вызовет проблемы с выравниванием, поскольку __int64 обязательно выравнивается по 8 байтам. Однако в приведенной выше функции я передаю 8-байтовый выровненный __int64, и Windows записывает в него. Он не меняет адрес указателя только потому, что это __int64. Это сломало бы все виды вещей. - person Goz; 11.03.2019
comment
Мне очень нравится этот ответ, потому что он имеет смысл в свете реального значения рекомендаций по документации; и описание Раймонда Чена. Спасибо за дельный совет! - person Mike Kaganski; 09.07.2019

вы можете попробовать следующий код. код из проекта хром

template <class Dest, class Source>
inline Dest bit_cast(const Source& source) {
    Dest dest;
    memcpy(&dest, &source, sizeof(dest));
    return dest;
}

//FILETIME to __int64

__int64 FileTimeToMicroseconds(const FILETIME& ft) {
    return bit_cast<__int64, FILETIME>(ft) / 10;
}

void MicrosecondsToFileTime(__int64 us, FILETIME* ft) {
    *ft = bit_cast<FILETIME, __int64>(us * 10);
}

int _tmain(int argc, _TCHAR* argv[])
{
    __int64 nTmpUint64 = 13060762249644841;

    time_t unixtime;
    FILETIME nTmpFileTm;
    MicrosecondsToFileTime(nTmpUint64,&nTmpFileTm);

    return 0;
}
person Stonexin    schedule 18.11.2014

У меня была точно такая же проблема, погуглил и пришел сюда. Но я также нашел полезную страницу поддержки Microsoft по адресу
https://support.microsoft.com/en-gb/help/188768/info-working-with-the-filetime-structure

В нем говорится:


Выполнение арифметических операций с временем файла

Часто бывает необходимо выполнить простую арифметику времени файла. Например, вам может понадобиться узнать, когда файлу исполнилось 30 дней. Чтобы выполнить арифметические действия над временем файла, необходимо преобразовать FILETIME в четверное слово (64-разрядное целое число), выполнить арифметические действия, а затем преобразовать результат обратно в FILETIME.

Предполагая, что ft является структурой FILETIME, содержащей время создания файла, следующий пример кода добавляет ко времени 30 дней:

   ULONGLONG qwResult;

   // Copy the time into a quadword.
   qwResult = (((ULONGLONG) ft.dwHighDateTime) << 32) + ft.dwLowDateTime;

   // Add 30 days.
   qwResult += 30 * _DAY;

   // Copy the result back into the FILETIME structure.
   ft.dwLowDateTime  = (DWORD) (qwResult & 0xFFFFFFFF );
   ft.dwHighDateTime = (DWORD) (qwResult >> 32 );

Изменить: я понимаю, что это просто подтверждает некоторые другие ответы, но я подумал, что стоит добавить для уточнения.

person InsanityPants    schedule 31.01.2020