Haskell: Newbie currying mistake
As I mentioned in my last post I’ve spent a bit of this evening writing a merge sort function and one of the mistakes I made a few times was incorrectly passing arguments to the recursive calls of 'merge'.
For example, this is one of the earlier versions of the function:
middle :: [Int] -> Int
middle = floor . (\y -> y / 2) . fromIntegral . length
msort :: [Int] -> [Int]
msort unsorted =
let n = middle unsorted
in
if n == 0 then unsorted
else
let (left, right) = splitAt n unsorted
in merge (msort left) (msort right)
where
merge [] right = right
merge left [] = left
merge left@(x:xs) right@(y:ys) = if x < y then x : merge(xs, right) else y : merge (left, ys)
Which doesn’t actually compile:
Couldn't match expected type `[a0]' with actual type `[a0] -> [a0]'
In the return type of a call of `merge'
In the second argument of `(:)', namely `merge (xs, right)'
In the expression: x : merge (xs, right)
My defence for this mistake is that many of the other languages I’ve programmed in take function parameters separated by a comma but in this case I’ve actually only succeeded in currying the 'merge' function.
i.e. it thinks I only wanted to pass one value to it and return a function, hence the error message!
One correction would be to change the code to read like this so we can explicitly see that a function which takes 2 arguments can be called with each argument separately:
msort :: [Int] -> [Int]
msort unsorted =
let n = middle unsorted
in
if n == 0 then unsorted
else
let (left, right) = splitAt n unsorted
in merge (msort left) (msort right)
where
merge [] right = right
merge left [] = left
merge left@(x:xs) right@(y:ys) = if x < y then x : merge(xs)(right) else y : merge (left)(ys)
We can do exactly the same with the '/' function to use a simpler example:
-- a long way of dividing 2 by 3
> ((/) 2) 3
0.6666666666666666
The type of '/' is:
> :t (/)
(/) :: Fractional a => a -> a -> a
which means it takes in a 'Fractional' and then returns a function which takes in a 'Fractional' and returns a 'Fractional'.
In Haskell function application is left associative which means that 'f x y' is the same as '(f x) y' so we can omit the parentheses and pass both arguments together:
> (/) 2 3
0.6666666666666666
And we can do the same thing in the merge sort of course:
msort :: [Int] -> [Int]
msort unsorted =
let n = middle unsorted
in
if n == 0 then unsorted
else
let (left, right) = splitAt n unsorted
in merge (msort left) (msort right)
where
merge [] right = right
merge left [] = left
merge left@(x:xs) right@(y:ys) = if x < y then x : merge xs right else y : merge left ys
A fairly simple mistake to make but it had me confused for a while!
Updated the explanation around how we can pass arguments to '/' after my colleague https://twitter.com/!/gavri[Gavri] https://twitter.com/!/gavri/status/182331266681671680[pointed out that the initial explanation was incorrect].
About the author
I'm currently working on short form content at ClickHouse. I publish short 5 minute videos showing how to solve data problems on YouTube @LearnDataWithMark. I previously worked on graph analytics at Neo4j, where I also co-authored the O'Reilly Graph Algorithms Book with Amy Hodler.