一連の行で構成されるかなり単純なファイル形式を解析しています。各行には、次のようなスペースで区切られたフィールドがあります。
l 0x9823 1
s 0x1111 3
l 0x1111 12
⋮
SWI-Prolog を使用しています。これは私がこれまで持っているDCGです:
:- consult(library(pure_input)).
load_trace(Filename, Traces) :-
phrase_from_file(trace_file_phrase(Traces), Filename).
trace_file_phrase([]) --> [].
trace_file_phrase([T|Ts]) --> trace_phrase(T), trace_file_phrase(Ts).
trace_phrase(access(Type, Address, SinceLast)) -->
access_type(Type), space,
address(Address), space,
nat(SinceLast), newline.
access_type(load) --> "l".
access_type(store) --> "s".
address(Number) --> "0x", hexnum(Number).
hexdigit(N) --> digit(N).
hexdigit(10) --> "a". hexdigit(11) --> "b". hexdigit(12) --> "c".
hexdigit(13) --> "d". hexdigit(14) --> "e". hexdigit(15) --> "f".
hexnum(N) --> hexdigit(D), hexnum(D, N).
hexnum(N, N) --> [].
hexnum(A, N) --> hexdigit(D), { A1 is A*16 + D }, hexnum(A1, N).
newline --> "\n".
space --> " ".
%% the following two productions are courtesy of Lars Mans at
%% https://stackoverflow.com/questions/3279822/parsing-numbers-with-multiple-digits-in-prolog
digit(0) --> "0". digit(1) --> "1". digit(2) --> "2".
digit(3) --> "3". digit(4) --> "4". digit(5) --> "5".
digit(6) --> "6". digit(7) --> "7". digit(8) --> "8".
digit(9) --> "9".
nat(N) --> digit(D), nat(D,N).
nat(N,N) --> [].
nat(A,N) --> digit(D), { A1 is A*10 + D }, nat(A1, N).
コメントで述べたように、Prolog で複数の数字を持つ数字を解析するから数字の処理を引用しました。
私が直面している問題は、これらのファイルの一部が 5 ~ 10 MB のように大きいことです。SWI-Prolog のデフォルト スタックはこれには不十分であり、これらのファイルの解析には 5 ~ 15 秒程度のかなりの時間がかかります。この状況についていくつか質問があります。
- このコードのどこに効率の問題がありますか?
trace_file_phrase//1
またはどちらかだと思いますnat//1
が、これらは単なる勘です。 - 問題がリストである場合、これよりも DCG を使用してリストを処理するためのより良い方法はありますか?
- 一般に、このような DCG のパフォーマンスの問題をどのように診断し、治療するのでしょうか?