58

入力: 日付とオプションの時刻を含む文字列。異なる表現は素晴らしいですが、必要です。文字列はユーザ​​ーが指定したものであり、形式が正しくない可能性があります。例:

  • "2004-03-21 12:45:33"(これがデフォルトのレイアウトだと思います)
  • "2004/03/21 12:45:33"(オプションのレイアウト)
  • "23.09.2004 04:12:21"(ドイツ語形式、オプション)
  • "2003-02-11"(時間がない場合があります)

必要な出力: エポック (1970/01/01 00:00:00) からの秒数またはその他の固定点。

おまけ: また、ローカル システム時間の UTC オフセットを読み取ることもできます。

入力は、問題のマシンの現地時間であると想定されます。出力は UTC である必要があります。システムは Linux のみです (Debian Lenny と Ubuntu が必要です)。

を使用しようとしましboost/date_timeたが、ドキュメンテーションに頭を悩ませることができないことを認めなければなりません。以下は、システムの現地時間から UTC への必要な変換なしで機能します。

std::string date = "2000-01-01";
boost::posix_time::ptime ptimedate = boost::posix_time::time_from_string(date);
ptimedate += boost::posix_time::hours(Hardcoded_UTC_Offset);// where to get from?
struct tm = boost::posix_time::to_tm(ptimedate);
int64_t ticks = mktime(&mTmTime);

必要な UTC オフセットを提供できると思いますboost::date_timeが、方法がわかりません。

4

4 に答える 4

69

ブーストで 1 桁の月の入力をフォーマットする方法はわかりませんが、2 桁の編集後に行うことができます。

#include <iostream>
#include <boost/date_time.hpp>
namespace bt = boost::posix_time;
const std::locale formats[] = {
std::locale(std::locale::classic(),new bt::time_input_facet("%Y-%m-%d %H:%M:%S")),
std::locale(std::locale::classic(),new bt::time_input_facet("%Y/%m/%d %H:%M:%S")),
std::locale(std::locale::classic(),new bt::time_input_facet("%d.%m.%Y %H:%M:%S")),
std::locale(std::locale::classic(),new bt::time_input_facet("%Y-%m-%d"))};
const size_t formats_n = sizeof(formats)/sizeof(formats[0]);

std::time_t pt_to_time_t(const bt::ptime& pt)
{
    bt::ptime timet_start(boost::gregorian::date(1970,1,1));
    bt::time_duration diff = pt - timet_start;
    return diff.ticks()/bt::time_duration::rep_type::ticks_per_second;

}
void seconds_from_epoch(const std::string& s)
{
    bt::ptime pt;
    for(size_t i=0; i<formats_n; ++i)
    {
        std::istringstream is(s);
        is.imbue(formats[i]);
        is >> pt;
        if(pt != bt::ptime()) break;
    }
    std::cout << " ptime is " << pt << '\n';
    std::cout << " seconds from epoch are " << pt_to_time_t(pt) << '\n';
}
int main()
{
    seconds_from_epoch("2004-03-21 12:45:33");
    seconds_from_epoch("2004/03/21 12:45:33");
    seconds_from_epoch("23.09.2004 04:12:21");
    seconds_from_epoch("2003-02-11");
}

seconds-from-epoch の出力は、日付が UTC であると仮定していることに注意してください。

~ $ ./test | head -2
ptime is 2004-Mar-21 12:45:33
seconds from epoch are 1079873133
~ $ date -d @1079873133
Sun Mar 21 07:45:33 EST 2004

入力が現在のタイム ゾーンにあると仮定すると、おそらくboost::posix_time::c_time::localtime()fromを使用し#include <boost/date_time/c_time.hpp>てこの変換を行うことができますが、かなり一貫性がありません。たとえば、私の場合、今日と夏時間が終了する翌月では結果が異なります。

于 2010-09-24T12:49:02.790 に答える
12

boost::gregorianこれ以上作業をしなくても、必要なものがいくつかあります。

using namespace boost::gregorian;
{
  // The following date is in ISO 8601 extended format (CCYY-MM-DD)
  std::string s("2000-01-01");
  date d(from_simple_string(s));
  std::cout << to_simple_string(d) << std::endl;
}

boost::posix_time here でUTC オフセットを使用する方法の例があります。

date_input_facetと を使用して、カスタム入力文字列形式から日付と時刻を生成できますtime_input_facetこのページには、作業を開始するのに役立つ I/O チュートリアルがあります。

于 2010-09-24T11:04:09.233 に答える
8

c-style が受け入れられる場合: strptime () が適しています。形式を指定でき、ロケールを考慮できるからです。

tm brokenTime;
strptime(str.c_str(), "%Y-%m-%d %T", &brokenTime);
time_t sinceEpoch = timegm(brokenTime);

さまざまなレイアウトを戻り値でチェックする必要があります (可能な場合)。システムクロックをチェックしてタイムゾーンを追加する必要があります (localtime_r() with time(), tm_zone)

于 2010-09-24T11:10:01.687 に答える
3

最も簡単でポータブルなソリューションは、次を使用することscanfです。

int year, month, day, hour, minute, second = 0;
int r = 0;

r = scanf ("%d-%d-%d %d:%d:%d", &year, &month, &day,
           &hour, &minute, &second);
if (r == 6) 
{
  printf ("%d-%d-%d %d:%d:%d\n", year, month, day, hour, minute,
          second);
}
else 
{
    r = scanf ("%d/%d/%d %d:%d:%d", &year, &month, &day,
           &hour, &minute, &second);
    // and so on ...

struct tmを値で初期化し、intそれを に渡してmktime、カレンダー時間を として取得しtime_tます。タイムゾーンの変換については、 の情報を参照gmtimeしてください。

于 2010-09-24T11:02:05.230 に答える