(summarize discussion at HaskellPrime mailing list from an advocators view)
(Yet another use case for if'. Although I'd prefer not to give this use case a name at all.)
Revision as of 09:18, 24 October 2006
1 Replace syntactic sugar by a function
For processing conditions,the
if' :: Bool -> a -> a -> a if' True x _ = x if' False _ y = y
Unfortunately there is no such function in the Prelude.
2.1 AdvantagesThe advantages of the function
are the same like for all such alternatives. So let me repeat two important non-syntactic strengths of Haskell:
- types: classification, documentation
- higher order functions: combinators
each language tool can process it without hassle. Haddock can generate documentation for it, a text editor can make suggestions for values to insert, Hoogle can retrieve that function.
For example, the Hoogle query
[Bool] -> [a] -> [a] -> [a]
2.2 Use casesEach of the following functions could be defined in terms of
Actually, they do not even need to be in Prelude because they can be constructed so easily.
That function is harder to explain in English, than by its implementation. :-)
zipIf :: [Bool] -> [a] -> [a] -> [a] zipIf = zipWith3 if'
Select a member of a pair.
This resembles the
cond?x:y operation of the C language.
infixr 1 ?: (?:) :: Bool -> (a,a) -> a (?:) = uncurry . if'
From a list of expressions choose the one, whose condition is true. The first parameter is the default value. It is returned if no condition applies.
select :: a -> [(Bool, a)] -> a select = foldr (uncurry if')
Easy lifting into monads (MonadReader in this case)
ifF :: (a -> Bool) -> (a -> b) -> (a -> b) -> (a -> b) ifF = liftM3 if'
2.3 Why add this function to Prelude?Actually people could define
where they need it,or import it from a
that must be provided in each project. Both solutions are tedious and contradict to modularization and software re-usage.The central question is, whether
that is so general that it should be in the Prelude, or not. I think it is, otherwise it wouldn't have get a special syntax.
2.4 If-Then-Else vs. guardsActually
2.5 Is If-Then-Else so important?Counting
in today's Haskell programsisn't a good measure for the importance a
- frequently guards are used instead of if-then-else
- there is no standard function, and this let people stick to work-arounds.
2.6 What is so bad about the if-then-else sugar?
Since syntactic sugar introduces its own syntactic rules, it is hard to predict how it interferes with other syntactic constructs. This special syntax for instance led to conflicts with do notation. A syntactic extension to solve this problem is proposed for Haskell'. It is not known what conflicts this extension might cause in future.
2.7 Why breaking lots of old and unmaintained code?Haskell without
makes Haskell more logical and consistent. There is no longer confusion to beginners like: "What is so special about if-then-else, that it needs a separate syntax? I though it could be simply replaced by a function. Maybe there is some subtlety that I'm not able to see right now." There is no longer confusion with the interference of
say compiler, text editor, analyzer and so on.
If we arrive at Haskell two some day, (http://haskell.org/hawiki/HaskellTwo) it will certainly be incompatible to former Haskell versions. This does not mean, that old code must be thrown away. There should be one tool, that converts Haskell 98 and Haskell' to Haskell-2. Having one tool for this purpose is better than blowing all language tools with legacy code.Syntactic replacements like
- Light proposal, compatible with Haskell 98: Add to the Prelude, maybe with a different name.if'
- Full proposal, incompatible with Haskell 98 and Haskell': Additionally remove syntaxif-then-else
2.9 See also
Haskell is not intended to be a minimalistic language, but to be one, that is easy to read.
It shows clearly which expression is returned on a fulfilled condition, and which one is returned for an unsatisfied condition. It is thus easier to read. The special syntax saves parentheses around its arguments. If properly indented, like
if a then b else c
if a then b else c