# Lazy Sort: Counting Comparisons

Published on September 17, 2020 under the tag haskell

### Introduction

Haskell’s laziness allows you to do many cool things. I’ve talked about searching an infinite graph before. Another commonly mentioned example is finding the smallest N items in a list.

Because programmers are lazy as well, this is often defined as:

```
smallestN_lazy :: Ord a => Int -> [a] -> [a]
= take n . sort smallestN_lazy n
```

This happens regardless of the language of choice if we’re confident that the list will not be too large. It’s more important to be correct than it is to be fast.

However, in strict languages we’re really sorting the entire list before taking the first N items. We can implement this in Haskell by forcing the length of the sorted list.

```
smallestN_strict :: Ord a => Int -> [a] -> [a]
= let l1 = sort l0 in length l1 `seq` take n l1 smallestN_strict n l0
```

If you’re at least somewhat familiar with the concept of laziness, you may intuitively realize that the lazy version of `smallestN`

is much better since it’ll only sort as far as it needs.

But how much better does it actually do, with Haskell’s default `sort`

?

### A better algorithm?

For the sake of the comparison, we can introduce a third algorithm, which does a slightly smarter thing by keeping a heap of the smallest elements it has seen so far. This code is far more complex than `smallestN_lazy`

, so if it performs better, we should still ask ourselves if the additional complexity is worth it.

```
smallestN_smart :: Ord a => Int -> [a] -> [a]
= do
smallestN_smart maxSize list <- Map.toList heap
(item, n) replicate n item
where
-- A heap is a map of the item to how many times it occurs in
-- the heap, like a frequency counter.
= foldl' (\acc x -> insert x acc) Map.empty list
heap
insert x heap0| Map.size heap0 < maxSize = Map.insertWith (+) x 1 heap0
| otherwise = case Map.maxViewWithKey heap0 of
Nothing -> Map.insertWith (+) x 1 heap0
Just ((y, yn), _) -> case compare x y of
EQ -> heap0
GT -> heap0
LT ->
let heap1 = Map.insertWith (+) x 1 heap0 in
if yn > 1
then Map.insert y (yn - 1) heap1
else Map.delete y heap1
```

So, we get to the main trick I wanted to talk about: how do we benchmark this, and can we add unit tests to confirm these benchmark results in CI? Benchmark execution times are very fickle. Instruction counting is awesome but perhaps a little overkill.

Instead, we can just count the number of comparisons.

### Counting comparisons

We can use a new type that holds a value and a number of ticks. We can increase the number of ticks, and also read the ticks that have occurred.

```
data Ticks a = Ticks {ref :: !(IORef Int), unTicks :: !a}
mkTicks :: a -> IO (Ticks a)
= Ticks <$> IORef.newIORef 0 <*> pure x
mkTicks x
tick :: Ticks a -> IO ()
= IORef.atomicModifyIORef' (ref t) $ \i -> (i + 1, ())
tick t
ticks :: Ticks a -> IO Int
= IORef.readIORef . ref ticks
```

`smallestN`

has an `Ord`

constraint, so if we want to count the number of comparisons we’ll want to do that for both `==`

and `compare`

.

```
instance Eq a => Eq (Ticks a) where
==) = tick2 (==)
(
instance Ord a => Ord (Ticks a) where
compare = tick2 compare
```

The actual ticking code goes in `tick2`

, which applies a binary operation and increases the counters of both arguments. We need `unsafePerformIO`

for that but it’s fine since this lives only in our testing code and not our actual `smallestN`

implementation.

```
tick2 :: (a -> a -> b) -> Ticks a -> Ticks a -> b
= unsafePerformIO $ do
tick2 f t1 t2
tick t1
tick t2pure $ f (unTicks t1) (unTicks t2)
{-# NOINLINE tick2 #-}
```

### Results

Let’s add some benchmarking that prints an ad-hoc CSV:

```
main :: IO ()
= do
main let listSize = 100000
= [smallestN_strict, smallestN_lazy, smallestN_smart]
impls 50, 100 .. 2000] $ \sampleSize -> do
forM_ [<- replicateM listSize randomIO :: IO [Int]
l <- fmap unzip $ forM impls $ \f -> do
(nticks, results) <- traverse mkTicks l
l1 let !r1 = sum . map unTicks $ f sampleSize l1
<- sum <$> traverse ticks l1
t1 pure (t1, r1)
. fail $
unless (equal results) "Different results: " ++ show results
putStrLn . intercalate "," . map show $ sampleSize : nticks
```

Plug that CSV into a spreadsheet and we get this graph. What conclusions can we draw?

Clearly, both the lazy version as well as the “smart” version are able to avoid a large number of comparisons. Let’s remove the strict version so we can zoom in.

What does this mean?

If the

`sampleSize`

is small, the heap implementation does less comparions. This makes sense: even if treat`sort`

as a black box, and don’t look at it’s implementation, we can assume that it is not*optimally lazy*; so it will always sort “a bit too much”.As

`sampleSize`

gets bigger, the insertion into the bigger and bigger heap starts to matter more and more and eventually the naive lazy implementation is faster!Laziness is awesome and

`take N . sort`

is absolutely the first implementation you should write, even if you replace it with a more efficient version later.Code where you count a number of calls is very easy to do in a test suite. It doesn’t pollute the application code if we can patch in counting through a typeclass (

`Ord`

in this case).

Can we say something about the complexity?

The complexity of

`smallestN_smart`

is basically inserting into a heap`listSize`

times. This gives us`O(listSize * log(sampleSize))`

.That is of course the worst case complexity, which only occurs in the special case where we need to insert into the heap at each step. That’s only true when the list is sorted, so for a random list the average complexity will be a lot better.

The complexity of

`smallestN_lazy`

is far harder to reason about. Intuitively, and with the information that`Data.List.sort`

is a merge sort, I came to something like`O(listSize * max(sampleSize, log(listSize)))`

. I’m not sure if this is correct, and the case with a random list seems to be faster.I would be very interested in knowing the actual complexity of the lazy version, so if you have any insights, be sure to let me know!

**Update**: Edward Kmett corrected me: the complexity of`smallestN_lazy`

is actually`O(listSize * min(sampleSize, listSize))`

, with`O(listSize * min(sampleSize, log(listSize))`

in expectation for a random list.

### Appendix

Helper function: check if all elements in a list are equal.

```
equal :: Eq a => [a] -> Bool
: y : zs) = x == y && equal (y : zs)
equal (x = True equal _
```