Search code examples
haskellsyntaxoperator-precedencefunction-composition

Why does function composition require parentheses?


Let's say I want to compose Text.pack with Text.strip.

:t (.) produces: (b -> c) -> (a -> b) -> a -> c

:t (Text.pack) produces: String -> Text

:t (Text.strip) produces: Text -> Text

So substituting strip for (b -> c) gives: b = Text c = Text

Substituting pack for (a -> b) gives: a = String b = Text

Lets verify: :t strip . pack produces: strip . pack :: String -> Text

ok, great lets try it:

strip.pack " example "

Produces:

Couldn't match expected type ‘a -> Text’ with actual type ‘Text’
Relevant bindings include
  it :: a -> Text (bound at <interactive>:31:1)
Possible cause: ‘pack’ is applied to too many arguments
In the second argument of ‘(.)’, namely ‘pack "    example     "’
In the expression: strip . pack "    example     "

(strip . pack) " example " works as expected.... why?


Solution

  • Function application has a higher priority than composition.

    strip.pack " example " is equivalent to strip.(pack " example "). This is one reason why people use $ to "suppress" application until after all the functions have been composed:

    strip . pack $ "    example     "