- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我试图编写将所有数据节点乘以给定数字的 monad。我不知道如何正确定义 Monad。
module Main where
data LinkedList a = NodeWithoutNext a | Node a (LinkedList a) deriving (Show)
instance Functor LinkedList where
fmap f (NodeWithoutNext x) = NodeWithoutNext (f x)
fmap f (Node a next) = Node (f a) (fmap f next)
instance Applicative LinkedList where
pure = NodeWithoutNext
(NodeWithoutNext f) <*> (NodeWithoutNext x) = NodeWithoutNext (f x)
(NodeWithoutNext f) <*> (Node a next) = Node (f a) (fmap f next)
instance Monad LinkedList where
return = NodeWithoutNext
(NodeWithoutNext a) >>= f = f a
**(Node a next) >>= f = Node (f a) (next >>= f)**
main :: IO ()
main = do
let list = Node 3 (Node 2 (NodeWithoutNext 7))
print (list)
print (list >>= (\x -> NodeWithoutNext (x+200)))
我收到错误 -
Bla.hs:16:39: error:
* Couldn't match type `b' with `LinkedList b'
`b' is a rigid type variable bound by
the type signature for:
(>>=) :: forall a b.
LinkedList a -> (a -> LinkedList b) -> Linked
at Bla.hs:15:24-26
Expected type: LinkedList (LinkedList b)
Actual type: LinkedList b
* In the second argument of `Node', namely `(next >>= f)'
In the expression: Node (f a) (next >>= f)
In an equation for `>>=':
(Node a next) >>= f = Node (f a) (next >>= f)
* Relevant bindings include
f :: a -> LinkedList b (bound at Bla.hs:16:22)
(>>=) :: LinkedList a -> (a -> LinkedList b) -> LinkedLi
(bound at Bla.hs:15:24)
|
16 | (Node a next) >>= f = Node (f a) (next >>= f)
| ^^^^^^^^^^
这里我需要修复什么?问题出在带有**的那一行。
最佳答案
类型系统会提示,因为“bind”运算符 >>=
有签名:
<b>(>>=) :: Monad m => m a -> (a -> m b) -> m b</b>
因此,让我们专门研究 LinkedList
的签名:
(>>=) :: <b>LinkedList a</b> -> (a -> <b>LinkedList b</b>) -> <b>LinkedList b</b>
所以该函数在左侧取 LinkedList a
,右侧是映射 a
的函数(!) 到 LinkedList b
,我们预计结果是 LinkedList b
。您定义的函数不能这样做:因为如果您构造一个 Node (f a) (next >>= f)
,结果将是 LinkedList (LinkedList b)
。确实,您申请f
至a
(列表的头部),产生 LinkedList b
(作为 Node
的负责人)。
如果我们看一下上面的签名,可能会出现一些想法,但最直接的想法可能是实现某种 concatMap :: [a] -> (a -> [b]) -> [b]
。 “绑定(bind)”必须满足以下规则:
return a >>= k = k a
;m >>= return = m
;和m >>= (\x -> k x >>= h) = (m >>= k) >>= h
.concatMap
函数满足三个约束。 (1) 如果我们包装一个值 x
进入LinkedList
,然后申请f
在(单例列表的)每个元素上,我们将获得一个包含一个列表的列表,即 k a
的结果。 ,但是通过使用concat
的concatMap
,我们再次将其展平。 (2) 如果我们使用m >>= return
使用 concat,我们会将每个元素包装在一个新的单例列表中,但是串联将再次将其解开。
所以我们可以将其实现为:
instance Monad LinkedList where
return = NodeWithoutNext
(NodeWithoutNext a) >>= f = f a
(Node a next) >>= f = build (f a)
where build (NodeWithoutNext x) = (Node x (next >>= f))
build (Node x xs) = Node x (build xs)
所以这里build
将迭代LinkedList
由 f a
制作,并且基本上构造一个几乎重复,除了当我们到达末尾时,我们将继续连接剩余元素的绑定(bind)结果。
关于haskell - 很难定义所有值的单子(monad),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47680039/
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 9 年前。 Improve th
我有一个带有输入字段的表单,使用javascript,当用户输入超过2个字符时,它会在第一个输入字段下方创建相同的输入字段。其代码是: Optie 1: 1 && treated[this.na
这是我的: char userInput; int position; vector userVector(7); vector someVector(7,1); cin >> userInput;
尝试使用 typescript 和 redux 构建一个简单的 react crud 应用程序并遇到以下问题。我有一个具有指定签名的函数,它将一个人对象作为参数,如此处所示。 export defau
哦,我多么希望 TCP 像 UDP 一样基于数据包! [查看评论] 但是,唉,事实并非如此,所以我正在尝试实现我自己的数据包层。这是到目前为止的事件链(忽略写入数据包) 哦,我的数据包结构非常简单:两
我想在我的页面底部放置一个包含不同数量图片的栏,这些图片(如果比页面宽)可以左右滚动。 页面宽度在变化,我希望 Pane 的宽度为 100%。 我试图通过让中间的 div 溢出并使用 jquery.a
我曾尝试在工作时将我的 Rails 应用程序 bundle 到我的 Mac 上。在家里它运行良好,我之前已经设法自己解决了它,但这次无论我尝试什么似乎都无法解决它。 我在运行 bundle/bundl
所以我有一个旧的网络表单站点,并且正在努力使其更易于维护。把它扔掉并重写它不是一种选择。 IoC 显然是它首先得到的东西之一,但这给我留下了服务定位器模式和糟糕的品味,并且想知道它是否可以做得更好。
我是一名优秀的程序员,十分优秀!