Could someone explain functional lenses to me? It's a surprisingly difficult subject to google for and I haven't made any progress. All I know is that they provide similar get/set functionality than in OO.
Asked
Active
Viewed 1.7k times
83
-
7There is [a nice introduction to lenses](http://www.youtube.com/watch?v=efv0SQNde5Q) by Edward Kmett on YouTube. The examples are in Scala, but it shouldn't be too hard to follow. – hammar Nov 29 '11 at 08:30
-
Yup, tried to watch those, but having enough time while I'm still alert, is not so easy :P – Masse Nov 29 '11 at 13:02
-
2@Jochen: The lenses described there don't really have much in common with the lenses this question is about. – sclv Nov 29 '11 at 17:50
-
3Here is nice introduction using pictures: [Lenses in Pictures](http://adit.io/posts/2013-07-22-lenses-in-pictures.html). – Debjit Mar 03 '15 at 03:29
2 Answers
62
A lens consists of two functions, a getter and a setter:
data Lens a b = Lens { getter :: a -> b, setter :: b -> a -> a }
For example, we might have lenses for the first and second parts of a pair:
fstLens :: Lens (a, b) a
fstLens = Lens fst $ \x (a, b) -> (x, b)
sndLens :: Lens (a, b) b
sndLens = Lens snd $ \x (a, b) -> (a, x)
The real convenience of lenses is that they compose:
compose :: Lens b c -> Lens a b -> Lens a c
compose f g = Lens (getter f . getter g) $
\c a -> setter g (setter f c (getter g a)) a
And they mechanically convert to State
transitions:
lensGet :: MonadState s m => Lens s a -> m a
lensGet = gets . getter
lensSet :: MonadState s m => Lens s b -> b -> m ()
lensSet f = modify . setter f
lensMod :: MonadState s m => Lens s b -> (b -> b) -> m ()
lensMod f g = modify $ setter f =<< g . getter f
(+=) :: (MonadState s m, Num b) => Lens s b -> b -> m ()
f += x = lensMod f (+ x)

Apocalisp
- 34,834
- 8
- 106
- 155
-
Your compose example didn't typecheck. GHC infers; Lens a a -> Lens a a -> Lens a a – Masse Nov 29 '11 at 21:17
-
-
It still doesn't type check into a->c. It infers into compose :: Lens a b -> Lens a a -> Lens a b – Masse Nov 30 '11 at 08:50
12
See the answer to question lenses, fclabels, data-accessor - which library for structure access and mutation is better - it has a very clear explanation on lenses.
Also, the documentation for the Data.Lenses and fclabel libraries give some good examples of them being used.

Community
- 1
- 1

David Miani
- 14,518
- 2
- 47
- 66