gpt4 book ai didi

c++ - 本地时间为在 cygwin shell 上运行的 Windows 程序返回 GMT

转载 作者:可可西里 更新时间:2023-11-01 18:25:44 25 4
gpt4 key购买 nike

考虑以下代码:

time_t t;
t = time( NULL );
elog << "timezone: " << getenv( "TZ" )
<< ", current local time: " << asctime( localtime( &t ));

如果我使用 MSVC 构建这段代码,并在 Windows DOS shell 下运行它,我会得到正确的本地时间:

timezone: , current local time: Wed Jul 25 13:05:08 2012

但是如果我在 bash 之类的 cygwin shell 下运行相同的程序,此代码将返回 GMT!

timezone: America/New_York, current local time: Wed Jul 25 18:05:08 2012

如果我在 Linux 或 OsX 中运行这个程序,它也会返回正确的本地时间。

为什么?

@Update:现在一年过去了,我发现我在下面给出的答案并不总是有效。

对于某些程序来说,取消设置 TZ 似乎并不总是有效。我不知道为什么。但是有一个麻烦的解决方法。基本上,在取消设置 TZ 之后,您必须检查本地时间是否确实不再返回 GMT,但前​​提是您实际上不在 GMT 时区,并在调用 localtime() 或时计算对 time_t 的手动调整制作时间()

u64 localTimeOffset = 0;

static void unsetTz()
{
static bool unsetTZ = false;
if ( !unsetTZ )
{
putenv( "TZ=" );
unsetTZ = true;

// unsetting TZ does not always work. So we have to check if it worked
// and make a manual adjustment if it does not. For long running programs
// that may span DST changes, this may cause the DST change to not take
// effect.
s32 tzOffset = getTzOffset();

if ( tzOffset )
{
static char timeBuf[48];
char* s = &(timeBuf[0]);
struct tm* timeInfoGMT;
struct tm* timeInfoLocal;

time_t zero = 86400;
timeInfoGMT = gmtime( &zero );
u32 GMTHour = timeInfoGMT->tm_hour;

timeInfoLocal = localtime( &zero );
u32 localHour = timeInfoLocal->tm_hour;

if ( localHour == GMTHour )
{
// unsetting tz failed. So we have to make a manual adjustment
localTimeOffset = tzOffset * 60 * 60;
}
}
}
}

s32 getTzOffset()
{
TIME_ZONE_INFORMATION tzInfo;
GetTimeZoneInformation( &tzInfo );
s32 tz = ( tzInfo.Bias / 60 );
return tz;
}

调用本地时间:

  time_t t = getAtTimeFromSomewhere();
t -= localTimeOffset;
timeInfo = localtime( &t );

调用 maketime:

 struct tm timestr;
makeATMFromAStringForExample( time, timestr );
time_t timet = mktime( &timestr );
timet += localTimeOffset;

好时光。

最佳答案

这花了我一些时间才弄清楚,我希望它对其他人有用。

localtime 等 POSIX 函数将使用环境变量 TZ 来确定要使用的时区。如果未设置 TZ,它将使用系统的默认时区。

如果我在 Linux 或 OS X 下运行,TZ 设置正确并且一切正常。如果我在 Windows 的 shell 中运行这个程序,TZ 没有设置,所以函数返回操作系统的默认时区,这再次产生正确的结果。

如果我在 Cygwin shell 中运行,TZ 已设置 - 但由于我使用 MSVC 构建程序,使用 MSVC 自己的 stdc 库 - 它无法解释 Cygwin 的 TZ 变量.所以它默认为 GMT。

如果该程序是在 Cygwin 下使用 GCC 构建的,我敢打赌它会在 Cygwin shell 中正常运行。

所以答案是确保在调用 POSIX 时间函数(如 localtime())的程序中,如果您希望时间函数在 Cygwin shell 下正常工作,您必须取消设置 TZ.

我是这样做的:

void getLocalTime()
{
#ifdef WIN32
static bool unsetTZ = false;
if ( !unsetTZ )
{
putenv( "TZ=" );
unsetTZ = true;
}
#endif // !WIN32

time_t t;
t = time( NULL );
elog << "timezone: " << getenv( "TZ" )
<< ", current local time: " << asctime( localtime( &t ));
}

关于c++ - 本地时间为在 cygwin shell 上运行的 Windows 程序返回 GMT,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11655003/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com