用tm加减时间
Adding and subtracting time with tm
假设我将 tm 中的时间设置为 23:00:00
ptm->tm_hour = 23; ptm->tm_min = 0; ptm->tm_sec = 0;
我想允许用户从中减去时间
ptm->tm_hour -= hourinput; ptm->tm_min -= minuteinput; ptm->tm_sec -= secondinput;
如果用户减去 0 小时 5 分 5 秒,而不是显示为 22:54:55,它将显示为 23:-5:-5。
我想我可以做一堆 if 语句来检查 ptm 是否低于 0 并解决这个问题,但是有没有更有效的方法来获得正确的时间?
是的,您可以为此使用 std::mktime
。它不只是将 std::tm
转换为 std::time_t
,如果某些字段超出范围,它还会修复 tm
。考虑这个例子,我们将当前时间加上 1000 秒。
#include <iostream>
#include <iomanip> // put_time
#include <ctime>
int main(int argc, char **argv) {
std::time_t t = std::time(nullptr);
std::tm tm = *std::localtime(&t);
std::cout << "Time: " << std::put_time(&tm, "%c %Z") << std::endl;
tm.tm_sec += 1000; // the seconds are now out of range
//std::cout << "Time in 1000 sec" << std::put_time(&tm, "%c %Z") << std::endl; this would crash!
std::mktime(&tm); // also returns a time_t, but we don't need that here
std::cout << "Time in 1000 sec: " << std::put_time(&tm, "%c %Z") << std::endl;
return 0;
}
我的输出:
Time: 01/24/19 09:26:46 W. Europe Standard Time
Time in 1000 sec: 01/24/19 09:43:26 W. Europe Standard Time
如您所见,时间从 09:26:46
变为 09:43:26
。
这是使用 Howard Hinnant's date library 的另一种方法,它正在进入 C++2a。
#include <iostream>
#include "date/date.h"
using namespace std::chrono_literals;
// Time point representing the start of today:
auto today = date::floor<date::days>(std::chrono::system_clock::now());
auto someTp = today + 23h; // Today, at 23h00
auto anotherTp = someTp - 5min - 5s; // ... should be self-explanatory now :)
std::cout << date::format("%b-%d-%Y %T\n", anotherTp);
如果要通过用户界面公开时间点的操作,编译时构造23h
、5min
等当然不可用。这些文字构造 std::chrono::duration
对象,因此您需要一种机制将用户输入转换为等效实例。
假设我将 tm 中的时间设置为 23:00:00
ptm->tm_hour = 23; ptm->tm_min = 0; ptm->tm_sec = 0;
我想允许用户从中减去时间
ptm->tm_hour -= hourinput; ptm->tm_min -= minuteinput; ptm->tm_sec -= secondinput;
如果用户减去 0 小时 5 分 5 秒,而不是显示为 22:54:55,它将显示为 23:-5:-5。
我想我可以做一堆 if 语句来检查 ptm 是否低于 0 并解决这个问题,但是有没有更有效的方法来获得正确的时间?
是的,您可以为此使用 std::mktime
。它不只是将 std::tm
转换为 std::time_t
,如果某些字段超出范围,它还会修复 tm
。考虑这个例子,我们将当前时间加上 1000 秒。
#include <iostream>
#include <iomanip> // put_time
#include <ctime>
int main(int argc, char **argv) {
std::time_t t = std::time(nullptr);
std::tm tm = *std::localtime(&t);
std::cout << "Time: " << std::put_time(&tm, "%c %Z") << std::endl;
tm.tm_sec += 1000; // the seconds are now out of range
//std::cout << "Time in 1000 sec" << std::put_time(&tm, "%c %Z") << std::endl; this would crash!
std::mktime(&tm); // also returns a time_t, but we don't need that here
std::cout << "Time in 1000 sec: " << std::put_time(&tm, "%c %Z") << std::endl;
return 0;
}
我的输出:
Time: 01/24/19 09:26:46 W. Europe Standard Time
Time in 1000 sec: 01/24/19 09:43:26 W. Europe Standard Time
如您所见,时间从 09:26:46
变为 09:43:26
。
这是使用 Howard Hinnant's date library 的另一种方法,它正在进入 C++2a。
#include <iostream>
#include "date/date.h"
using namespace std::chrono_literals;
// Time point representing the start of today:
auto today = date::floor<date::days>(std::chrono::system_clock::now());
auto someTp = today + 23h; // Today, at 23h00
auto anotherTp = someTp - 5min - 5s; // ... should be self-explanatory now :)
std::cout << date::format("%b-%d-%Y %T\n", anotherTp);
如果要通过用户界面公开时间点的操作,编译时构造23h
、5min
等当然不可用。这些文字构造 std::chrono::duration
对象,因此您需要一种机制将用户输入转换为等效实例。