7

考虑以下代码:

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;

美好时光。

4

3 回答 3

11

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

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

如果我在 Linux 或 OS X 下运行,TZ设置正确并且一切正常。如果我在 Windows 上的 shell 中运行此程序,TZ则未设置,因此该函数返回操作系统的默认时区,这又会产生正确的结果。

如果我在 Cygwin shell 中运行,TZ则设置 - 但由于我使用 MSVC 构建程序,使用 MSVC 自己的 stdc 库 - 它无法解释 Cygwin 的TZ变量。所以它默认为格林威治标准时间。

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

所以答案是确保在调用 POSIX 时间函数的程序中localtime(),如果您希望时间函数在 Cygwin 外壳下正常工作,则必须取消设置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 ));
}
于 2012-07-25T17:15:42.233 回答
1

在 Cygwin shell 中使用export TZ=""可能是最好的方法。


@更新嗨,图纳基。感谢您的编辑!

于 2016-09-03T04:04:43.167 回答
0

尝试在 struct tm 中设置tm_isdst标志。

如果夏令时有效,夏令时标志 (tm_isdst) 大于零,如果夏令时无效,则为零,如果信息不可用,则小于零。

如果我没记错的话,在 Windows 中它必须是 -1 才能正常工作。

于 2012-07-25T17:26:10.890 回答