Extreme branchless: Retrospective
3 months ago, I have started a branchless journey, the goal was to push the branchless contraint to its limits, I have applied to the following 14 katas:
- Game of Life
- Mars Rover
- Supermarket pricing
- Bowling kata
- Cupcake
- Cupcake
- FooBarQix
- Gilded rose
- Gilded rose
- Langton Ant
- Mastermind
- Minesweeper
- Pagination Seven
Few takeaways:
- Branchless is engineered, not discovered
Taking Pagination Seven as an example, the main tactic used here was to store rendering strategies in lists and picking the right one according to the total/current page.
It takes a conscious effort to not use a if/then/else statement.
- Branchless can spread responsibilities, reducing cohesion
We can use Mars Rover to show it:
north =
  Direction
  { displayDirection = "N"
  , nextRotateLeft = west
  , nextRotateRight = east
  , moveForward = \position -> position {y = position.y + 1}
  }
east =
  Direction
  { displayDirection = "E"
  , nextRotateLeft = north
  , nextRotateRight = south
  , moveForward = \position -> position {x = position.x + 1}
  }
south =
  Direction
  { displayDirection = "S"
  , nextRotateLeft = east
  , nextRotateRight = west
  , moveForward = \position -> position {y = position.y - 1}
  }
west =
  Direction
  { displayDirection = "W"
  , nextRotateLeft = south
  , nextRotateRight = north
  , moveForward = \position -> position {x = position.x - 1}
  }
With a sum type, we would have 4 functions, pattern-matching on each constructor.
I think it's a tradeoff, it could be a good solution if you are in a really open/extendable/dynamic context.
- Branchless can make simple thing hard, but it can also make it simpler
If you had a look at FooBarQix, you may recall my trick to had division:
splitBase10 n = splitBase10 u <> [mod]
  where (u, mod) = chainedOf n $ divMod10_0 0
data Chained a = Chained { value :: a, next :: Chained a }
chainedOf (Natural n) = (.value) . n (.next)
divMod10_0, divMod10_1, divMod10_2, divMod10_3, divMod10_4, divMod10_5, divMod10_6, divMod10_7, divMod10_8, divMod10_9 :: Natural -> Chained (Natural, Natural)
divMod10_0 u = Chained (u, 0) $ divMod10_1 u
divMod10_1 u = Chained (u, 1) $ divMod10_2 u
divMod10_2 u = Chained (u, 2) $ divMod10_3 u
divMod10_3 u = Chained (u, 3) $ divMod10_4 u
divMod10_4 u = Chained (u, 4) $ divMod10_5 u
divMod10_5 u = Chained (u, 5) $ divMod10_6 u
divMod10_6 u = Chained (u, 6) $ divMod10_7 u
divMod10_7 u = Chained (u, 7) $ divMod10_8 u
divMod10_8 u = Chained (u, 8) $ divMod10_9 u
divMod10_9 u = Chained (u, 9) $ divMod10_0 (u + 1)
I reworked it on my own a bit later to:
splitBase10 n = runNatural n (const nonNull) []
  where (u, mod) = head $ runNatural n tail $ zip (concatMap (replicate 10) [0..]) (cycle [0..9])
        nonNull = splitBase10 u <> [mod]
- Branchless can be derived mechanically
There's actually two ways to go branchless:
From algebraic data-types (ADT):
data T = A Int | B String Bool
-- Becomes
newtype T = T { runT :: forall a. (Int -> a) -> (String -> Bool -> a) -> a }
Or from control structures:
agedBrie (sellIn - 1) . min 50 $
  if sellIn < 1
  then quality + 2
  else quality + 1
-- Becomes
agedBrie (sellIn - 1) . min 50 $
  ((quality + 2) : repeat (quality + 1)) `at` sellIn
- I won't use it daily, kind of
Brancheless is not something we see in really-world code bases, it's the opposite.
We have plenty of nested conditionals, which comes from, either a lack of understanding for the product team, or a specifically requests from the users or the business units.
However, doing these katas has shifted my mind on software design, like I wasn't for years, and my designing style has changed since.