[Haskell-cafe] A voyage of undiscovery
Andrew Coppin
andrewcoppin at btinternet.com
Thu Jul 16 14:34:18 EDT 2009
I've been working hard this week, and I'm stumbled upon something which
is probably of absolutely no surprise to anybody but me.
Consider the following expression:
(foo True, foo 'x')
Is this expression well-typed?
Astonishingly, the answer depends on where "foo" is defined. If "foo" is
a local variable, then the above expression is guaranteed to be
ill-typed. However, if we have (for example)
foo :: x -> x
as a top-level function, then the above expression becomes well-typed.
I had never ever noticed this fact before. I'm still trying to bend my
mind around exactly why it happens. As best as I can tell, top-level
functions (and value constructors, for that matter) seem to get a "new"
set of type variables each time they're used, but local variables each
get a single type variable, so every mention of a local variable must be
of the exact same type.
Could this be what GHC's weird "forall" syntax is about? Does "forall
x." mean that "x" gets replaced with a unique type variable each time?
It's an interesting hypothesis...
Anyway, not that anybody is likely to care, but I've just spent an
entire week writing a program which can type-check simple Haskell
expressions. As in, you type in an expression and give types to any free
variables it involves (including value constructor functions), and it
tells you the type of the expression and all its subexpressions. (Or
tells you that it's ill-typed.) It turns out that this is radically less
trivial than you'd imagine. (The ramblings above being just one of the
issues I blundered into. Others include the subtleties of writing an
expression parser, building a pretty-printer with bracketing that works
correctly, and the fact that expression processing malfunctions horribly
if you don't make all the variable names unique first...) Other issues
were mostly related to the difficulty of constantly refactoring code
because you're not quite sure what you're trying to do or how you're
trying to do it. (And obscure type checker warnings about GADTs...)
Well there we are. I don't suppose anybody will be overly impressed, but
I'm glad to have finally got it to work. Now, if it could parse more
than 10% of Haskell's syntax sugar, it might even be useful for something...
More information about the Haskell-Cafe
mailing list