Why does GHC infer a monomorphic type here, even with MonomorphismRestriction disabled? The Next CEO of Stack OverflowResolving the type of `f = f (<*>) pure`NoMonomorphismRestriction helps preserve sharing?How can eta-reduction of a well typed function result in a type error?Can I write such polymorphic function? What language extensions do I need?GHC rewrite rule specialising a function for a type classType Inference in PatternsHow to type check recursive definitions using Algorithm W?What is the monomorphism restriction?Why are higher rank types so fragile in HaskellWhy can't GHC typecheck this function involving polymorphism and existential types?Problems With Type Inference on (^)
Is it possible to replace duplicates of a character with one character using tr
How to avoid supervisors with prejudiced views?
Help understanding this unsettling image of Titan, Epimetheus, and Saturn's rings?
How to write a definition with variants?
A Man With a Stainless Steel Endoskeleton (like The Terminator) Fighting Cloaked Aliens Only He Can See
How to get from Geneva Airport to Metabief, Doubs, France by public transport?
How is this set of matrices closed under multiplication?
What did we know about the Kessel run before the prequels?
Why, when going from special to general relativity, do we just replace partial derivatives with covariant derivatives?
How did people program for Consoles with multiple CPUs?
Would a completely good Muggle be able to use a wand?
Why the difference in type-inference over the as-pattern in two similar function definitions?
Example of a Mathematician/Physicist whose Other Publications during their PhD eclipsed their PhD Thesis
How to check if all elements of 1 list are in the *same quantity* and in any order, in the list2?
The past simple of "gaslight" – "gaslighted" or "gaslit"?
Why doesn't UK go for the same deal Japan has with EU to resolve Brexit?
Would a grinding machine be a simple and workable propulsion system for an interplanetary spacecraft?
Rotate a column
Bartok - Syncopation (1): Meaning of notes in between Grand Staff
Why didn't Khan get resurrected in the Genesis Explosion?
Legal workarounds for testamentary trust perceived as unfair
Can you be charged for obstruction for refusing to answer questions?
Can I use the load factor to estimate the lift?
Unclear about dynamic binding
Why does GHC infer a monomorphic type here, even with MonomorphismRestriction disabled?
The Next CEO of Stack OverflowResolving the type of `f = f (<*>) pure`NoMonomorphismRestriction helps preserve sharing?How can eta-reduction of a well typed function result in a type error?Can I write such polymorphic function? What language extensions do I need?GHC rewrite rule specialising a function for a type classType Inference in PatternsHow to type check recursive definitions using Algorithm W?What is the monomorphism restriction?Why are higher rank types so fragile in HaskellWhy can't GHC typecheck this function involving polymorphism and existential types?Problems With Type Inference on (^)
This was prompted by Resolving the type of `f = f (<*>) pure`, which discusses a more complicated example, but this one works too.
The following definition compiles without problem:
w :: Integral a => a
w = fromInteger w
...Of course it doesn't work runtime-wise, but that's beside the question. The point is that the definition of w
itself uses a specialised version of w :: Integer
. Clearly that is a suitable instantiation, and therefore typechecks.
However, if we remove the signature, then GHC infers not the above type, but only the concrete one:
w' = fromInteger w'
GHCi> :t w
w :: Integral a => a
GHCi> :t w'
w' :: Integer
Well, when I saw this, I was fairly sure this was the monomorphism restriction at work. It's well known that also e.g.
i = 3
GHCi> :t i
i :: Integer
although i :: Num p => p
would be perfectly possible. And indeed, i :: Num p => p
is inferred if -XNoMonomorphismRestriction
is active, i.e. if the monomorphism restriction is disabled.
However, in case of w'
only the type Integer
is inferred even when the monomorphism restriction is disabled.
To count out that this has something to do with defaulting:
fromFloat :: RealFrac a => Float -> a
q :: RealFrac a => a
q = fromFloat q
q' = fromFloat q'
GHCi> :t q
q :: RealFrac a => a
GHCi> :t q'
q' :: Float
Why is the polymorphic type not inferred?
haskell recursion type-inference parametric-polymorphism
add a comment |
This was prompted by Resolving the type of `f = f (<*>) pure`, which discusses a more complicated example, but this one works too.
The following definition compiles without problem:
w :: Integral a => a
w = fromInteger w
...Of course it doesn't work runtime-wise, but that's beside the question. The point is that the definition of w
itself uses a specialised version of w :: Integer
. Clearly that is a suitable instantiation, and therefore typechecks.
However, if we remove the signature, then GHC infers not the above type, but only the concrete one:
w' = fromInteger w'
GHCi> :t w
w :: Integral a => a
GHCi> :t w'
w' :: Integer
Well, when I saw this, I was fairly sure this was the monomorphism restriction at work. It's well known that also e.g.
i = 3
GHCi> :t i
i :: Integer
although i :: Num p => p
would be perfectly possible. And indeed, i :: Num p => p
is inferred if -XNoMonomorphismRestriction
is active, i.e. if the monomorphism restriction is disabled.
However, in case of w'
only the type Integer
is inferred even when the monomorphism restriction is disabled.
To count out that this has something to do with defaulting:
fromFloat :: RealFrac a => Float -> a
q :: RealFrac a => a
q = fromFloat q
q' = fromFloat q'
GHCi> :t q
q :: RealFrac a => a
GHCi> :t q'
q' :: Float
Why is the polymorphic type not inferred?
haskell recursion type-inference parametric-polymorphism
Doesn't the monomorphism restriction apply only to simple bindings anyways (andw' = fromInteger w'
, being recursive, is not simple)?
– Alec
2 days ago
@Alec possible, but still – why does something like the monomorphism restriction seem to kick in here?
– leftaroundabout
2 days ago
I'm probably being dense and missing something here, butfromInteger
has type(Num a) => Integer -> a
, and sincew'
is used as the input tofromInteger
, doesn't that meanInteger
is the only possible type for it? Indeed I'm rather surprised that the version with the polymorphic type signature compiles. (So as I said, probably missing something.)
– Robin Zigmond
2 days ago
@RobinZigmondInteger
is certainly the only possible monomorphic type forw'
, but asw
demonstrates a polymorphic type is perfectly fine as well. After all, a polymorphic type can be instantiated to a monomorphic one, provided it fulfills the constraints.
– leftaroundabout
2 days ago
add a comment |
This was prompted by Resolving the type of `f = f (<*>) pure`, which discusses a more complicated example, but this one works too.
The following definition compiles without problem:
w :: Integral a => a
w = fromInteger w
...Of course it doesn't work runtime-wise, but that's beside the question. The point is that the definition of w
itself uses a specialised version of w :: Integer
. Clearly that is a suitable instantiation, and therefore typechecks.
However, if we remove the signature, then GHC infers not the above type, but only the concrete one:
w' = fromInteger w'
GHCi> :t w
w :: Integral a => a
GHCi> :t w'
w' :: Integer
Well, when I saw this, I was fairly sure this was the monomorphism restriction at work. It's well known that also e.g.
i = 3
GHCi> :t i
i :: Integer
although i :: Num p => p
would be perfectly possible. And indeed, i :: Num p => p
is inferred if -XNoMonomorphismRestriction
is active, i.e. if the monomorphism restriction is disabled.
However, in case of w'
only the type Integer
is inferred even when the monomorphism restriction is disabled.
To count out that this has something to do with defaulting:
fromFloat :: RealFrac a => Float -> a
q :: RealFrac a => a
q = fromFloat q
q' = fromFloat q'
GHCi> :t q
q :: RealFrac a => a
GHCi> :t q'
q' :: Float
Why is the polymorphic type not inferred?
haskell recursion type-inference parametric-polymorphism
This was prompted by Resolving the type of `f = f (<*>) pure`, which discusses a more complicated example, but this one works too.
The following definition compiles without problem:
w :: Integral a => a
w = fromInteger w
...Of course it doesn't work runtime-wise, but that's beside the question. The point is that the definition of w
itself uses a specialised version of w :: Integer
. Clearly that is a suitable instantiation, and therefore typechecks.
However, if we remove the signature, then GHC infers not the above type, but only the concrete one:
w' = fromInteger w'
GHCi> :t w
w :: Integral a => a
GHCi> :t w'
w' :: Integer
Well, when I saw this, I was fairly sure this was the monomorphism restriction at work. It's well known that also e.g.
i = 3
GHCi> :t i
i :: Integer
although i :: Num p => p
would be perfectly possible. And indeed, i :: Num p => p
is inferred if -XNoMonomorphismRestriction
is active, i.e. if the monomorphism restriction is disabled.
However, in case of w'
only the type Integer
is inferred even when the monomorphism restriction is disabled.
To count out that this has something to do with defaulting:
fromFloat :: RealFrac a => Float -> a
q :: RealFrac a => a
q = fromFloat q
q' = fromFloat q'
GHCi> :t q
q :: RealFrac a => a
GHCi> :t q'
q' :: Float
Why is the polymorphic type not inferred?
haskell recursion type-inference parametric-polymorphism
haskell recursion type-inference parametric-polymorphism
edited 2 days ago
leftaroundabout
asked 2 days ago
leftaroundaboutleftaroundabout
80.2k3119237
80.2k3119237
Doesn't the monomorphism restriction apply only to simple bindings anyways (andw' = fromInteger w'
, being recursive, is not simple)?
– Alec
2 days ago
@Alec possible, but still – why does something like the monomorphism restriction seem to kick in here?
– leftaroundabout
2 days ago
I'm probably being dense and missing something here, butfromInteger
has type(Num a) => Integer -> a
, and sincew'
is used as the input tofromInteger
, doesn't that meanInteger
is the only possible type for it? Indeed I'm rather surprised that the version with the polymorphic type signature compiles. (So as I said, probably missing something.)
– Robin Zigmond
2 days ago
@RobinZigmondInteger
is certainly the only possible monomorphic type forw'
, but asw
demonstrates a polymorphic type is perfectly fine as well. After all, a polymorphic type can be instantiated to a monomorphic one, provided it fulfills the constraints.
– leftaroundabout
2 days ago
add a comment |
Doesn't the monomorphism restriction apply only to simple bindings anyways (andw' = fromInteger w'
, being recursive, is not simple)?
– Alec
2 days ago
@Alec possible, but still – why does something like the monomorphism restriction seem to kick in here?
– leftaroundabout
2 days ago
I'm probably being dense and missing something here, butfromInteger
has type(Num a) => Integer -> a
, and sincew'
is used as the input tofromInteger
, doesn't that meanInteger
is the only possible type for it? Indeed I'm rather surprised that the version with the polymorphic type signature compiles. (So as I said, probably missing something.)
– Robin Zigmond
2 days ago
@RobinZigmondInteger
is certainly the only possible monomorphic type forw'
, but asw
demonstrates a polymorphic type is perfectly fine as well. After all, a polymorphic type can be instantiated to a monomorphic one, provided it fulfills the constraints.
– leftaroundabout
2 days ago
Doesn't the monomorphism restriction apply only to simple bindings anyways (and
w' = fromInteger w'
, being recursive, is not simple)?– Alec
2 days ago
Doesn't the monomorphism restriction apply only to simple bindings anyways (and
w' = fromInteger w'
, being recursive, is not simple)?– Alec
2 days ago
@Alec possible, but still – why does something like the monomorphism restriction seem to kick in here?
– leftaroundabout
2 days ago
@Alec possible, but still – why does something like the monomorphism restriction seem to kick in here?
– leftaroundabout
2 days ago
I'm probably being dense and missing something here, but
fromInteger
has type (Num a) => Integer -> a
, and since w'
is used as the input to fromInteger
, doesn't that mean Integer
is the only possible type for it? Indeed I'm rather surprised that the version with the polymorphic type signature compiles. (So as I said, probably missing something.)– Robin Zigmond
2 days ago
I'm probably being dense and missing something here, but
fromInteger
has type (Num a) => Integer -> a
, and since w'
is used as the input to fromInteger
, doesn't that mean Integer
is the only possible type for it? Indeed I'm rather surprised that the version with the polymorphic type signature compiles. (So as I said, probably missing something.)– Robin Zigmond
2 days ago
@RobinZigmond
Integer
is certainly the only possible monomorphic type for w'
, but as w
demonstrates a polymorphic type is perfectly fine as well. After all, a polymorphic type can be instantiated to a monomorphic one, provided it fulfills the constraints.– leftaroundabout
2 days ago
@RobinZigmond
Integer
is certainly the only possible monomorphic type for w'
, but as w
demonstrates a polymorphic type is perfectly fine as well. After all, a polymorphic type can be instantiated to a monomorphic one, provided it fulfills the constraints.– leftaroundabout
2 days ago
add a comment |
1 Answer
1
active
oldest
votes
Polymorphic recursion (where a function calls itself at a different type than the one at which it was called) always requires a type signature. The full explanation is in Section 4.4.1 of the Haskell 2010 Report:
If a variable
f
is defined without providing a corresponding type signature declaration, then each use off
outside its own declaration group (see Section 4.5) is treated as having the corresponding inferred, or principal type. However, to ensure that type inference is still possible, the defining occurrence, and all uses off
within its declaration group must have the same monomorphic type (from which the principal type is obtained by generalization, as described in Section 4.5.2).
The same section later presents an example of polymorphic recursion supported by a type signature.
My understanding is that unaided type inference is generally undecidable in the presence of polymorphic recursion, so Haskell doesn't even try.
In this case, the type checker starts with
w :: a
where a
is a meta-variable. Since fromInteger
is called with w
as an argument within its own declaration (and therefore within its declaration group), the type checker unifies a
with Integer
. There are no variables left to generalize.
A slight modification of your program gives a different result for the same reason:
v = fromIntegral v
By your original reasoning, Haskell would infer v :: forall a. Num a => a
, defaulting the v
on the RHS to type Integer
:
v :: forall a. Num a => a
v = fromIntegral (v :: Integer)
But instead, it starts with v :: a
. Since v
is passed to fromIntegral
, it imposes Integral a
. Finally, it generalizes a
. In the end, the program turns out to be
v :: forall a. Integral a => a
v = fromIntegral (v :: a)
2
My bachelor was about a simple type inferencer for a subset of Haskell including typeclasses & polymorphic recursion. A very simple approach is to limit the depth of the polymorphic recursion up tok
depth. Most useful cases of polymorphic recursion can be inferred with a very low depth bound (like k=1 or k=2). Anyway Haskell type inference is already undecidable so that's not the only reason why it's not allowed. An other reason is probably performance, it surely makes type inference O(k·f(n)) instead of O(f(n)) since you may need to do all over again for k times.
– Bakuriu
2 days ago
2
@Bakuriu, I am pretty sure that Haskell 2010 without polymorphic recursion has full type inference--it's basically Hindley-Milner at that point, plus type classes and defaulting. Do you have a reference saying otherwise? As for some limited recursion depth: that sounds like a potentially useful extension, but it has a very different flavor from what the Haskell Report tends to do. I would find such a feature most useful for discovering the right type signatures for polymorphic recursive code.
– dfeuer
2 days ago
Yes, but I think all extensions to the type systems on top of Haskell2010 make type inference undecidable. Note that for example Type families are "artificially" limited to avoid undecidable instances by forbidding certain well-formed programs by default, so allowing a "k-recursive" polymorphic recursion would not be very different from that case, IMHO.
– Bakuriu
2 days ago
@Bakuriu, just for my own curiosity: how much ofData.Sequence
could be inferred with bounded polymorphic recursion depth? I'm not quite sure what you mean when you say "depth" in this context.
– dfeuer
2 days ago
1
The reason there is no inference for polymorphic recursion is exactly because it’s undecidable. This is a decision from long before ghc gained a lot of other undecidable features. You can trust me on this one, I was on the Haskell committee at that point. :)
– augustss
yesterday
add a comment |
Your Answer
StackExchange.ifUsing("editor", function ()
StackExchange.using("externalEditor", function ()
StackExchange.using("snippets", function ()
StackExchange.snippets.init();
);
);
, "code-snippets");
StackExchange.ready(function()
var channelOptions =
tags: "".split(" "),
id: "1"
;
initTagRenderer("".split(" "), "".split(" "), channelOptions);
StackExchange.using("externalEditor", function()
// Have to fire editor after snippets, if snippets enabled
if (StackExchange.settings.snippets.snippetsEnabled)
StackExchange.using("snippets", function()
createEditor();
);
else
createEditor();
);
function createEditor()
StackExchange.prepareEditor(
heartbeatType: 'answer',
autoActivateHeartbeat: false,
convertImagesToLinks: true,
noModals: true,
showLowRepImageUploadWarning: true,
reputationToPostImages: 10,
bindNavPrevention: true,
postfix: "",
imageUploader:
brandingHtml: "Powered by u003ca class="icon-imgur-white" href="https://imgur.com/"u003eu003c/au003e",
contentPolicyHtml: "User contributions licensed under u003ca href="https://creativecommons.org/licenses/by-sa/3.0/"u003ecc by-sa 3.0 with attribution requiredu003c/au003e u003ca href="https://stackoverflow.com/legal/content-policy"u003e(content policy)u003c/au003e",
allowUrls: true
,
onDemand: true,
discardSelector: ".discard-answer"
,immediatelyShowMarkdownHelp:true
);
);
Sign up or log in
StackExchange.ready(function ()
StackExchange.helpers.onClickDraftSave('#login-link');
);
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
StackExchange.ready(
function ()
StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fstackoverflow.com%2fquestions%2f55402733%2fwhy-does-ghc-infer-a-monomorphic-type-here-even-with-monomorphismrestriction-di%23new-answer', 'question_page');
);
Post as a guest
Required, but never shown
1 Answer
1
active
oldest
votes
1 Answer
1
active
oldest
votes
active
oldest
votes
active
oldest
votes
Polymorphic recursion (where a function calls itself at a different type than the one at which it was called) always requires a type signature. The full explanation is in Section 4.4.1 of the Haskell 2010 Report:
If a variable
f
is defined without providing a corresponding type signature declaration, then each use off
outside its own declaration group (see Section 4.5) is treated as having the corresponding inferred, or principal type. However, to ensure that type inference is still possible, the defining occurrence, and all uses off
within its declaration group must have the same monomorphic type (from which the principal type is obtained by generalization, as described in Section 4.5.2).
The same section later presents an example of polymorphic recursion supported by a type signature.
My understanding is that unaided type inference is generally undecidable in the presence of polymorphic recursion, so Haskell doesn't even try.
In this case, the type checker starts with
w :: a
where a
is a meta-variable. Since fromInteger
is called with w
as an argument within its own declaration (and therefore within its declaration group), the type checker unifies a
with Integer
. There are no variables left to generalize.
A slight modification of your program gives a different result for the same reason:
v = fromIntegral v
By your original reasoning, Haskell would infer v :: forall a. Num a => a
, defaulting the v
on the RHS to type Integer
:
v :: forall a. Num a => a
v = fromIntegral (v :: Integer)
But instead, it starts with v :: a
. Since v
is passed to fromIntegral
, it imposes Integral a
. Finally, it generalizes a
. In the end, the program turns out to be
v :: forall a. Integral a => a
v = fromIntegral (v :: a)
2
My bachelor was about a simple type inferencer for a subset of Haskell including typeclasses & polymorphic recursion. A very simple approach is to limit the depth of the polymorphic recursion up tok
depth. Most useful cases of polymorphic recursion can be inferred with a very low depth bound (like k=1 or k=2). Anyway Haskell type inference is already undecidable so that's not the only reason why it's not allowed. An other reason is probably performance, it surely makes type inference O(k·f(n)) instead of O(f(n)) since you may need to do all over again for k times.
– Bakuriu
2 days ago
2
@Bakuriu, I am pretty sure that Haskell 2010 without polymorphic recursion has full type inference--it's basically Hindley-Milner at that point, plus type classes and defaulting. Do you have a reference saying otherwise? As for some limited recursion depth: that sounds like a potentially useful extension, but it has a very different flavor from what the Haskell Report tends to do. I would find such a feature most useful for discovering the right type signatures for polymorphic recursive code.
– dfeuer
2 days ago
Yes, but I think all extensions to the type systems on top of Haskell2010 make type inference undecidable. Note that for example Type families are "artificially" limited to avoid undecidable instances by forbidding certain well-formed programs by default, so allowing a "k-recursive" polymorphic recursion would not be very different from that case, IMHO.
– Bakuriu
2 days ago
@Bakuriu, just for my own curiosity: how much ofData.Sequence
could be inferred with bounded polymorphic recursion depth? I'm not quite sure what you mean when you say "depth" in this context.
– dfeuer
2 days ago
1
The reason there is no inference for polymorphic recursion is exactly because it’s undecidable. This is a decision from long before ghc gained a lot of other undecidable features. You can trust me on this one, I was on the Haskell committee at that point. :)
– augustss
yesterday
add a comment |
Polymorphic recursion (where a function calls itself at a different type than the one at which it was called) always requires a type signature. The full explanation is in Section 4.4.1 of the Haskell 2010 Report:
If a variable
f
is defined without providing a corresponding type signature declaration, then each use off
outside its own declaration group (see Section 4.5) is treated as having the corresponding inferred, or principal type. However, to ensure that type inference is still possible, the defining occurrence, and all uses off
within its declaration group must have the same monomorphic type (from which the principal type is obtained by generalization, as described in Section 4.5.2).
The same section later presents an example of polymorphic recursion supported by a type signature.
My understanding is that unaided type inference is generally undecidable in the presence of polymorphic recursion, so Haskell doesn't even try.
In this case, the type checker starts with
w :: a
where a
is a meta-variable. Since fromInteger
is called with w
as an argument within its own declaration (and therefore within its declaration group), the type checker unifies a
with Integer
. There are no variables left to generalize.
A slight modification of your program gives a different result for the same reason:
v = fromIntegral v
By your original reasoning, Haskell would infer v :: forall a. Num a => a
, defaulting the v
on the RHS to type Integer
:
v :: forall a. Num a => a
v = fromIntegral (v :: Integer)
But instead, it starts with v :: a
. Since v
is passed to fromIntegral
, it imposes Integral a
. Finally, it generalizes a
. In the end, the program turns out to be
v :: forall a. Integral a => a
v = fromIntegral (v :: a)
2
My bachelor was about a simple type inferencer for a subset of Haskell including typeclasses & polymorphic recursion. A very simple approach is to limit the depth of the polymorphic recursion up tok
depth. Most useful cases of polymorphic recursion can be inferred with a very low depth bound (like k=1 or k=2). Anyway Haskell type inference is already undecidable so that's not the only reason why it's not allowed. An other reason is probably performance, it surely makes type inference O(k·f(n)) instead of O(f(n)) since you may need to do all over again for k times.
– Bakuriu
2 days ago
2
@Bakuriu, I am pretty sure that Haskell 2010 without polymorphic recursion has full type inference--it's basically Hindley-Milner at that point, plus type classes and defaulting. Do you have a reference saying otherwise? As for some limited recursion depth: that sounds like a potentially useful extension, but it has a very different flavor from what the Haskell Report tends to do. I would find such a feature most useful for discovering the right type signatures for polymorphic recursive code.
– dfeuer
2 days ago
Yes, but I think all extensions to the type systems on top of Haskell2010 make type inference undecidable. Note that for example Type families are "artificially" limited to avoid undecidable instances by forbidding certain well-formed programs by default, so allowing a "k-recursive" polymorphic recursion would not be very different from that case, IMHO.
– Bakuriu
2 days ago
@Bakuriu, just for my own curiosity: how much ofData.Sequence
could be inferred with bounded polymorphic recursion depth? I'm not quite sure what you mean when you say "depth" in this context.
– dfeuer
2 days ago
1
The reason there is no inference for polymorphic recursion is exactly because it’s undecidable. This is a decision from long before ghc gained a lot of other undecidable features. You can trust me on this one, I was on the Haskell committee at that point. :)
– augustss
yesterday
add a comment |
Polymorphic recursion (where a function calls itself at a different type than the one at which it was called) always requires a type signature. The full explanation is in Section 4.4.1 of the Haskell 2010 Report:
If a variable
f
is defined without providing a corresponding type signature declaration, then each use off
outside its own declaration group (see Section 4.5) is treated as having the corresponding inferred, or principal type. However, to ensure that type inference is still possible, the defining occurrence, and all uses off
within its declaration group must have the same monomorphic type (from which the principal type is obtained by generalization, as described in Section 4.5.2).
The same section later presents an example of polymorphic recursion supported by a type signature.
My understanding is that unaided type inference is generally undecidable in the presence of polymorphic recursion, so Haskell doesn't even try.
In this case, the type checker starts with
w :: a
where a
is a meta-variable. Since fromInteger
is called with w
as an argument within its own declaration (and therefore within its declaration group), the type checker unifies a
with Integer
. There are no variables left to generalize.
A slight modification of your program gives a different result for the same reason:
v = fromIntegral v
By your original reasoning, Haskell would infer v :: forall a. Num a => a
, defaulting the v
on the RHS to type Integer
:
v :: forall a. Num a => a
v = fromIntegral (v :: Integer)
But instead, it starts with v :: a
. Since v
is passed to fromIntegral
, it imposes Integral a
. Finally, it generalizes a
. In the end, the program turns out to be
v :: forall a. Integral a => a
v = fromIntegral (v :: a)
Polymorphic recursion (where a function calls itself at a different type than the one at which it was called) always requires a type signature. The full explanation is in Section 4.4.1 of the Haskell 2010 Report:
If a variable
f
is defined without providing a corresponding type signature declaration, then each use off
outside its own declaration group (see Section 4.5) is treated as having the corresponding inferred, or principal type. However, to ensure that type inference is still possible, the defining occurrence, and all uses off
within its declaration group must have the same monomorphic type (from which the principal type is obtained by generalization, as described in Section 4.5.2).
The same section later presents an example of polymorphic recursion supported by a type signature.
My understanding is that unaided type inference is generally undecidable in the presence of polymorphic recursion, so Haskell doesn't even try.
In this case, the type checker starts with
w :: a
where a
is a meta-variable. Since fromInteger
is called with w
as an argument within its own declaration (and therefore within its declaration group), the type checker unifies a
with Integer
. There are no variables left to generalize.
A slight modification of your program gives a different result for the same reason:
v = fromIntegral v
By your original reasoning, Haskell would infer v :: forall a. Num a => a
, defaulting the v
on the RHS to type Integer
:
v :: forall a. Num a => a
v = fromIntegral (v :: Integer)
But instead, it starts with v :: a
. Since v
is passed to fromIntegral
, it imposes Integral a
. Finally, it generalizes a
. In the end, the program turns out to be
v :: forall a. Integral a => a
v = fromIntegral (v :: a)
edited 2 days ago
answered 2 days ago
dfeuerdfeuer
33.6k349133
33.6k349133
2
My bachelor was about a simple type inferencer for a subset of Haskell including typeclasses & polymorphic recursion. A very simple approach is to limit the depth of the polymorphic recursion up tok
depth. Most useful cases of polymorphic recursion can be inferred with a very low depth bound (like k=1 or k=2). Anyway Haskell type inference is already undecidable so that's not the only reason why it's not allowed. An other reason is probably performance, it surely makes type inference O(k·f(n)) instead of O(f(n)) since you may need to do all over again for k times.
– Bakuriu
2 days ago
2
@Bakuriu, I am pretty sure that Haskell 2010 without polymorphic recursion has full type inference--it's basically Hindley-Milner at that point, plus type classes and defaulting. Do you have a reference saying otherwise? As for some limited recursion depth: that sounds like a potentially useful extension, but it has a very different flavor from what the Haskell Report tends to do. I would find such a feature most useful for discovering the right type signatures for polymorphic recursive code.
– dfeuer
2 days ago
Yes, but I think all extensions to the type systems on top of Haskell2010 make type inference undecidable. Note that for example Type families are "artificially" limited to avoid undecidable instances by forbidding certain well-formed programs by default, so allowing a "k-recursive" polymorphic recursion would not be very different from that case, IMHO.
– Bakuriu
2 days ago
@Bakuriu, just for my own curiosity: how much ofData.Sequence
could be inferred with bounded polymorphic recursion depth? I'm not quite sure what you mean when you say "depth" in this context.
– dfeuer
2 days ago
1
The reason there is no inference for polymorphic recursion is exactly because it’s undecidable. This is a decision from long before ghc gained a lot of other undecidable features. You can trust me on this one, I was on the Haskell committee at that point. :)
– augustss
yesterday
add a comment |
2
My bachelor was about a simple type inferencer for a subset of Haskell including typeclasses & polymorphic recursion. A very simple approach is to limit the depth of the polymorphic recursion up tok
depth. Most useful cases of polymorphic recursion can be inferred with a very low depth bound (like k=1 or k=2). Anyway Haskell type inference is already undecidable so that's not the only reason why it's not allowed. An other reason is probably performance, it surely makes type inference O(k·f(n)) instead of O(f(n)) since you may need to do all over again for k times.
– Bakuriu
2 days ago
2
@Bakuriu, I am pretty sure that Haskell 2010 without polymorphic recursion has full type inference--it's basically Hindley-Milner at that point, plus type classes and defaulting. Do you have a reference saying otherwise? As for some limited recursion depth: that sounds like a potentially useful extension, but it has a very different flavor from what the Haskell Report tends to do. I would find such a feature most useful for discovering the right type signatures for polymorphic recursive code.
– dfeuer
2 days ago
Yes, but I think all extensions to the type systems on top of Haskell2010 make type inference undecidable. Note that for example Type families are "artificially" limited to avoid undecidable instances by forbidding certain well-formed programs by default, so allowing a "k-recursive" polymorphic recursion would not be very different from that case, IMHO.
– Bakuriu
2 days ago
@Bakuriu, just for my own curiosity: how much ofData.Sequence
could be inferred with bounded polymorphic recursion depth? I'm not quite sure what you mean when you say "depth" in this context.
– dfeuer
2 days ago
1
The reason there is no inference for polymorphic recursion is exactly because it’s undecidable. This is a decision from long before ghc gained a lot of other undecidable features. You can trust me on this one, I was on the Haskell committee at that point. :)
– augustss
yesterday
2
2
My bachelor was about a simple type inferencer for a subset of Haskell including typeclasses & polymorphic recursion. A very simple approach is to limit the depth of the polymorphic recursion up to
k
depth. Most useful cases of polymorphic recursion can be inferred with a very low depth bound (like k=1 or k=2). Anyway Haskell type inference is already undecidable so that's not the only reason why it's not allowed. An other reason is probably performance, it surely makes type inference O(k·f(n)) instead of O(f(n)) since you may need to do all over again for k times.– Bakuriu
2 days ago
My bachelor was about a simple type inferencer for a subset of Haskell including typeclasses & polymorphic recursion. A very simple approach is to limit the depth of the polymorphic recursion up to
k
depth. Most useful cases of polymorphic recursion can be inferred with a very low depth bound (like k=1 or k=2). Anyway Haskell type inference is already undecidable so that's not the only reason why it's not allowed. An other reason is probably performance, it surely makes type inference O(k·f(n)) instead of O(f(n)) since you may need to do all over again for k times.– Bakuriu
2 days ago
2
2
@Bakuriu, I am pretty sure that Haskell 2010 without polymorphic recursion has full type inference--it's basically Hindley-Milner at that point, plus type classes and defaulting. Do you have a reference saying otherwise? As for some limited recursion depth: that sounds like a potentially useful extension, but it has a very different flavor from what the Haskell Report tends to do. I would find such a feature most useful for discovering the right type signatures for polymorphic recursive code.
– dfeuer
2 days ago
@Bakuriu, I am pretty sure that Haskell 2010 without polymorphic recursion has full type inference--it's basically Hindley-Milner at that point, plus type classes and defaulting. Do you have a reference saying otherwise? As for some limited recursion depth: that sounds like a potentially useful extension, but it has a very different flavor from what the Haskell Report tends to do. I would find such a feature most useful for discovering the right type signatures for polymorphic recursive code.
– dfeuer
2 days ago
Yes, but I think all extensions to the type systems on top of Haskell2010 make type inference undecidable. Note that for example Type families are "artificially" limited to avoid undecidable instances by forbidding certain well-formed programs by default, so allowing a "k-recursive" polymorphic recursion would not be very different from that case, IMHO.
– Bakuriu
2 days ago
Yes, but I think all extensions to the type systems on top of Haskell2010 make type inference undecidable. Note that for example Type families are "artificially" limited to avoid undecidable instances by forbidding certain well-formed programs by default, so allowing a "k-recursive" polymorphic recursion would not be very different from that case, IMHO.
– Bakuriu
2 days ago
@Bakuriu, just for my own curiosity: how much of
Data.Sequence
could be inferred with bounded polymorphic recursion depth? I'm not quite sure what you mean when you say "depth" in this context.– dfeuer
2 days ago
@Bakuriu, just for my own curiosity: how much of
Data.Sequence
could be inferred with bounded polymorphic recursion depth? I'm not quite sure what you mean when you say "depth" in this context.– dfeuer
2 days ago
1
1
The reason there is no inference for polymorphic recursion is exactly because it’s undecidable. This is a decision from long before ghc gained a lot of other undecidable features. You can trust me on this one, I was on the Haskell committee at that point. :)
– augustss
yesterday
The reason there is no inference for polymorphic recursion is exactly because it’s undecidable. This is a decision from long before ghc gained a lot of other undecidable features. You can trust me on this one, I was on the Haskell committee at that point. :)
– augustss
yesterday
add a comment |
Thanks for contributing an answer to Stack Overflow!
- Please be sure to answer the question. Provide details and share your research!
But avoid …
- Asking for help, clarification, or responding to other answers.
- Making statements based on opinion; back them up with references or personal experience.
To learn more, see our tips on writing great answers.
Sign up or log in
StackExchange.ready(function ()
StackExchange.helpers.onClickDraftSave('#login-link');
);
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
StackExchange.ready(
function ()
StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fstackoverflow.com%2fquestions%2f55402733%2fwhy-does-ghc-infer-a-monomorphic-type-here-even-with-monomorphismrestriction-di%23new-answer', 'question_page');
);
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function ()
StackExchange.helpers.onClickDraftSave('#login-link');
);
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function ()
StackExchange.helpers.onClickDraftSave('#login-link');
);
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function ()
StackExchange.helpers.onClickDraftSave('#login-link');
);
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Doesn't the monomorphism restriction apply only to simple bindings anyways (and
w' = fromInteger w'
, being recursive, is not simple)?– Alec
2 days ago
@Alec possible, but still – why does something like the monomorphism restriction seem to kick in here?
– leftaroundabout
2 days ago
I'm probably being dense and missing something here, but
fromInteger
has type(Num a) => Integer -> a
, and sincew'
is used as the input tofromInteger
, doesn't that meanInteger
is the only possible type for it? Indeed I'm rather surprised that the version with the polymorphic type signature compiles. (So as I said, probably missing something.)– Robin Zigmond
2 days ago
@RobinZigmond
Integer
is certainly the only possible monomorphic type forw'
, but asw
demonstrates a polymorphic type is perfectly fine as well. After all, a polymorphic type can be instantiated to a monomorphic one, provided it fulfills the constraints.– leftaroundabout
2 days ago