How do I append 3 lists efficiently in Prolog?

前端 未结 3 1951
一向
一向 2021-01-13 22:43

I know how to do it for 2 lists:

append([],L,L).
append([H|T],L,[H|R]):-append(T,L,R).

but how to do it for 3? Without using the append for

相关标签:
3条回答
  • 2021-01-13 22:59

    Hope I understood the question (and I don't think the following is more efficient than the other solutions here), but did you mean something like this?

    append([],[],L,L).
    append([],[H|T],L,[H|R]) :- append([],T,L,R).
    append([H|T],L0,L1,[H|R]) :- append(T,L0,L1,R).
    
    0 讨论(0)
  • 2021-01-13 23:06

    To append lists efficiently, consider using difference lists. A difference list is a list expressed using a term with two lists. The most common representation uses (-)/2 as the functor for the term. For example, the list [1,2,3] can be expressed as:

    [1,2,3| Tail]-Tail.
    

    By keeping track of the list tail, i.e. of its open end, you can do several operations efficiently. For example, you can append an element to end of the list in O(1) by instantiating the tail:

    add_to_end_of_list(List-Tail, Element, List-Tail2) :-
        Tail = [Element| Tail2].
    

    Or simply:

    add_to_end_of_list(List-[Element| Tail2], Element, List-Tail2).
    

    Let's try it:

    ?- add_to_end_of_list([1,2,3| Tail]-Tail, 4, Result).
    Tail = [4|_G1006],
    Result = [1, 2, 3, 4|_G1006]-_G1006.
    

    Now, appending two lists is similar and also O(1). Instead of appending an element, we want to append a list of elements:

    dappend(List1-Tail1, Tail1-Tail2, List1-Tail2).
    

    For example:

    ?- dappend([1,2,3 | Tail1]-Tail1, [4,5,6| Tail2]-Tail2, Result).
    Tail1 = [4, 5, 6|Tail2],
    Result = [1, 2, 3, 4, 5, 6|Tail2]-Tail2.
    

    I leave to you as an exercise to answer your own question using difference lists. Note that going from a difference list to a closed list, is simply a question of instantiating the open end to the empty list. For example:

    ?- dappend([1,2,3 | Tail1]-Tail1, [4,5,6| Tail2]-Tail2, Result-[]).
    Tail1 = [4, 5, 6],
    Tail2 = [],
    Result = [1, 2, 3, 4, 5, 6].
    

    However, going from a closed list to a difference list does requires you to traverse the list, which is O(n):

    as_difflist([], Back-Back).
    as_difflist([Head| Tail], [Head| Tail2]-Back) :-
        as_difflist(Tail, Tail2-Back).
    

    The cost of constructing the difference lists may or may not be an issue, of course, depending on how you get the initial lists and how often you will be appending lists in your application.

    0 讨论(0)
  • 2021-01-13 23:12
    append3(Xs, Ys, Zs, XsYsZs) :-
       append(Xs, YsZs, XsYsZs),
       append(Ys, Zs, YsZs).
    

    Is as efficient, as it can get. Cost is about |Xs|+|Ys| inferences. However, you might have attempted to define it like the following with about 2|Xs|+|Ys| inferences.

    append3bad(Xs, Ys, Zs, XsYsZs) :-
       append(Xs, Ys, XsYs),
       append(XsYs, Zs, XsYsZs).
    

    Also, termination is much better in the first case:

    append3(Xs, Ys, Zs, XsYsZs)
       terminates_if b(Xs),b(Ys);b(XsYsZs)
    

    meaning that either Xs and Ys or XsYsZs needs to be known to make append3/4 terminate ... versus

    append3bad(Xs, Ys, Zs, XsYsZs)
       terminates_if b(Xs),b(Ys);b(Xs),b(XsYsZs)
                                 ^^^^^
    

    for append3bad/4, where XsYsZs is not sufficient, but additionally also Xs has to be known.

    0 讨论(0)
提交回复
热议问题