" strptime"在SWI-Prolog

时间:2015-12-17 02:20:33

标签: prolog swi-prolog

如何将原子'2015-12-15T05 PST'转换为时间戳或日期时间?

我已经尝试parse_time/3

?- parse_time('2015-12-15T05 PST', '%Y-%m-%dT%H %Z', Stamp)
false.

format_time/3

?- format_time('2015-12-15T05 PST', '%Y-%m-%dT%H %Z', date(Y,M,D,H,M,S,O,TZ,DST)).
ERROR: format_time/3: Arguments are not sufficiently instantiated

2 个答案:

答案 0 :(得分:3)

根据文档,format_time/3的模式无法真正帮助您,因为它期望传递所有内容:

  

format_time(+ Out,+ Format,+ StampOrDateTime)

这意味着您提供每个参数。你希望看到一个带有-前缀的东西,这意味着它正在回复一些东西,这似乎意味着parse_time/3,但文档说:

  

Text支持的格式如下表所示。

然后它会精确列出两个选项,rfc_1123iso_8601,这两个选项都不匹配您的格式。似乎没有办法在这里提供格式代码,我觉得这很令人费解,因为这里有基础的Unix库当然可以做到这一点。

然而,问题可以通过每个人最喜欢的工具来解决:明确的句子语法!这是我的解决方案:

:- use_module(library(dcg/basics)).

myformat(date(Y,M,D,H,_,_,_,TZ,_)) -->
    integer(Y), "-", integer(M), "-", integer(D),
    "T", integer(H), " ", timezone(TZ).

timezone('UTC') --> "UTC".
timezone('UTC') --> "GMT".
timezone(-18000) --> "PST".
timezone(Secs) -->
    [Sign], digit(H0), digit(H1), digit(M0), digit(M1),
    {
     (Sign = 0'+ ; Sign = 0'-),
     number_codes(Hour, [H0,H1]),
     number_codes(Minutes, [M0, M1]),
     (Sign = 0'+
     -> Secs is Hour * 3600 + Minutes * 60
     ;  Secs is -(Hour * 3600 + Minutes * 60))
    }.

my_time_parse(Atom, Date) :-
    atom_codes(Atom, Codes),
    phrase(myformat(Date), Codes).

我确信学到的东西会看到改进这些代码的方法,但它为我提供了样本数据的技巧。遗憾的是,您需要枚举时区或在其上找到更好的数据源(可能解析系统时区定义?)但如果您事先知道只需要处理PST,则可以尝试使用它。这是一个例子:

?- my_time_parse('2015-12-15T05 PST', Date).
Date = date(2015, 12, 15, 5, _G3204, _G3205, _G3206, -18000, _G3208) ;
false.

希望这有帮助!

答案 1 :(得分:0)

迟到这个问题:所需的格式是接近iso_8601,它将采用格式"2015-12-15T05-08"。要将时区 " PST" 替换为 "-08",我们可以使用 append。对于代码:

:- set_prolog_flag(double_quotes, chars).

timezone(" PST", "-08").
timezone(" PDT", "-07").
%% etc

convert_time(Date, Stamp) :- 
    timezone(Zone, Delta),
    append(Front, Zone, Date),
    append(Front, Delta, Date2),
    string_chars(Date3, Date2),
    parse_time(Date3, iso_8601, Stamp).

test :-
    convert_time("2015-12-15T05 PST", Stamp),
    writeln('Stamp'=Stamp),
    fail.

带输出

?- test.
Stamp=1450184400.0
false.