2016-03-31 109 views
4

我使用yyyy-MM-dd'T'HH:mm:ss.SSSXXX格式字符串與icu :: SimpleDateFormat使用ICU解析ISO 8601日期時間(可選小數秒)

雖然它適用於具有小數位數的日期時間字符串。例子:

2016-03-31T15:04:32.049Z

2016-03-31T15:04:32.05Z

2016-03-31T15:04:32.3Z

它不解析沒有小數秒的字符串(例如2016-03-31T15:08:51Z),返回錯誤代碼U_ILLEGAL_ARGUMENT_ERROR

我嘗試了一些其他格式組合無濟於事:一些失敗,錯誤代碼,其他人忽略毫秒。

ICU是否支持解析可選小數秒?

+0

此[自由,開放源碼的,報頭僅C++ 11/14/17文庫(HTTPS://howardhinnant.github .io/date/date.html)將使用語法字符串'「%FT%TZ」'來完成。只需解析成一個'std :: chrono :: system_clock :: time_point'。 –

+0

我期望最終的解決方案將涉及嘗試使用多個解析器進行解析......首先使用嘗試指定秒和下一個解析器的解決方案。但是,當我嘗試這種方法時,我一直在收到U_ILLEGAL_ARGUMENT_ERROR,所以我不確定我在做什麼錯。我預計這些問題是相似的,所以我正在爲此付出代價。我使用的格式來自https://www.w3.org/TR/NOTE-datetime – markt1964

回答

1

通過ICU的documentation,我沒有找到任何方式如何在模式規範中輸入可選部分。

但是,這個問題可以在應用程序級別以許多不同的方式解決。一種這樣的方式(簡化的)是使用多個日期時間解析器如下:

#include <iostream> 
#include <vector> 
#include "unicode/datefmt.h" 
#include "unicode/smpdtfmt.h" 

int main() { 

    UErrorCode err(U_ZERO_ERROR); 

    UnicodeString patternWithMilliseconds("yyyy-MM-dd'T'hh:mm:ss.SSSXXX"); 
    UnicodeString patternPlane("yyyy-MM-dd'T'hh:mm:ssX"); 

    // init ICU parsers 
    std::vector<SimpleDateFormat*> parsers = { 
      new SimpleDateFormat(patternWithMilliseconds, err), 
      new SimpleDateFormat(patternPlane, err) 
    }; 

    // get dates to convert 
    std::vector<UnicodeString> dates = { 
      UnicodeString("2016-03-31T15:04:32.049Z"), 
      UnicodeString("2017-10-30T15:05:33Z"), 
    }; 

    SimpleDateFormat resultFormatter(patternWithMilliseconds, err); 

    for(const auto& date : dates) { 

     UDate parsedDate; 
     UnicodeString result; 
     std::string resultConverted; 

     for(const auto& parser : parsers) { 
      err = U_ZERO_ERROR; 
      parsedDate = parser->parse(date, err); 
      if (err <= 0) break; 
     } 

     resultFormatter.format(parsedDate, result); 
     result.toUTF8String(resultConverted); 

     std::cout << resultConverted << std::endl; 
    } 

    for(auto& parser : parsers) { 
     delete parser; 
    } 

    return 0; 
}