Visual 2015上的std :: get_time在错误的日期不会失败

时间:2017-04-05 15:48:18

标签: c++ c++11 visual-c++ visual-studio-2015 std

我正在使用Visual Studio 2015在Windows上执行以下代码。基本上我使用std::get_time来解析日期,但是当日期无效时,例如,大于31的日期,它似乎不会设置流上的失败位。

我在Ubuntu上使用g ++ 5.4.0尝试了这个,它设置了失败位并打印“Parsing failed!”。这是Windows上的错误还是我做错了什么。

提前致谢!

std::string date = "2019-2-37 23:00:00"; // day (37) is wrong. 
std::string format = "%Y-%m-%d %H:%M:%S";
std::tm tm_object{};
std::istringstream input(date);        
input.imbue(std::locale(std::setlocale(LC_ALL, nullptr)));
input >> std::get_time(&tm_object, format.c_str());
if (input.fail()) 
{
    std::cout << "Parsing failed!";
}
else
{
    std::cout << "Parsing ok!\n";
    std::cout << "Day is : " << tm_object.tm_mday;
}

2 个答案:

答案 0 :(得分:3)

您可以在Windows上使用Howard Hinnant's free, open-source header-only datetime library为您提供所需的行为。语法略有不同,使用起来更容易,并且与<chrono>兼容。它比C ++标准的时间解析部分更好地记录和指定。

#include "date.h"
#include <iostream>
#include <sstream>
#include <string>

int
main()
{
    std::string date = "2019-2-37 23:00:00"; // day (37) is wrong. 
    std::string format = "%Y-%m-%d %H:%M:%S";
    date::sys_seconds tm_object{};
    std::istringstream input(date);        
    input >> date::parse(format, tm_object);
    if (input.fail()) 
    {
        std::cout << "Parsing failed!";
    }
    else
    {
        std::cout << "Parsing ok!\n";
        date::year_month_day ymd = date::floor<date::days>(tm_object);
        std::cout << "Day is : " << ymd.day();
    }
}

输出:

Parsing failed!

此库也适用于g ++ 5.4.0和clang。

您还可以将format简化为"%F %T",如果需要,也可以使用亚秒精度。

答案 1 :(得分:2)

欢迎来到标准灰色区域的世界! std::get_time(各种重载)在第22章本地化库和27标准输入/输出库中指定。

AFAIK,标准要求:

  • 如果输入字符串正确描述了有效格式的有效日期,则相应地填充std::tm对象
  • 如果无法解析输入字符串,则应设置输入字符串上的失败位

但是在这里,可以解析输入字符串,并且一些实现可以假设行军32只是4月1日。换句话说,标准没有指定实现应该对输入值做什么控制。非数字日应该给出错误,但这取决于实现。