8 sorting compare list prolog stable-sort
这是关于对术语的特定参数进行排序的问题的答案的后续,而没有为a创建新列表keysort
(如果我正确理解原始问题).
假设我们想predsort/3
表现得如下sort/2
:如果我理解正确,这将意味着将其称为:
?- predsort(compare, List, Sorted).
Run Code Online (Sandbox Code Playgroud)
现在说我们想用它predsort/3
来实现排序msort/2
(参见这个问题).一种方法是定义一个比较谓词Pred(-Delta, +A, +B)
,它不Delta
与=
元素实际上相等的时候统一:
mcompare(Delta, A, B) :-
compare(Delta0, A, B),
( Delta0 == (=)
-> Delta = (<)
; Delta = Delta0
).
?- predsort(mcompare, List, Sorted).
Run Code Online (Sandbox Code Playgroud)
问题:这是否真的只是排序而不删除重复项,就像msort/2
这样做?好像应该这样.
继续:说我们想要按照术语中第n个参数的标准顺序对arity> n进行排序.干净的方法是:
sort_argn(N, List, Sorted) :-
map_list_to_pairs(arg(N), List, Pairs),
keysort(Pairs, Sorted_pairs),
pairs_values(Sorted_pairs, Sorted).
Run Code Online (Sandbox Code Playgroud)
如果我们想用来predsort/3
达到相同的效果,我们可以尝试使用比较谓词,如下所示:
compare_argn(N, Delta, A, B) :-
arg(N, A, AN),
arg(N, B, BN),
compare(Delta, AN-A, BN-B).
Run Code Online (Sandbox Code Playgroud)
并对第二个论点进行排序:
?- predsort(compare_argn(2), List, Sorted).
Run Code Online (Sandbox Code Playgroud)
然而,这是不一样的sort_argn/3
上述使用keysort/2
.它将删除重复项,如果两个项的第二个参数恰好相同,它将根据原始完整项的标准顺序排序复合项:
?- predsort(compare_argn(2), [f(b,2), f(a,1), f(a,1), f(a,2)], Sorted).
Sorted = [f(a, 1), f(a, 2), f(b, 2)].
?- sort_argn(2, [f(b,2), f(a,1), f(a,1), f(a,2)], Sorted).
Sorted = [f(a, 1), f(a, 1), f(b, 2), f(a, 2)].
Run Code Online (Sandbox Code Playgroud)
作出假设,对于每对A
和B
传递到比较谓词Pred(Delta, A, B)
,A
之前谈到B
在原始列表.我们可以定义一个比较:
compare_argn_stable(N, Delta, A, B) :-
arg(N, A, AN),
arg(N, B, BN),
compare(Delta0, AN, BN),
( Delta0 == (=)
-> Delta = (<)
; Delta = Delta0
).
Run Code Online (Sandbox Code Playgroud)
此时,当且仅当任何两个元素A
并且B
总是以与它们在原始列表中相同的顺序传递给比较谓词时,这应该与sort_argn/3
上面的行为相同:
?- predsort(compare_argn_stable(N), List, Sorted).
Run Code Online (Sandbox Code Playgroud)
当然,当两个"键"相等时,compare_argn_stable/4
统一它是很重要的.此外,行为是依赖于实现的,并且仅与示例iff 相同,iff 在将元素传递给比较谓词时保持元素的原始顺序.Delta
<
keysort
predsort/3
问题是对的吗?
问题是否有任何标准涵盖这方面predsort/3
?
小智 4
既然没有人回答,而且我现在对此非常确定:
是的,您可以用来predsort/3
模拟任何其他类型。该问题详细描述了如何进行。
然而:由于多种原因,这是一个坏主意。
predsort/3
(参见问题)predsort/3
本身不属于任何标准(据我所知)msort/2
比keysort/2
predsort/3
在极少数情况下,列表元素的大小可能比我们正在排序的列表的长度大得多,这个小舞蹈:
list_to_keyval_pairs(List, Pairs), % defined by the user as necessary
keysort(Pairs, Sorted_pairs),
pairs_values(Sorted_pairs, Sorted)
Run Code Online (Sandbox Code Playgroud)
(参见此处predsort(keycmp, List, Sorted)
)实际上比使用keycmp/3
由用户定义的更昂贵(更慢) 。即使如此,具有等效键的结果的顺序不仅取决于 的(用户)定义keycmp/3
,还取决于 的实现predsort/3
。
换句话说,“稳定”排序predsort/3
是一个坏主意。
归档时间: |
|
查看次数: |
345 次 |
最近记录: |