我正在尝试理解this article中的代码。它解释了使用归纳图,这似乎非常好,并且在某些时候它定义了深度优先搜索归纳图。它的代码如下:
dfs :: Graph.Node -> Gr a b -> [Node]
dfs start graph = go [start] graph
where go [] _ = []
go _ g | Graph.isEmpty g = []
go (n:ns) (match n -> (Just c, g)) =
n : go (Graph.neighbors' c ++ ns) g
go (_:ns)
我不明白这两行:
go (n:ns) (match n -> (Just c, g)) =
n : go (Graph.neighbors' c ++ ns) g
似乎它正在定义函数go
,它将列表作为第一个参数,由(n:ns)
进行模式匹配。然而,第二个论点我不明白:(match n -> (Just c, g))
。运营商->
在这里意味着什么?通过查看运算符,它可以是以下三种情况之一:
由于没有case
语句,也没有lambda表达式的反斜杠转义变量,因此只能是函数类型映射运算符。在这种情况下,我不知道它是如何绑定这些变量c
和g
的?它到底是什么意思,怎么会出现在争论中呢?
提前致谢!
答案 0 :(得分:8)
->
既不表示函数类型,也不表示lambda-definition或case-mapping。这是view pattern。
go (n:ns) (match n -> (Just c, g)) =
n : go (Graph.neighbors' c ++ ns) g
相当于
go (n:ns) g'
| (Just c, g) <- match n g'
= n : go (Graph.neighbors' c ++ ns) g
pattern guard (Just c, g) <- match n g'
依次为
go (n:ns) g' = case match n g' of
(Just c, g) -> n : go (Graph.neighbors' c ++ ns) g
(Nothing, g) -> ...
Nothing
子句需要代表go
定义的后一句。
答案 1 :(得分:3)
这是view pattern。这个扩展允许我们在匹配之前以某种方式转换参数。
如果没有此扩展名,代码必须按如下方式编写:
go (n:ns) g' =
case match n g' of
(Just c, g) -> ...
...
这有点冗长。