Welcome to Software Development on Codidact!
Will you help us build our independent community of developers helping developers? We're small and trying to grow. We welcome questions about all aspects of software development, from design to code to QA and more. Got questions? Got answers? Got code you'd like someone to review? Please join us.
Comments on How to use function composition for applying a function to first elements of a list?
Parent
How to use function composition for applying a function to first elements of a list?
Can anyone explain to me why my Haskell function gives rise to a type-definition error?
Originally, I wrote the following function to subtract one from the first n
elements in a list:
dec_first :: Int -> [Int] -> [Int]
dec_first 0 l = l
dec_first n (x:xs) = (x-1):dec_first (n-1) xs
However, as a challenge, I wanted to write this function using available standard functions, using function compositions. This is how I ended up writing the function (for the subtraction part)
dec_first :: Int -> [Int] -> [Int]
dec_first = map (subtract 1) . take
but this function does not compile due to a type mismatch.
In my attempt to fix the error, I also tried the following variant:
dec_first :: Int -> [Int] -> [Int]
dec_first n = map (subtract 1) . take n
This function actually does work as expected. However, if I specify all parameters again, things breaks again:
dec_first :: Int -> [Int] -> [Int]
dec_first n l = map (subtract 1) . take n l
Can anyone help me understand why this function only works with one variable, but not in the other settings?
Post
The following users marked this post as Works for me:
User | Comment | Date |
---|---|---|
mr Tsjolder | (no comment) | Jul 13, 2023 at 17:33 |
The quick fix is:
map (subtract 1) . take n $ l
When I put your function in my own Haskell compiler, I get:
Possible cause: ‘take’ is applied to too many arguments
Remember that in Haskell, we don't have functions with multiple parameters. We have functions with a single parameter, that return either values or new functions. (That, in turn, only take a single parameter).
map (subtract 1) . take n l
is read by Haskell as (map (subtract 1)) . (take n l)
.
- The first part is a function that maps a list of numbers to a list of numbers. It has type
(Num b) => [b] -> [b]
. - The second part is a function that maps an integer to a function that takes a list and returns a list. It has type
Int -> [a] -> [a]
.
So it tries to do function composition, where the first function is of type ([b] -> [b])
, and the second of type (Int -> [a] -> [a])
. That's not compatible.
We can make it work by writing it as (map (subtract 1)) . take n)( l )
, or the shorthand function map (subtract 1)) . take n $ l
. Now we have a function that subtracts 1 from every element in a list, and takes the first n elements of the result. This function then gets our list as argument.
In your comment, you pointed out that you also tried:
dec_first = map (subtract 1) . take
This gave you a different error message:
Probable cause: `take' is applied to too few arguments
This time, we have the opposite situation. The function map (subtract 1)
is a neat function that takes a list of Num
and returns a list of Num
: it is of type (Num a) => [a] -> [a]
.
You then try to apply this to take
, whose type is Int -> [a] -> [a]
. It's a function that takes an integer, to yield a function that takes a list and yields a list. But... Haskell doesn't see any integers!
We can fix this by giving it the Integer it wants:
dec_first n = map (subtract 1) . take n
If you're using GHCi, you can put :t
in front of a function to see its type. For example:
ghci> :t map (subtract 1)
map (subtract 1) :: Num b => [b] -> [b]
This can be very helpful to see if the type of your function is what you expected it to be, and to see if it matches the functions you're feeding it.
0 comment threads