Lua 中的 UTC 日期前后转换问题

我在将一个 lua 日期转换为时间戳,然后从时间戳获取原始日期方面遇到了问题,对于非 UTC 日期可以正常工作,但是对于 UTC 日期却不行。

目前我示例代码如下:

local dt1 = os.date( "*t" );
print( dt1.hour );

local dt2 = os.date( "*t", os.time( dt1 ) );
print( dt2.hour );

print( "-=-=-" );

local dt1 = os.date( "!*t" );
print( dt1.hour );

local dt2 = os.date( "!*t", os.time( dt1 ) );
print( dt2.hour );

local dt2 = os.date( "*t", os.time( dt1 ) );
print( dt2.hour );

它的输出结果为:

12
12
-=-=-
10
9
11

所以,在使用 os.time( os.date( "!*t" ) ) 获取时间戳之后,我不知道如何获取原始日期,我做错了什么?

点赞
用户1979882
用户1979882

我发现处理日期、时间和时区的唯一解决方法是使用自定义的 c-build 方法:

例如,要以指定格式检索字符串,我使用以下方法:

static int idateformat(lua_State *L) {

    time_t t0 = (time_t) lua_tonumber(L,1);
    const char* ptrn = lua_tostring(L,2);
    const char *tz_value = lua_tostring(L,3);

    int isinenv = 0;
    if(getenv("TZ") != NULL){
        isinenv = 1;
    }
    char old_tz[64];
    if (isinenv == 1) {
        strcpy(old_tz, getenv("TZ"));
    }
    setenv("TZ", tz_value, 1);
    tzset();
    char new_tz[64];
    strcpy(new_tz, getenv("TZ"));

    struct tm *lt = localtime(&t0);
    //"%Y-%m-%d %H:%M:%S"

    char buffer[256];
    strftime(buffer, sizeof(buffer), ptrn, lt);

    if (isinenv == 1) {
        setenv("TZ", old_tz, 1);
        tzset();
    }
    //printf("%ld = %s (TZ=%s)\n", (long)t0, buffer, new_tz);
    lua_pushstring(L, buffer);
    return 1;
}

我只会建议这种方式。

2017-03-28 10:58:27
用户107090
用户107090

你没有做错什么。

Lua 使用 ISO C 和 POSIX 日期时间函数。特别地,os.time 使用 mktime,它仅根据环境变量 TZ 中当前时区设置来解释输入结构。不幸的是,这些标准没有提供一种根据 GMT/UTC 解释输入结构的函数。

2017-03-28 11:03:20
用户1847592
用户1847592

在 Lua 中使用“日期表”

假设 dt 是一个“日期表”。

例如,由 os.date(“*t”) 返回的值是一个“日期表”。


如何标准化“日期表”

例如,在当前时间添加 1.5 小时后

local dt = os.date(“*t”); dt.min = dt.min + 90

您需要对表字段进行标准化。

function normalize_date_table(dt)
   return os.date("*t", os.time(dt))
end

此函数将返回等效于其参数“dt”的新日期表,无论“dt”的内容的含义如何:它是否包含本地时间或 GMT 时间。


如何将 Unix 时间转换为“本地日期表”

dt = os.date("*t", ux_time)

如何将 Unix 时间转换为“GMT 日期表”

dt = os.date(“!*t” ,ux_time)

如何将“本地日期表”转换为 Unix 时间

ux_time = os.time(dt)

如何将“GMT 日期表”转换为 Unix 时间

 - 对于这种转换,我们需要预先计算出“zone_diff”值
local tmp_time = os.time()
local d1 = os.date("*t",  tmp_time)
local d2 = os.date(“!*t” ,tmp_time)
d1.isdst = false
local zone_diff = os.difftime(os.time(d1), os.time(d2))
-- zone_diff 值只能在程序开始时计算一次。

-- 现在我们可以执行转换 (dt -> ux_time):
dt.sec = dt.sec + zone_diff
ux_time = os.time(dt)
2017-03-28 12:12:35