I am always interested in learning new languages, a fact that keeps me on my toes and makes me (I believe) a better programmer. My attempts at conquering Haskell come and go - twice so far - and I decided it was time to try again. 3rd time's the charm, right?
Nope. I re-read my old notes... and get disappointed :-(
The problem that made me lose faith last time, was an easy one: permutations of integers. i.e. from a list of integers, to a list of lists - a list of their permutations:
[int] -> [[int]]
This is in fact a generic problem, so replacing 'int' above with 'a', would still apply.
From my notes:
I code it first on my own, I succeed. Hurrah!
I send my solution to a good friend of mine - Haskell guru, it usually helps to learn from gurus - and he sends me this, which I am told, "expresses the true power of the language, the use of generic facilities to code your needs". All for it, I recently drank the kool-aid, let's go:
permute :: [a] -> [[a]] permute = foldr (concatMap.ins) [[]] where ins x [] = [[x]] ins x (y:ys) = (x:y:ys):[ y:res | res <- ins x ys]
Hmm. Let's break this down:
bash$ cat b.hs ins x [] = [[x]] ins x (y:ys) = (x:y:ys):[ y:res | res <- ins x ys] bash$ ghci Prelude> :load b.hs [1 of 1] Compiling Main ( b.hs, interpreted ) Ok, modules loaded: Main. *Main> ins 1 [2,3] [[1,2,3],[2,1,3],[2,3,1]]
OK, so far, so good. Took me a minute to understand the second line of "ins", but OK: It places the 1st arg in all possible positions in the list. Cool.
Now, to understand the foldr and concatMap. in "Real world Haskell", the DOT was explained...
(f . g) x
...as just another syntax for...
f (g x)
And in the code the guru sent, DOT was used from a foldr, with the "ins" function as the fold "collapse":
*Main> let g=concatMap . ins *Main> g 1 [[2,3]] [[1,2,3],[2,1,3],[2,3,1]]
OK, since I want to understand how the DOT is used by the guru, I try the equivalent expression according to the DOT definition, (f . g) x = f (g x) ...
*Main> concatMap (ins 1 [[2,3]]) <interactive>:1:11: Couldn't match expected type `a -> [b]' against inferred type `[[[t]]]' In the first argument of `concatMap', namely `(ins 1 [[2, 3]])' In the expression: concatMap (ins 1 [[2, 3]]) In the definition of `it': it = concatMap (ins 1 [[2, 3]])
What!?! Why? OK, I check concatMap's signature, and find that it needs a lambda and a list, but that's just a human thinking; how does GHC cope? According to the definition of DOT above...
(f.g)x = f(g x),
...what I did was valid, replace-wise:
(concatMap . ins) x y = concatMap (ins x y)
Scratching head...
*Main> concatMap (ins 1) [[2,3]] [[1,2,3],[2,1,3],[2,3,1]]
So... The DOT explanation was apparently too simplistic... DOT must be somehow clever enough to understand that we in fact wanted "ins" to get curri-ed away and "eat" the first argument - thus becoming a function that only wants to operate on [t] (and "intersperse" them with '1' in all possible positions).
But where was this specified? How did GHC knew to do this, when we invoked:
*Main> (concatMap . ins) 1 [[2,3]] [[1,2,3],[2,1,3],[2,3,1]]
Did the "ins" signature somehow conveyed this... "eat my first argument" policy?
*Main> :info ins ins :: t -> [t] -> [[t]] -- Defined at b.hs:1:0-2
I don't see nothing special - "ins" is a function that takes a 't', a list of 't', and proceeds to create a list with all "interspersals". Nothing about "eat your first argument and curry it away".
So there... I am baffled. I understand (after an hour of looking at the code!) what goes on, but... God almighty... Perhaps GHC makes attempts to see how many arguments it can "peel off"?
let's try with no argument "curried" into "ins", oh gosh, boom, let's try with one argument "curried" into "ins", yep, works, that must be it, proceed)
Again - yikes...
And since I am always comparing the languages I am learning with what I already know, how would "ins" look in Python?
a=[2,3] print [a[:x]+[1]+a[x:] for x in xrange(len(a)+1)] [[1, 2, 3], [2, 1, 3], [2, 3, 1]]
Be honest, now... which is simpler?
I mean, I know I am a newbie in Haskell, but I feel like an idiot... Looking at 4 lines of code for an hour, and ending up assuming that the compiler... tries various interpretations until it finds something that "clicks"?
To quote from Lethal weapon, "I am too old for this"...
In Haskell, all functions are considered curried: That is, all functions in Haskell take just one argument. This is mostly hidden in notation, and so may not be apparent to a new Haskeller.
Currying: A function returning another function that might return another function, but every returned function must take only one parameter at a time. Partial application: A function returning another function that might return another function, but each returned function can take several parameters.
Currying is helpful when you have to frequently call a function with a fixed argument. Considering, for example, the following function: If we want to define the function error , warn , and info , for every type, we have two options. Currying provides a shorter, concise, and more readable solution.
(f . g) x = f (g x)
This is true. You concluded from that that
(f . g) x y = f (g x y)
must also be true, but that is not the case. In fact, the following is true:
(f . g) x y = f (g x) y
which is not the same.
Why is this true? Well (f . g) x y
is the same as ((f . g) x) y
and since we know that (f . g) x = f (g x)
we can reduce that to (f (g x)) y
, which is again the same as f (g x) y
.
So (concatMap . ins) 1 [[2,3]]
is equivalent to concatMap (ins 1) [[2,3]]
. There is no magic going on here.
Another way to approach this is via the types:
.
has the type (b -> c) -> (a -> b) -> a -> c
, concatMap
has the type (x -> [y]) -> [x] -> [y]
, ins
has the type t -> [t] -> [[t]]
. So if we use concatMap
as the b -> c
argument and ins
as the a -> b
argument, then a
becomes t
, b
becomes [t] -> [[t]]
and c
becomes [[t]] -> [[t]]
(with x
= [t]
and y
= [t]
).
So the type of concatMap . ins
is t -> [[t]] -> [[t]]
, which means a function taking a whatever and a list of lists (of whatevers) and returning a list of lists (of the same type).
I'd like to add my two cents. The question and answer make it sound like .
is some magical operator that does strange things with re-arranging function calls. That's not the case. .
is just function composition. Here's an implementation in Python:
def dot(f, g): def result(arg): return f(g(arg)) return result
It just creates a new function which applies g
to an argument, applies f
to the result, and returns the result of applying f
.
So (concatMap . ins) 1 [[2, 3]]
is saying: create a function, concatMap . ins
, and apply it to the arguments 1
and [[2, 3]]
. When you do concatMap (ins 1 [[2,3]])
you're instead saying, apply the function concatMap
to the result of applying ins
to 1
and [[2, 3]]
- completely different, as you figured out by Haskell's horrendous error message.
UPDATE: To stress this even further. You said that (f . g) x
was another syntax for f (g x)
. This is wrong! .
is just a function, as functions can have non-alpha-numeric names (>><
, ..
, etc., could also be function names).
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With