- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我写了一些谓词,它们采用列表的长度并附加了一些约束(这是要使用的正确词汇表吗?):
clp_length([], 0).
clp_length([_Head|Rest], Length) :-
Length #>= 0, Length #= Length1 + 1,
clp_length(Rest, Length1).
clp_length2([], 0).
clp_length2([_Head|Rest], Length) :-
Length #= Length1 + 1,
clp_length2(Rest, Length1).
第一个终止于这个简单的查询,但第二个没有:
?- Small in 1..2, clp_length(Little, Small).
Small = 1,
Little = [_1348] ;
Small = 2,
Little = [_1348, _2174] ;
false.
?- Small in 1..2, clp_length2(Little, Small).
Small = 1,
Little = [_1346] ;
Small = 2,
Little = [_1346, _2046] ;
% OOPS %
这对我来说很奇怪,因为长度显然大于 0。要弄清楚你可以搜索,找到零,并推断从零开始添加只能增加数字,或者你可以传播 in 1..2
约束向下。感觉这个附加条款是多余的!这并不意味着我的 clpfd 心智模型完全错误。
所以我想我有两个问题(希望对第二个问题的回答作为评论)
具体来说,为什么这个额外的约束会导致查询正常工作?
一般来说,有没有我可以用来了解 clpfd 是如何实现的资源,而不是仅仅查看一些如何使用它的示例?我宁愿不必阅读 Markus Triska 的论文,但这是我能找到的唯一来源。如果我想回答这样的问题,这是我唯一的选择吗?
最佳答案
1mo,命名有问题。请引用之前的回答 mat和 me推荐关系名称。使用不恰当的名称不会让您走得太远。所以 list_length/2
或 list_fdlength/2
将是一个合适的名称。因此我们有 list_fdlength/2
和 list_fdlength2/2
。
2做,考虑list_fdlength2/2
的规则。没有任何迹象表明 0 与您相关。因此,如果您使用 0 或 1 或 -1 或其他作为基本情况,则该规则将完全相同。那么这个糟糕的规则怎么会意识到 0 就是你的终点呢?更好的是,考虑一个概括:
list_fdlength2(fake(N), N) :- % Extension to permit fake lists N #< 0.list_fdlength2([], 0).list_fdlength2([_Head|Rest], Length) :- Length #= Length1 + 1, list_fdlength2(Rest, Length1).
此概括显示了所有真实答案和虚假答案。请注意,我没有更改规则,我只是添加了这个替代事实。因此,伪造的解决方案实际上是由规则引起的:
?- list_fdlength2(L, 1). L = [_A] ; L = [_A, _B|fake(-1)] ; L = [_A, _B, _C|fake(-2)] ; ... .?- list_fdlength2(L, 0). L = [] ; L = [_A|fake(-1)] ; L = [_A, _B|fake(-2)] ; ... .
每个子句都试图在该子句的范围内为解决方案做出贡献。但是无法(通过内置的 Prolog 执行机制)推导出某些规则不再相关。您必须像以前那样用冗余约束明确说明这一点。
现在,回到包含冗余约束 Length #>= 0
的原始解决方案。根本不应该有任何这样的假解决方案。
list_fdlength(fake(N), N) :- N #< 0.list_fdlength([], 0).list_fdlength([_Head|Rest], Length) :- Length #>= 0, Length #= Length1 + 1, list_fdlength(Rest, Length1).?- list_fdlength(L, 1). L = [_A] ; L = [_A, _B|fake(-1)] % totally unexpected ; false.?- list_fdlength(L, 0). L = [] ; L = [_A|fake(-1)] % eek ; false.
也有假答案!多么丑陋!至少,它们的数量是有限的。但是,你可以通过使用做得更好Length #>= 1
代替 Length #>=0
。通过这个小改动,当 N 为非负时,不再有任何伪解,因此您的原始程序也会更好。
关于prolog - 为什么在我添加冗余约束之前此 clpfd 查询不终止?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/50992433/
给出以下代码示例: example(Ls) :- Ls = [X,Y], Ls ins 1..2, Cost #= max((X #= 1)*3 + (Y #= 1)*5,
是否可以对整数进行约束,使其不能是(完美)平方数? 我有: square(Square):- N#>0, Square #= N*N. 如何定义 notsquare(Notsquare):- ...
我定义了具体化的变体 clpfd约束 (#=)/2和 (#>)/2 : :- use_module(library(clpfd)). ltA(X,Y,Truth) :- X # B, bool01_
CLPFD 系统的主要目标不是有效地处理二次方程,但是,是否有更好的方法来制定如下问题? 似乎问题归结为如下等式。 SWI 与 library(clpfd)给: ?- 时间( ((L+7)^2#=L^
假设我想像这样表示整数:integer:Sign:[FirstDigit,SecondDigit,...] 。例如,42 将表示为 integer:positive:[4,2] . 我需要一个谓词来根
我尝试用 clpfd 解决“Escape from Zurg”问题。 https://web.engr.oregonstate.edu/~erwig/papers/Zurg_JFP04.pdf玩具从左
我写了一些谓词,它们采用列表的长度并附加了一些约束(这是要使用的正确词汇表吗?): clp_length([], 0). clp_length([_Head|Rest], Length) :- L
我想弄清楚 clpfd 中的边界传播是什么,但似乎无法在任何地方找到一个好的解释。 我正在修改 Prolog 和 clpfd 并遇到了这个问题,但查看讲义对我来说没有意义。谁能解释一下边界传播的实际含
我和一个 friend 正在编写一个应该解决 CLP 问题的程序。我们想使用最小化来优化解决方案,但它不起作用,因为它一直说我们从 sum(P,#=,S) 得到的数字介于两个数字之间(例如 5..7)
我可以用这种方式指定变量的域: MyVar in 1..10 或 MyVar in {1,10,15} 但我有一个变量,我想这样指定: Activity_1__room in {room_1, roo
我是约束编程的 prolog 新手。我有一个 CLPFD 没有像我期望的那样减少域的问题。这可能真的很简单。 [A,B] ins 1..5,A*B#=5. 我希望它将 A 和 B 的域减少到 1\/
我做了两个实现来解决 Shikaku 难题。一个使用顶部、左侧、宽度和高度 (TLWH) 作为每个矩形的参数,另一个使用顶部、左侧、底部、右侧 (TLBR)。 出于某种原因,使用 TLBR 的速度要快
我正在开发一个 (SWI-)Prolog 程序,该程序使用 CLP(FD) 约束来寻找特定问题的解决方案。为此,我碰巧需要两个列表的“未定位”重叠。那是: 列表 La长度为 A 列表 Lb长度为 B
例如,假设我有这个程序(仅在 swi-prolog 中测试过): :- use_module(library(clpfd)). :- use_module(library(lists)). % Sor
我最近在 Google Play 应用商店发现了一个小游戏,叫做 Cryptogram .有几十个类似的应用程序。这个想法是将数字与颜色相匹配,以使所有方程式听起来都正确。 我能够很快地通过手工解决问
我正在尝试写类似“如果你之前得到球,你就得到了球,并且从那以后没有给它”: :- use_module(library(clpfd)). time(T1, has_ball) :- time(
我想最大化两个变量之间的差异: :- use_module(library(clpfd)). maximize(X) :- X = [A,B], X ins 1..5, % I
好的,所以我有一个叫做 CuFrog 的谜题,它包括在每个位置填充一个数字的 3x3x3 立方体,但从一个位置跳到另一个位置时跳过一个位置。例如,考虑一个展平的立方体,第 1 侧 (1,1) 右侧的有
我有一个即将到来的逻辑考试,并且一直在学习我类(class)中的一些过去的论文。我遇到了一个关于物化的问题,并将其发布在下面; 用具体化来表示变量 B 的性质 取值为 1 或 8。 在阅读了一些资源并
我得到了一个使用我选择的约束求解器解决斑马拼图的练习,我使用 Prolog clpfd library 进行了尝试。 . 我知道在 Prolog 中还有其他更惯用的方法来解决这个问题,但这个问题专门针
我是一名优秀的程序员,十分优秀!