- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
在 Haskell 中,有一个称为“as”运算符(有时称为别名)的语言功能。这个想法如下:假设您有一个函数,例如将列表作为输入并希望返回所有尾部,您可以将其实现为:
tails a@(_:xs) = a : tails xs
tails [] = [[]]
@
确保您拥有对整个参数的引用以及对参数结构某些部分的引用。这是智能性能方面的(它更像是一种性能黑客,因为在第一行的主体中重建数组 (x:xs)
),如果编译器没有优化的话,将导致分配新对象、修改字段等。请参阅here了解更多信息。
我想知道 Prolog 是否有等效的东西:例如,如果你想在 Prolog 中实现尾部,可以通过以下方式完成:
tails([H|T],[[H|T]|TA]) :-
tails(T,TA).
tails([],[[]]).
但是如果有一个“as”运算符,比如:
tails(L@[_|T],[L|TA]) :- %This does not compile
tails(T,TA).
tails([],[[]]).
是否有这样的构造或语言扩展?
最佳答案
TL;DR:好主意1!加速似乎限制在约 20%(对于大多数列表大小)。
在这个答案中,我们比较了关于 @
的三个不同谓词。类似数据重用:
list_tails([], [[]]). % (1) like `tails/2` given by the OP ...list_tails([E|Es], [[E|Es]|Ess]) :- % ....... but with a better name :-) list_tails(Es, Ess).list_sfxs1(Es, [Es|Ess]) :- % (2) "re-use, mutual recursion" aux_list_sfxs1(Es, Ess). % "sfxs" is short for "suffixes"aux_list_sfxs1([], []).aux_list_sfxs1([_|Es], Ess) :- list_sfxs1(Es, Ess).list_sfxs2([], [[]]). % (3) "re-use, direct recursion"list_sfxs2(Es0, [Es0|Ess]) :- Es0 = [_|Es], list_sfxs2(Es, Ess).
为了测量运行时间,我们使用以下代码:
:-( dif(D,sicstus), current_prolog_flag(dialect,D) ; use_module(library(between)) ).run_benchs(P_2s, P_2, L, N, T_ms) :- between(1, 6, I), L is 10^I, N is 10^(8-I), length(Xs, L), member(P_2, P_2s), garbage_collect, call_walltime(run_bench_core(P_2,Xs,N), T_ms).run_bench_core(P_2, Xs, N) :- between(1, N, _), call(P_2, Xs, _), false.run_bench_core(_, _, _).
测量wall-time 2,我们利用call_<a href="https://en.wikipedia.org/wiki/Wall-clock_time" rel="noreferrer noopener nofollow">walltime</a>/2
— call_time/2
的变体:
call_walltime(G, T_ms) :- statistics(walltime, [T0|_]), G, statistics(walltime, [T1|_]), T_ms is T1 - T0.
让我们测试一下上面的代码变体...
L
...N
(为了更好的准确性)。首先,我们使用swi-prolog版本 7.3.14(64 位):
?- run_benchs([list_sfxs1,list_sfxs2,list_tails], P_2, L, N, T_ms). P_2 = list_sfxs1, L*N = 10*10000000, T_ms = 7925; P_2 = list_sfxs2, L*N = 10*10000000, T_ms = 7524; P_2 = list_tails, L*N = 10*10000000, T_ms = 6936; P_2 = list_sfxs1, L*N = 100*1000000, T_ms = 6502; P_2 = list_sfxs2, L*N = 100*1000000, T_ms = 5861; P_2 = list_tails, L*N = 100*1000000, T_ms = 5618; P_2 = list_sfxs1, L*N = 1000*100000, T_ms = 6434; P_2 = list_sfxs2, L*N = 1000*100000, T_ms = 5817; P_2 = list_tails, L*N = 1000*100000, T_ms = 9916; P_2 = list_sfxs1, L*N = 10000*10000, T_ms = 6328; P_2 = list_sfxs2, L*N = 10000*10000, T_ms = 5688; P_2 = list_tails, L*N = 10000*10000, T_ms = 9442; P_2 = list_sfxs1, L*N = 100000*1000, T_ms = 10255; P_2 = list_sfxs2, L*N = 100000*1000, T_ms = 10296; P_2 = list_tails, L*N = 100000*1000, T_ms = 14592; P_2 = list_sfxs1, L*N = 1000000*100, T_ms = 6955; P_2 = list_sfxs2, L*N = 1000000*100, T_ms = 6534; P_2 = list_tails, L*N = 1000000*100, T_ms = 9738.
然后,我们使用 sicstus-prolog 重复之前的查询3版本 4.3.2(64 位):
?- run_benchs([list_sfxs1,list_sfxs2,list_tails], P_2, L, N, T_ms). P_2 = list_sfxs1, L*N = 10*10000000, T_ms = 1580; P_2 = list_sfxs2, L*N = 10*10000000, T_ms = 1610; P_2 = list_tails, L*N = 10*10000000, T_ms = 1580; P_2 = list_sfxs1, L*N = 100*1000000, T_ms = 710; P_2 = list_sfxs2, L*N = 100*1000000, T_ms = 750; P_2 = list_tails, L*N = 100*1000000, T_ms = 840; P_2 = list_sfxs1, L*N = 1000*100000, T_ms = 650 ; P_2 = list_sfxs2, L*N = 1000*100000, T_ms = 660; P_2 = list_tails, L*N = 1000*100000, T_ms = 740; P_2 = list_sfxs1, L*N = 10000*10000, T_ms = 620; P_2 = list_sfxs2, L*N = 10000*10000, T_ms = 650; P_2 = list_tails, L*N = 10000*10000, T_ms = 740; P_2 = list_sfxs1, L*N = 100000*1000, T_ms = 670; P_2 = list_sfxs2, L*N = 100000*1000, T_ms = 650; P_2 = list_tails, L*N = 100000*1000, T_ms = 750; P_2 = list_sfxs1, L*N = 1000000*100, T_ms = 12610; P_2 = list_sfxs2, L*N = 1000000*100, T_ms = 12560; P_2 = list_tails, L*N = 1000000*100, T_ms = 33460.
摘要:
脚注1:为什么要进行推杆特技(@)/2
在规则头?最终得到非 idiomatic Prolog 代码?
脚注 2:我们对总运行时间感兴趣。为什么?因为垃圾收集成本随着数据大小的增加而增加!
脚注 3:为了便于阅读,答案序列已经过后处理。
脚注 4: 自 release 4.3.0 起可用。当前的目标架构包括IA-32和 AMD64 .
关于performance - Prolog 有像 Haskell 一样的别名 "operator"吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34458299/
上下文:我已经在域根上设置了一个用于 GIS 测试和 dokuwiki 的虚拟机服务器。我想在子域上提供 gis Web 应用程序,这样 dokuwiki url 重命名就不会发生冲突(而且感觉更干净
这个问题在这里已经有了答案: If two variables point to the same object, why doesn't reassigning one variable affe
代码如下: Select Branch.BranchName, Sum(Inventory.OnHand) From Inventory, Branch Inventory.BranchNum = B
我的目标是创建一种“Javascript 库”,如果您可以这样调用它的话。我打算在浏览网页时使用它来操作页面,将其作为 greasemonkey 脚本动态加载。这个想法是将“win”映射到window
是否可以在 javascript 中以某种方式为本地变量分配别名/引用? 我的意思是类似 C 的东西: function foo() { var x = 1; var y = &x; y+
SQLite 别名 您可以暂时把表或列重命名为另一个名字,这被称为别名。使用表别名是指在一个特定的 SQLite 语句中重命名表。重命名是临时的改变,在数据库中实际的表的名称不会改变。 列别名用来
我有以下导入(在同一个 TS 源文件中): import {Vector as sourceVector} from "ol/source"; import {Vector} from "ol/lay
我想弄清楚 this 到底是什么行是在这个 github json 项目的 cmake 文件中, add_library(${NLOHMANN_JSON_TARGET_NAME} INTERFACE)
使用集合时如何创建别名 (AS)。 $collection->addAttributeToSelect('total_qty') 可以这样做: $collection->getSelect()->co
这个问题在这里已经有了答案: Pass an argument to a Git alias command (4 个回答) 去年关闭。 我的团队在不同的分支上工作,通常,我需要快速从分支切换来帮助他
我正在寻找一种向我的 vimrc 添加自定义命令或别名的方法,以便快速切换到 Vim 中的不同目录。我曾经设置过一些东西,但丢失了那个配置。这是用于 Linux 设置。 基本上,我想设置一个别名(如果
我使用 Delphi 和 Firebird 1.5 开发了一个应用程序,其中服务器与应用程序位于同一台计算机上。我现在正在将应用程序部署到另一个站点,其中 Firebird 服务器( super 服务
我创建了一个全新的 React 应用 create-react-app demo 我需要为某些目录/组件创建别名,例如: import { Header } from '@uicomponents'
我正在尝试打印有向图,并且不断更改各种节点属性,例如颜色和形状。有什么方法可以使用一次定义并用于多个节点的变量?理想情况下,我想要这样的东西: digraph g { building_color =
server { listen 80; server_name pwta; root html; location /test/{ alia
我想匹配多个数组,并在匹配时构建另一个数组。该键可以在任意数量的数组中匹配,也可以根本不匹配。 [ [{ 'a': 13 }, { 'b': 62 }, { 'c': 93 }, { 'd': 52
经过审查后,我需要更改一些代码并将所有更改代码重新推送到远程分支上。 所以我想使用别名(例如:git repushall)在 git 上自动化这些不同的命令: git add . git commit
我有一个这样的查询: select a1.name, b1.info from (select name, id, status from table1 a) as a1 righ
我想给一些 racket 2htdp 函数/宏起别名,这样我就可以为我的 child 将它们翻译成另一种语言。 我可以简单地用define给函数起别名。我在使用 big-bang 结构时遇到了麻烦;例
经过审查后,我需要更改一些代码并将所有更改代码重新推送到远程分支上。 所以我想使用别名(例如:git repushall)在 git 上自动化这些不同的命令: git add . git commit
我是一名优秀的程序员,十分优秀!