使用这些库:
http://howardhinnant.github.io/date/date.html
http://howardhinnant.github.io/date/tz.html
这就是我解决问题的方法。先上代码,再解释:
#include "tz.h"
#include "date.h"
#include <iostream>
int
main()
{
using namespace date;
using namespace std::chrono;
std::cout << "Enter birthday [day month year]:";
int di, mi, yi;
std::cin >> di >> mi >> yi;
if (std::cin.fail())
{
std::cout << "Invalid date\n";
return 1;
}
auto y = year{yi};
if (!y.ok())
{
std::cout << "Invalid year\n";
return 1;
}
auto m = month(mi);
if (!m.ok())
{
std::cout << "Invalid month\n";
return 1;
}
auto d = day(di);
if (!d.ok())
{
std::cout << "Invalid day\n";
return 1;
}
auto birthday = y/m/d;
if (!birthday.ok())
{
std::cout << "Invalid birthday\n";
return 1;
}
auto local_time = current_zone()->to_local(system_clock::now());
auto today = year_month_day{floor<days>(local_time)};
auto age = today.year() - birthday.year();
if (birthday + age > today)
--age;
if (age >= years{18})
std::cout << "full aged at " << age.count() << "\n";
else
std::cout << "not full aged at " << age.count() << "\n";
}
我会先去检查用户输入的有效性。我下面的内容似乎是最低限度的:
- 它必须是整数输入。
- 每个整数输入必须有一个合理的值(例如,月份必须在 [1, 12] 范围内。
- y/m/d 组合必须是有效日期。
一个更强大的程序可能会给用户一些关于他输入的反馈,并给他另一个机会来纠正他的错误。
一旦确定我们有一个有效的生日,我们需要获取当地时区的当前日期。这个:
auto local_time = current_zone()->to_local(system_clock::now());
获取当地时间。
这个本地时间可以转换为当地的年、月和日:
auto today = year_month_day{floor<days>(local_time)};
此计算遵循您的生日从当地午夜开始的习俗,无论您实际出生在一天中的什么时间(以及地球上的哪个地方)。换句话说,一旦当地的年/月/日建立起来,这个问题就与当地的时区无关,甚至与当地的一天中的时间无关。
接下来,计算当前年龄:
auto age = today.year() - birthday.year();
if (birthday + age > today)
--age;
今天的年份和生日之间的差异是年龄的第一个近似值。这个近似值是通过计算你今年生日的日期来改进的。如果今年的生日还在未来,那么按照惯例,我们算作小一岁。如果我们做的事情不那么倾向于法律体系,而更倾向于科学工作,我们可能会以其他方式计算,例如四舍五入到最近的年份(也很容易使用这个库)。
如果生日是 2 月 29 日,上面的代码仍然有效: birthday + age
将导致(75% 的机会)无效日期,例如:feb/29/2015。然而,这个无效的日期将正确地比较大于 2015 年 2 月 28 日和小于 2015 年 3 月 1 日,这正是我们需要的!无效日期是你的朋友,而不是你的敌人——你只需要知道它们的存在以及如何处理它们。
现在报告您的发现是一件简单的事情:
if (age >= years{18})
std::cout << "full aged at " << age.count() << "\n";
else
std::cout << "not full aged at " << age.count() << "\n";