问题 - 序言中的形式语言

发布于 2024-10-10 08:14:40 字数 521 浏览 11 评论 0原文

我正在尝试构建一个 DCG,它可以识别与此形式匹配的所有列表:a^nb^2m c^2m d^n
我写下了以下规则:
<代码>s --> [].
<代码>s -->广告。
广告 --> a、ad、d。
广告 -->公元前。
<代码>BC --> b、b、bc、c、c。
<代码>BC --> [].
一个 --> [一]。
<代码>b --> [b].
<代码>c --> [c].
<代码>d --> [d].

当我尝试使用这些规范评估字符串时,例如列表[a,b,b,c,c,d],它可以工作。但是,当我尝试评估查询短语(s,X)以便我可以看到此语法返回的所有可能的字符串时,它会无限循环。

我构建 DCG 的方式有问题吗?

I am trying to build a DCG which recognizes all lists which match this form : a^n b^2m c^2m d^n.
I have written up the following rules:
s --> [].
s --> ad.
ad --> a, ad, d.
ad --> bc.
bc --> b, b, bc, c, c.
bc --> [].
a --> [a].
b --> [b].
c --> [c].
d --> [d].

When I try to evaluate a string with those specifications, like the list [a,b,b,c,c,d], it works. But when I try to evaluate the query phrase(s, X) so that I can see all the possible strings returned by this grammar, it loops to infinity.

Is there something wrong with the way I have build the DCG?

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(4

痞味浪人 2024-10-17 08:14:40

您可以通过迭代加深来公平地枚举字符串:

?- length(Ls, _), phrase(s, Ls).
Ls = [] ;
Ls = [] ;
Ls = [a, d] ;
Ls = [a, a, d, d] ;
Ls = [b, b, c, c] ;
etc.

You can enumerate the strings fairly with iterative deepening:

?- length(Ls, _), phrase(s, Ls).
Ls = [] ;
Ls = [] ;
Ls = [a, d] ;
Ls = [a, a, d, d] ;
Ls = [b, b, c, c] ;
etc.
落墨 2024-10-17 08:14:40

@Simon 使用 DGC 的方法是正确的,但您的尝试存在两个问题。

第一个是您在如下子句中保留了递归:

ad --> a, ad, d.

这就是它循环到无穷大的原因之一。

第二个问题是这种形式语言无法构建为上下文无关语法,因此您需要一个额外的变量,例如 count。

s(Count) --> a(Count),b(Count),c(Count),d(Count).
a(0) --> [].
a(succ(Count)) --> [a],a(Count).
b(0) --> [].
b(succ(Count)) --> [b,b],b(Count).
c(0) --> [].
c(succ(Count)) --> [c,c],c(Count).
d(0) --> [].
d(succ(Count)) --> [d],d(Count).

然后使用以下目标 s(_, L, []) 进行查询。

我知道这是一个老问题,但也许从现在开始有人会发现正确的答案很有用。

@Simon the approach of using DGCs is correct but there are two issues in your attempt.

The first one is that you have left recursion in clauses like the following:

ad --> a, ad, d.

And this is one of the reasons why it loops to infinity.

The second issue is that this formal language cannot be built as a context free grammar, therefore you need an extra variable such as count.

s(Count) --> a(Count),b(Count),c(Count),d(Count).
a(0) --> [].
a(succ(Count)) --> [a],a(Count).
b(0) --> [].
b(succ(Count)) --> [b,b],b(Count).
c(0) --> [].
c(succ(Count)) --> [c,c],c(Count).
d(0) --> [].
d(succ(Count)) --> [d],d(Count).

Then query it using the following goal s(_, L, []).

I know this is an old question but maybe someone finds the correct answer useful from now on.

北音执念 2024-10-17 08:14:40

我没有看到这个问题的序言部分。答案很大程度上取决于你如何实现这个语法。

我最好的猜测是颠倒规则的顺序,首先应用终止规则。

I don't see the prolog part of this question. The answer highly depends on how you implemented this grammar.

My best guess would be to reverse the order of the rules, to have the terminating rules applied first.

粉红×色少女 2024-10-17 08:14:40

我写这篇文章是为了帮助限制解决方案,即使有无限的解决方案。但我意识到有一种方法可以重新排列规则以首先获得更短的结果。

因为 ad --> a, ad, d.ad -->; 之前评估bc. 它试图满足 ad --> a, ad, a.ad --> 之前公元前。。我会放 ad --> bc. 位于 ad 之前 -->一个,广告,一个。bc 也是如此 --> b、b、bc、c、c。bc --> []. 规则

正如 arian 指出的那样,首先应用终止规则,将确保首先找到较短的解决方案。

我还想指出的是 [] s 和 s -> 有两种解决方案广告-> BC-> [] 我会消除 s --> []. 因为它是多余的。

总而言之,我会尝试这个解决方案:

s --> ad.
a --> [a].
b --> [b].
c --> [c].
d --> [d].
ad --> bc.
bc --> [].
ad --> a, ad, d.
bc --> b, b, bc, c, c.

原始帖子:

我必须查找如何进行计数(自从我做序言以来已经有一段时间了)但是数量是无限的,并且由于序言试图找到所有解决方案,它永远不会停止寻找,尽管我确信在此之前您会遇到堆栈溢出或一些错误:)。

限制结果数量的一种方法是限制解的大小

 phrase(s, X), length(X, 4).

获取恰好具有 4 个值的所有解,将其

X = [a, a, d, d]
X = [b, b, c, c]

增加到 6 将产生:

X = [a, a, a, d, d, d]
X = [a, b, b, c, c, d]

或使用范围:

 phrase(s, X), length(X, Y), Y >= 4 , Y < 10, Y != 6.

I wrote this as a way to help limit solutions even though there are infinite solutions. But I realized there would be a way to rearrange the rules to get shorter results first.

Because ad --> a, ad, d. is evaluated before ad --> bc. it tries to satisfy ad --> a, ad, a. before ad --> bc.. I would put ad --> bc. before ad --> a, ad, a.. The same goes for the bc --> b, b, bc, c, c. and bc --> []. rules

As arian pointed out have the terminating rules applied first, would ensure the shorter solutions are found first.

I also want to point out that there are two solutions for [] s and s -> ad -> bc -> [] I would eliminate s --> []. as it's redundant.

All in all I would try this solution:

s --> ad.
a --> [a].
b --> [b].
c --> [c].
d --> [d].
ad --> bc.
bc --> [].
ad --> a, ad, d.
bc --> b, b, bc, c, c.

ORIGINAL POST:

I had to look up how to do counting (it's been a while since I did prolog) But there are an infinite number and since prolog tries to find all solutions it never stops looking, although I'm sure you'll hit a stack over flow or some error before then :).

One way to limit the number of results is to limit the size of the solution

 phrase(s, X), length(X, 4).

Gets all solutions with exactly 4 values, which would be

X = [a, a, d, d]
X = [b, b, c, c]

increasing to 6 would yield:

X = [a, a, a, d, d, d]
X = [a, b, b, c, c, d]

Or use ranges:

 phrase(s, X), length(X, Y), Y >= 4 , Y < 10, Y != 6.
~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文