I'm currently trying to learn Haskell, a purely functional language, so with that in mind, I figured I'd try to implement the bowling kata as a first exercise, using the HUnit test framework to test it. I'm sure that this implementation will broadcast clearly to anyone in the know that I come from an OO imperative programming world, so I'll welcome any constructive criticism as usual.
For those interested in looking into Haskell, both of these tutorials have been of great help so far.
For HUnit, Getting Started With HUnit has been great to quickly get the hang of it.
Anyway, without further ado, the Bowling Kata, Haskell style.
Bowling Kata in Haskell
I'm following the same test order as I did with the C# implementation, so my first test is for a game of all gutterballs. Just to show how the test module is set up, this first code snippet also includes the header info.
module ScorerTest where import HUnit import Scorer main = runTestTT $ TestList [allGuttersReturnZero] allGuttersReturnZero = TestCase $ assertEqual "All gutterballs" 0 (calculateScore $ replicate 20 0)And the following implementation of calculateScore looks like this. In short, define a function that takes a list of type number, and returns a number. To make the first test pass, we accept the list xs, but return zero no matter what.
module Scorer (calculateScore) where calculateScore :: (Num a) => [a] -> a calculateScore xs = 0The next test is to roll all ones, avoiding the more complicated spare and strike cases.
allBallsKnockDownOnePin = TestCase $ assertEqual "All balls roll one" 20 (calculateScore $ replicate 20 1)To make this pass for both test cases, we simply sum up all of the rolls passed in.
calculateScore :: (Num a) => [a] -> a calculateScore xs = sum xsThen we begin to get to the meat of the problem, this time rolling a strike on the first frame and following with ones for the rest of the rolls.
firstFrameStrikeRestOnes = TestCase $ assertEqual "First frame is strike, rest ones" 30 (calculateScore $ [10] ++ replicate 18 1)To make this test pass, a bit more work is needed.
First of all, since we're introducing recursion at this stage, the edge case needs to be covered, which is calculateScore being called with an empty list. We set it to return zero if that is the case.
Secondly, we split up the list coming in into head and tails (x:xs), and then check the head (x) to see if it's equal to 10 - a strike. If it is, we sum it up along with the next two values in the list (take 2 xs) and then run the calculation again on the rest of the list. If the value is not 10, we just take off the first element, and add it to the score calculation of the rest of the list.
More succinctly, it looks like the following.
calculateScore :: (Num a) => [a] -> a calculateScore [] = 0 calculateScore (x:xs) | x == 10 = x + sum (take 2 xs) + calculateScore xs | otherwise = x + calculateScore xsMoving on with the tests, we devise one where all of the rolls are strikes to see how the code performs. As expected, we get a score of 330 instead of 300, since the algorithm doesn't know to not count the last two balls as fully scoring strikes. So we consider frames instead of individual balls.
allFramesAreStrikes = TestCase $ assertEqual "All frames are strikes" 300 (calculateScore $ replicate 12 10)I have to admit that I got a little stuck here, since I'm still trying to get the hang of the functional way of thinking, so thanks to Tom Moertel, Isaac Gouy and Phillip Schwarz, whose solutions I found online to help me adjust my plan of attack.
Switching over to using frames as a basis instead of balls required a bit more extensive code rewriting, including breaking frame scoring out into its own function, which takes the list of rolls, the current total and the frame as arguments.
The function itself is split into three cases. First a case to make sure the calculation stops after the 10th frame, then a case to handle the last frame if there are no special rolls. These first two are our edge cases.
Lastly, the general case for the remaining rolls scores the list recursively. It splits the list of rolls into four components - the first three balls and the remaining list. If the first roll is 10 (a strike), it then scores the next frame along with the remaining list, at the same time adding the first three balls to the score. Otherwise it scores the same way, but adding only the first two balls to the score.
This is how it looked when I wrote it.
calculateScore :: (Num a) => [a] -> a calculateScore rolls = scorePerFrame rolls 0 0 scorePerFrame :: (Num a) => [a] -> a -> a -> a scorePerFrame rolls score 10 = score scorePerFrame [x,y] score frame = score+x+y scorePerFrame (x:y:z:rest) score frame | x == 10 = scorePerFrame (y:z:rest) (score+x+y+z) (frame+1) | otherwise = scorePerFrame (z:rest) (score+x+y) (frame+1)Once this framework is in place, adding the case for spares is pretty simple. First the test case.
firstFrameSpareRestOnes = TestCase $ assertEqual "First frame is spare, rest ones" 29 (calculateScore $ [5,5] ++ replicate 18 1)And then we just add a line that matches a spare.
scorePerFrame (x:y:z:rest) score frame | x == 10 = scorePerFrame (y:z:rest) (score+x+y+z) (frame+1) | x + y == 10 = scorePerFrame (z:rest) (score+x+y+z) (frame+1) | otherwise = scorePerFrame (z:rest) (score+x+y) (frame+1)In theory, this solution should work fine if I roll all spares as well, but for my own sense of security, I'll write that test anyway.
allFramesAreSpares = TestCase $ assertEqual "All frames are spares" 150 (calculateScore $ replicate 21 5)It passes right away, so no code changes needed. So there you have it, the bowling scorer in Haskell.
The final code follows below. I tried to do a little refactoring, but in the end I found the current terse style was more legible. Part of that I think is a nature of the Haskell syntax. x, y and z could perhaps be replaced with "first, second and third," but I thought it just made it look bulky.
The Tests
module ScorerTest where import HUnit import Scorer main = runTestTT $ TestList [ allGuttersReturnZero, allBallsKnockDownOnePin, firstFrameStrikeRestOnes, allFramesAreStrikes, firstFrameSpareRestOnes, allFramesAreSpares ] allGuttersReturnZero = TestCase $ assertEqual "All gutterballs" 0 (calculateScore $ replicate 20 0) allBallsKnockDownOnePin = TestCase $ assertEqual "All balls roll one" 20 (calculateScore $ replicate 20 1) firstFrameStrikeRestOnes = TestCase $ assertEqual "First frame is strike, rest ones" 30 (calculateScore $ [10] ++ replicate 18 1) allFramesAreStrikes = TestCase $ assertEqual "All frames are strikes" 300 (calculateScore $ replicate 12 10) firstFrameSpareRestOnes = TestCase $ assertEqual "First frame is spare, rest ones" 29 (calculateScore $ [5,5] ++ replicate 18 1) allFramesAreSpares = TestCase $ assertEqual "All frames are spares" 150 (calculateScore $ replicate 21 5)The Code
module Scorer (calculateScore) where calculateScore :: (Num a) => [a] -> a calculateScore rolls = scorePerFrame rolls 0 0 scorePerFrame :: (Num a) => [a] -> a -> a -> a scorePerFrame rolls score 10 = score scorePerFrame [x,y] score frame = score+x+y scorePerFrame (x:y:z:rest) score frame | x == 10 = scorePerFrame (y:z:rest) (score+x+y+z) (frame+1) | x + y == 10 = scorePerFrame (z:rest) (score+x+y+z) (frame+1) | otherwise = scorePerFrame (z:rest) (score+x+y) (frame+1)