- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
因此,我试图将格式为“2000-01-01”的日期转换为表示自某个任意来源(例如 1900/01/01)以来的天数的整数,以便我可以将它们视为整数索引。为此,我编写了一个转换函数,它在 Windows XP 下的 MinGW 上运行良好,但在 Vista 下运行不佳。我添加了一些日志记录代码:
int dateStrToInt(string date) {
int ymd[3];
tm tm1, tm0;
istringstream iss(date);
string s;
for (int i = 3; i; --i) {
getline(iss, s, '-');
ymd[3-i] = str2<int>(s);
}
cout << ymd[0] << ' ' << ymd[1] << ' ' << ymd[2] << ' ' << endl;
tm1.tm_year = ymd[0] - 1900;
tm1.tm_mon = ymd[1] - 1;
tm1.tm_mday = ymd[2];
time_t t1 = mktime(&tm1);
tm0.tm_year = 0;
tm0.tm_mon = 0;
tm0.tm_mday = 0;
time_t t0 = mktime(&tm0);
//cout << "times: " << mktime(&origin) << ' ' << mktime(&time) << endl;
cout << "times: " << t0 << ' ' << t1 << endl;
cout << "difftime: " << difftime(t1, t0) << endl;
return difftime(mktime(&tm1), mktime(&tm0)) / (60*60*24);
}
int i = dateStrToInt("2000-01-01");
我从中得到的输出是
2000 1 1
times: -1 -1
difftime: 0
这显然是错误的。我该怎么办?
编辑:正如下面的答案所说,1970 年之前的年份似乎存在问题。为了避免这种情况,我已经手动编写了自己的日计数函数:
int dateStrToInt(string date) {
int ymd[3];
istringstream iss(date);
string s;
for (int i = 0; i < 3; ++i) {
getline(iss, s, '-');
ymd[i] = str2<int>(s);
}
const static int cum_m_days[12] = {0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334};
int year = ymd[0]+10000, month = ymd[1], day = ymd[2];
int days = year*365 + cum_m_days[month-1] + day;
// handle leap years
if (month <= 2)
--year;
days = days + (year/4) - (year/100) + (year/400);
return days;
}
最佳答案
将所有其他 struct tm
字段保留为默认值(在本例中为随机)不一定是个好主意。
该标准并没有明确说明在调用 mktime
之前需要设置哪些字段,但它确实说明它设置了 tm_wday
和 tm_yday
基于其他字段,并且那些其他字段不限于有效。
标准做展示的一件事是示例代码,它设置了所有字段,除了上面提到的那两个,所以这就是我的目标。
尝试更改计算时间的段:
tm1.tm_year = ymd[0] - 1900;
tm1.tm_mon = ymd[1] - 1;
tm1.tm_mday = ymd[2];
time_t t1 = mktime(&tm1);
tm0.tm_year = 0;
tm0.tm_mon = 0;
tm0.tm_mday = 0;
time_t t0 = mktime(&tm0);
类似于:
// Quick and dirty way to get decent values for all fields.
time_t filled_in;
time (&filled_in);
memcpy (&tm1, localtime ( &filled_in ), sizeof (tm1));
memcpy (&tm0, &tm1, sizeof (tm0));
// Now do the modifications to relevant fields, and calculations.
tm1.tm_year = ymd[0] - 1900;
tm1.tm_mon = ymd[1] - 1;
tm1.tm_mday = ymd[2];
time_t t1 = mktime(&tm1);
tm0.tm_year = 0;
tm0.tm_mon = 0;
tm0.tm_mday = 0;
time_t t0 = mktime(&tm0);
此外,在 XP 下使用 CygWin 进行的一些实验导致 mktime
似乎总是为 struct tm
结构返回 -1,其中 tm_year
是少于两个。这是否是一个真正的错误值得怀疑,因为我经常发现实现并不总是支持纪元(1970 年 1 月 1 日)之前的日期。
某些 UNIX 确实允许您指定小于 70 的 tm_year
值,并且它们通常可以使用 time_t
的这些“负”值来访问 1970 年以前的年份。
但是,由于标准并没有真正涉及到这一点,所以它留给了实现。在 7.23.1/4 中可以找到继承到 C++ 的 C99 标准(可能还有更早的迭代)的相关位:
The range and precision of times representable in clock_t and time_t are implementation-defined.
最安全的做法是使用纪元开始后的日期作为基线日期。这显示在以下代码中:
#include <iostream>
#include <sstream>
#include <string>
#include <ctime>
#include <cstring>
#include <cstdlib>
int dateStrToInt(std::string date) {
int ymd[3];
tm tm1, tm0;
std::istringstream iss(date);
std::string s;
// Test code.
ymd[0] = 2000; ymd[1] = 1; ymd[2] = 1;
std::cout << ymd[0] << ' ' << ymd[1] << ' ' << ymd[2] << ' ' << std::endl;
time_t filled_in;
time (&filled_in);
std::memcpy (&tm0, localtime ( &filled_in ), sizeof (tm0));
std::memcpy (&tm1, &tm0, sizeof (tm1));
tm1.tm_year = ymd[0] - 1900;
tm1.tm_mon = ymd[1] - 1;
tm1.tm_mday = ymd[2];
time_t t1 = mktime(&tm1);
tm0.tm_year = 1970 - 1900; // Use epoch as base date.
tm0.tm_mon = 0;
tm0.tm_mday = 1;
time_t t0 = mktime(&tm0);
std::cout << "times: " << t0 << ' ' << t1 << std::endl;
std::cout << "difftime: " << difftime(t1, t0) << std::endl;
return difftime(mktime(&tm1), mktime(&tm0)) / (60*60*24);
}
int main (void) {
int i = dateStrToInt("2000-01-01");
double d = i; d /= 365.25;
std::cout << i << " days, about " << d << " years." << std::endl;
return 0;
}
这会输出预期的结果:
2000 1 1
times: 31331 946716131
difftime: 9.46685e+08
10957 days, about 29.9986 years.
作为附录,POSIX 有 this说:
自纪元以来 4.14 秒
一个近似于自纪元以来经过的秒数的值。协调世界时名称(以秒 (tm_sec)、分钟 (tm_min)、小时 (tm_hour)、自当年 1 月 1 日以来的天数 (tm_yday) 和减去 1900 的日历年 (tm_year) 指定)与根据以下表达式,时间表示为自纪元以来的秒数。
如果年份<1970 或值为负,则关系未定义。如果年份 >=1970 且值为非负数,则根据 C 语言表达式,该值与协调世界时名称相关,其中 tm_sec、tm_min、tm_hour、tm_yday 和 tm_year 均为整数类型:
tm_sec + tm_min*60 + tm_hour*3600 + tm_yday*86400 +
(tm_year-70)*31536000 + ((tm_year-69)/4)*86400 -
((tm_year-1)/100)*86400 + ((tm_year+299)/400)*86400
一天中的实际时间与自纪元以来秒数的当前值之间的关系未指定。
自 Epoch 以来秒值的任何更改如何与当前实际时间对齐以符合所需的关系是实现定义的。以纪元以来的秒数表示,每一天都应准确计算 86400 秒。
注意:表达式的最后三项在从大纪元以来第一个闰年开始的闰年之后的每一年中添加一天。第一项从 1973 年开始每 4 年增加一天,第二项从 2001 年开始每 100 年减少一天,第三项从 2001 年开始每 400 年增加一天。公式中的除法是整数除法;也就是说,余数被丢弃,只留下整数商。
换句话说(请参阅“如果年份是 <1970 年或值为负数,则关系未定义”),使用 1970 年之前的日期需要您自担风险。
关于C++:Windows Vista 上 MinGW 下的 mktime 错误?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6937606/
我想从 string 得到一个 unix 时间戳包含形式 YYYYMMDDThhmmss+TZ 的时间表示. 为此,我将字符串转换为 struct tm然后使用 mktime将其转换为unix时间戳。
这是一个 PHP 问题。当我测试这段代码时 echo date("d-m-y h:i:S"); 在我的本地服务器和我的托管网站(这两个具有不同的时区)上,它们返回不同的日期时间值,正如预期的那样。但是
我想在微 Controller 上使用 mktime(或者至少使用 32 位宽的时间戳)。我从 avr libc source files 添加了所需的文件(在 Atmel Studio 7 中,时间
我正在将 azure-sdk 用于 python,特别是 azure-servicebus 0.50.0 from azure.servicebus import QueueClient servic
我在 mktime() 中遇到了年份问题。 每次我将低于 1970 年的年份传递到 struct tm 中,然后运行 mktime() 函数来转换我的结构时,都会失败(返回 -1) 。 有人知道为
我正在开发按含义分隔字符的程序。目前我遇到了 konverzia 函数的问题。它应该在日期字符串中添加日期名称(星期三、星期四、星期五...),但它根本不在乎。日期必须类似于周三 2012-02-01
我正在尝试将以 HHMMSS.SS,DD,MM,YYYY 格式给出的时间转换为 unix 时间。问题是,然后调用 mktime,返回相同的 time_t。 ConvertTime(std::strin
我有一个日期操作类。当我尝试用相同的值初始化它时,第一个实例工作正常,但所有其他实例每次都返回不同的值。我将日期保存在 time_t DATE 变量中。 它的构造函数: CDate (int y,in
在一个简单的测试中,获取从 de Unix Epoch (01/01/1970 00:00:00 UTC) 到 2019 年 2 月 22 日的耗时(以秒为单位),它返回了大量的 1844674407
我正在尝试将一周重复添加到包含日期信息的事件结构中。我这样做是为了在某个时间之前创建一个事件的多个实例。我的 mktime 函数出现段错误,其中 full_time = mktime(&caltime
我想用澳大利亚/悉尼时区构造一个struct tm,所以我首先使用: setenv("TZ","Australia/Sydney",1); tzset() 然后我将struct tm设置为: stru
这是将 Fri Jan 1 00:00:00 IST 1970 转换为 EPOCH 的代码段 memset(&Date_st,0,sizeof(struct tm)); Date_st.tm_ye
我正在尝试将用户输入日期添加到现有时间结构中,但我收到了一个我不明白的错误。当我试图修改月份日期时,编译器告诉我工作日超出范围。代码贴在下面。 struct tm date; int m, d, y,
这个问题在这里已经有了答案: C++ mktime returning random dates (3 个答案) 关闭 6 年前。 我准备了以下结构: struct tm tDepartureTim
在尝试编写返回比给定时间少 24 小时的代码时,mktime() 显示不一致的输出。我的计算类似于:current_time(GMT) - 86400 应该返回正确的值。我们需要做的就是根据输入的时间
这个问题已经有答案了: Python | mktime overflow error (2 个回答) 已关闭 6 年前。 使用 Python 时 time模块我收到此错误: OverflowError
继续 my attempt to create a DateTime class ,我试图在我的函数中存储“纪元”时间: void DateTime::processComponents(int mo
我现在正在使用 gcc 编译器在我的 RedHat Linux 机器上编写一个 C 程序,它将接受两个日期作为输入并计算它们之间的差异。 但是,我发现 mktime() 有一个奇怪的行为,如下面的代码
我做的一个网站突然出现以下错误,到目前为止一直运行良好: A PHP Error was encountered Severity: Warning Message: mktime() [functi
我正在研究一些与时间相关的功能,我选择始终使用 UTC 时间并将时间戳存储为整数以保持一致性。 但是,我注意到当我使用mktime 时,当前设置的时区似乎对mktime 的返回值有影响。从文档中我了解
我是一名优秀的程序员,十分优秀!