如何使用镜头列表?

时间:2017-11-05 08:15:33

标签: haskell traversal lens

我可以使用Traversal列表吗?以下代码:

f::[Int] -> [[Int]]
f l = [l & i .~ 1 |  i<-[ix 0], (l^? i) == Just 0]

产生错误:

  • Couldn't match type ‘Const (Data.Monoid.First Int) [Int]’
                     with ‘Identity [Int]’
      Expected type: ASetter [Int] [Int] Int Integer
        Actual type: Getting (Data.Monoid.First Int) [Int] Int
    • In the first argument of ‘(.~)’, namely ‘i’
      In the second argument of ‘(&)’, namely ‘i .~ 1’
      In the expression: l & i .~ 1

看着this question我想我需要以某种方式明确地给我一个类型,但我的每次尝试都失败了。

1 个答案:

答案 0 :(得分:3)

问题不在于明确指定类型。每当你想要一个镜头或遍历的容器(镜头内对,镜头内部列表,镜头内Maybe),你需要使用ReifiedLens

请参阅此问题以获得解释:

Why do we need Control.Lens.Reified?

所以你的例子应该这样写:

import Control.Lens

f :: [Int] -> [[Int]]
f l = [ l & i .~ 1 
      | Traversal i <- [Traversal $ ix 0]
      , l ^? i == Just 0
      ]
  

请注意,此处TraversalReifiedTraversal类型的构造函数。

它的工作原理如下:

ghci> f [0,0,0]
[[1,0,0]]