# Difference between revisions of "List comprehension"

m |
m (add linebreaks for better readability) |
||

Line 14: | Line 14: | ||

One may have multiple generators, separated by commas, such as |
One may have multiple generators, separated by commas, such as |
||

<haskell> |
<haskell> |
||

− | [(i,j) | i <- [1,2], |
+ | [(i,j) | i <- [1,2], |

+ | j <- [1..4] ] |
||

</haskell> |
</haskell> |
||

yielding the result |
yielding the result |
||

Line 22: | Line 22: | ||

Note how each successive generator refines the results of the previous generator. Thus, if the second list is infinite, one will never reach the second element of the first list. For example, |
Note how each successive generator refines the results of the previous generator. Thus, if the second list is infinite, one will never reach the second element of the first list. For example, |
||

<haskell> |
<haskell> |
||

− | take 10 [ (i,j) | i <- [1,2], |
+ | take 10 [ (i,j) | i <- [1,2], |

+ | j <- [1..] ] |
||

</haskell> |
</haskell> |
||

yields |
yields |
||

Line 30: | Line 30: | ||

In such a situation, a nested sequence of list comprehensions may be appropriate. For example, |
In such a situation, a nested sequence of list comprehensions may be appropriate. For example, |
||

<haskell> |
<haskell> |
||

− | take 5 [[ (i,j) | i <- [1,2]] | j <- [1..]] |
+ | take 5 [ [ (i,j) | i <- [1,2] ] | j <- [1..] ] |

</haskell> |
</haskell> |
||

yields |
yields |
||

Line 39: | Line 39: | ||

One can also provide boolean guards. For example, |
One can also provide boolean guards. For example, |
||

<haskell> |
<haskell> |
||

− | take 10 [ (i,j) | i <- [1..], |
+ | take 10 [ (i,j) | i <- [1..], |

+ | j <- [1..i-1], |
||

+ | gcd i j == 1 ] |
||

</haskell> |
</haskell> |
||

yields |
yields |
||

Line 48: | Line 48: | ||

Finally, one can also make local let declarations. For example, |
Finally, one can also make local let declarations. For example, |
||

<haskell> |
<haskell> |
||

− | take 10 [ (i,j) | i <- [1..], |
+ | take 10 [ (i,j) | i <- [1..], |

+ | let k = i*i, |
||

+ | j <- [1..k] ] |
||

</haskell> |
</haskell> |
||

yields |
yields |

## Revision as of 11:32, 23 April 2016

List comprehensions are syntactic sugar like the expression

```
import Data.Char (toUpper)
[toUpper c | c <- s]
```

where `s :: String`

is a string such as `"Hello"`

.
Strings in Haskell are lists of characters; the generator `c <- s`

feeds each character of `s`

in turn to the left-hand expression `toUpper c`

, building a new list.
The result of this list comprehension is `"HELLO"`

.
(Of course, in this simple example you would just write `map toUpper s`

.)

## Examples

One may have multiple generators, separated by commas, such as

```
[(i,j) | i <- [1,2],
j <- [1..4] ]
```

yielding the result

```
[(1,1),(1,2),(1,3),(1,4),(2,1),(2,2),(2,3),(2,4)]
```

Note how each successive generator refines the results of the previous generator. Thus, if the second list is infinite, one will never reach the second element of the first list. For example,

```
take 10 [ (i,j) | i <- [1,2],
j <- [1..] ]
```

yields

```
[(1,1),(1,2),(1,3),(1,4),(1,5),(1,6),(1,7),(1,8),(1,9),(1,10)]
```

In such a situation, a nested sequence of list comprehensions may be appropriate. For example,

```
take 5 [ [ (i,j) | i <- [1,2] ] | j <- [1..] ]
```

yields

```
[[(1,1),(2,1)], [(1,2),(2,2)], [(1,3),(2,3)], [(1,4),(2,4)], [(1,5),(2,5)]]
```

One can also provide boolean guards. For example,

```
take 10 [ (i,j) | i <- [1..],
j <- [1..i-1],
gcd i j == 1 ]
```

yields

```
[(2,1),(3,1),(3,2),(4,1),(4,3),(5,1),(5,2),(5,3),(5,4),(6,1)]
```

Finally, one can also make local let declarations. For example,

```
take 10 [ (i,j) | i <- [1..],
let k = i*i,
j <- [1..k] ]
```

yields

```
[(1,1),(2,1),(2,2),(2,3),(2,4),(3,1),(3,2),(3,3),(3,4),(3,5)]
```

Here is an example of a nested sequence of list comprehensions, taken from code implementing the Sieve of Atkin:

```
[[[ poly x y
| i <- [0..], let x = m + 60*i, test x y ]
| j <- [0..], let y = n + 60*j ]
| m <- [1..60], n <- [1..60], mod (poly m n) 60 == k ]
```

The result is a list of infinite lists of infinite lists.

The specification of list comprehensions is given in The Haskell 98 Report: 3.11 List Comprehensions.

The GHC compiler supports parallel list comprehensions as an extension; see GHC User's Guide: 7.3.4. Parallel List Comprehensions.

## List monad

In the first versions of Haskell, the comprehension syntax was available for all monads. (See History of Haskell)
Later the comprehension syntax was restricted to lists.
Since lists are an instance of monads, you can get list comprehension in terms of the `do`

notation.
Because of this, several Haskell programmers consider the list comprehension unnecessary now.

The examples from above can be translated to list monad as follows:

```
do c <- s
return (toUpper c)
```

```
do i <- [1,2]
j <- [1..4]
return (i,j)
```

or

```
liftM2 (,) [1,2] [1..4]
```

```
do j <- [1..]
return
(do i <- [1,2]
return (i,j))
```

```
do i <- [1..]
j <- [1..i-1]
guard (gcd i j == 1)
return (i,j)
```

```
do i <- [1..]
let k = i*i
j <- [1..k]
return (i,j)
```

```
do m <- [1..60]
n <- [1..60]
guard (mod (poly m n) 60 == k)
return $
do j <- [0..]
let y = n + 60*j
return $
do i <- [0..]
let x = m + 60*i
guard (test x y)
return (poly x y)
```