微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

哈斯克尔如何使用GADT实施自定义monad转换器?

如何解决哈斯克尔如何使用GADT实施自定义monad转换器?

我正在尝试通过实现自己的方法来学习monad变压器。编码 如果我使用newType DummyT m x y z = DummyT { runDummyT :: m (Dummy x y z) }则编译 但如果我使用GADT,则不会。但是我需要使用GADT,但似乎更多 难。我收到此错误

Expecting one more argument to ‘m’   
Expected a type,but ‘m’ has kind ‘* -> *’                                              
In the first argument of ‘DummyT’,namely ‘m’                                          
In the first argument of ‘Functor’,namely ‘(DummyT m x y)`

对于应用实例和monad实例,我得到相同的错误。 为什么会出现此错误?任何帮助表示赞赏。

我的单子:

instance Monad (Dummy x y) where
  return = Return
  (>>=) = Bind

data Dummy x y z where
  Return :: z -> Dummy x y z
  Bind :: Dummy x y z -> (z -> Dummy x y q) -> Dummy x y q

变压器尝试:

instance Monad m => Functor (DummyT m x y) where
  fmap = liftM

instance Monad m => applicative (DummyT m x y) where
  pure = return
  (<*>) = ap

instance Monad m => Monad (DummyT m x y) where
  return = Return
  (>>=) = Bind

data DummyT m x y z where
  Return :: z -> DummyT m x y z
  Bind :: DummyT m x y z -> (z -> DummyT m x y q) -> DummyT m x y q

解决方法

newType DummyT m x y z = DummyT { runDummyT :: m (Dummy x y z) }

m的种类显然是* -> *,因为m被应用于一种类型。

data DummyT m x y z where
  Return :: z -> DummyT m x y z
  Bind :: DummyT m x y z -> (z -> DummyT m x y q) -> DummyT m x y q

m未使用,因此GHC推断类型为*,因为这是最简单的。也许可以使这种多面性,但是没有。

如果您想要另一种,请明确要求它:

data DummyT (m :: * -> *) x y z where
  Return :: z -> DummyT m x y z
  Bind :: DummyT m x y z -> (z -> DummyT m x y q) -> DummyT m x y q

不过,我想知道,如果不使用m,是否有意义?

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其他元素将获得点击?
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。)
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbcDriver发生异常。为什么?
这是用Java进行XML解析的最佳库。
Java的PriorityQueue的内置迭代器不会以任何特定顺序遍历数据结构。为什么?
如何在Java中聆听按键时移动图像。
Java“Program to an interface”。这是什么意思?