入力:日付とオプションの時刻を含む文字列。別の表現がいいですが、必要です。文字列はユーザー指定であり、不正な形式である可能性があります。例:
"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);
boost::date_time
は必要なUTCオフセットを提供できると思いますが、どうすればよいかわかりません。
ブーストで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
_
おそらく_#include <boost/date_time/c_time.hpp>
_のboost::posix_time::c_time::localtime()
を使用して、入力が現在のタイムゾーンにあると仮定してこの変換を実行できますが、それはかなり一貫性がありません。たとえば、私にとっては、結果は今日と翌月、夏時間が終了します。
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チュートリアルがあります。
C-styleが受け入れられる場合: strptime ()が方法です。フォーマットを指定でき、ロケールを考慮できるためです。
tm brokenTime;
strptime(str.c_str(), "%Y-%m-%d %T", &brokenTime);
time_t sinceEpoch = timegm(brokenTime);
可能であれば、異なるレイアウトを戻り値で確認する必要があります。システムクロックを確認してタイムゾーンを追加する必要があります(localtime_r()とtime()、tm_zone)
最も簡単で移植可能なソリューションは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
で 情報を参照 にしてください。