我试图理解逻辑编程语言(在我的例子中是 Prolog)和 Haskell 的类型系统之间的关系。
我知道根据关系使用统一和变量来查找值(或类型,在 Haskell 的类型系统中)。为了更好地理解它们之间的异同,我尝试在 Haskell 的类型级别重写一些简单的 prolog 程序,但是我在某些部分遇到了麻烦。
首先,我重写了这个简单的 prolog 程序:
numeral(0).
numeral(succ(X)) :- numeral(X).
add(0,Y,Y).
add(succ(X),Y,succ(Z)) :- add(X,Y,Z).
作为:
class Numeral a where
numeral :: a
numeral = u
data Zero
data Succ a
instance Numeral Zero
instance (Numeral a) => Numeral (Succ a)
class (Numeral a, Numeral b, Numeral c) => Add a b c | b c -> a where
add :: b -> c -> a
add = u
instance (Numeral a) => Add a Zero a
instance (Add x y z) => Add (Succ x) (Succ y) z
它工作正常,但我无法用这个 Prolog 扩展它:
greater_than(succ(_),0).
greater_than(succ(X),succ(Y)) :- greater_than(X,Y).
我尝试的是这样的:
class Boolean a
data BTrue
data BFalse
instance Boolean BTrue
instance Boolean BFalse
class (Numeral a, Numeral b, Boolean r) => Greaterthan a b r | a b -> r where
greaterthan :: a -> b -> r
greaterthan = u
instance Greaterthan Zero Zero BFalse
instance (Numeral a) => Greaterthan (Succ a) Zero BTrue
instance (Numeral a) => Greaterthan Zero (Succ a) BFalse
instance (Greaterthan a b BTrue) => Greaterthan (Succ a) (Succ b) BTrue
instance (Greaterthan a b BFalse) => Greaterthan (Succ a) (Succ b) BFalse
此代码的问题是最后两个实例导致fundep 冲突。我可以理解为什么,但在我看来这不应该是一个问题,因为它们的保护部分(或者不管它叫什么,我的意思是(Greaterthan a b c) =>
部分)是不同的,所以最后两个实例声明中a
的 s 和b
s 实际上是不同的价值观,没有冲突。
我试图重写的另一个程序是:
child(anne,bridget).
child(bridget,caroline).
child(caroline,donna).
child(donna,emily).
descend(X,Y) :- child(X,Y).
descend(X,Y) :- child(X,Z),
descend(Z,Y).
(顺便说一句,示例来自Learn Prolog Now书)
data Anne
data Bridget
data Caroline
data Donna
data Emily
class Child a b | a -> b where
child :: a -> b
child = u
instance Child Anne Bridget
instance Child Bridget Caroline
instance Child Caroline Donna
instance Child Donna Emily
class Descend a b | b -> a where
descend :: b -> a
descend = u
instance (Child a b) => Descend a b
instance (Child a c, Descend c b) => Descend a b
我在最后一行收到“重复实例”错误。我认为这是一个类似的问题,即使我有不同的防护部件,我也会出错,因为身体部位(我的意思是Descend a b
部件)是相同的。
因此,如果可能的话,我正在寻找将 Prolog 程序移植到 Haskell 类型级别的方法。任何帮助将不胜感激。
编辑:
Ed'ka 的解决方案有效,但方式完全不同。我仍在尝试了解我们何时可以在类型系统中运行 Prolog 程序,何时/为什么我们需要编写不同的算法以使其工作(如在 Ed'ka 的解决方案中),以及何时/为什么没有办法在 Haskell 的类型系统中实现一个程序。
也许我可以在阅读“Fun With Functional Dependencies”之后找到一些关于此的提示。