-
Notifications
You must be signed in to change notification settings - Fork 13.2k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Confusing type-deduction of integer literals in generic contexts #39255
Comments
I prefer explicit over implicit, but the current situation is already doing a lot implicitly. As a rule of thumb, if you are going to do something implicitly, you better make sure that it always works as the user expects. In this situation, this is sadly not the case. Since we cannot make this explicit without breaking backwards compatibility, maybe doing a bit more work implicitly to improve the ergonomics is worth it. The issue is with rule 2, which defaults to i32 when the type is under-constrained (that is, there are multiple alternatives). My example breaks because it provides an under-constrained context that does not accept i32. A backward compatible fix is to complicate rule 2 as follows:
I think it is backwards compatible because it only admits new programs that did not type-check before. It solves the situation since now both cases, This fix relies on "try to type-check, and if that fails, then retry". I am not familiar enough with rust type system and the compiler internals to know how feasible this is, but I cannot think of an easier way to fix this preserving backwards compatibility. |
Triage: no change |
Retagging as language feature request, but it seems likely that this would need an RFC |
I don't think the suggested change is very viable -- Niko has commented that extending fallback is in general quite hard, and this sounds like a very hard extension. I would definitely expect an RFC or similar to this, so closing. |
The way in which the types of integer literals are deduced in generic contexts is confusing. For example, the following:
works for
foo
but fails forbar
. Weirder is that uncommenting theimpl UInt for u16 {}
makes it work.IMO both cases should either work or fail (I would prefer work).
The error messages is:
I interpret from this that rustc is only trying with
i32
(for whatever reason) even though the code does not contain any explicit mention ofi32
anywhere. Still, removing one of the impls ofUInt
makes the program type-check, so rustc is able to deduce unsigned integer types in some situations.EDIT: After going through the book without any luck, I found the following in the Rust language reference:
I guess that removing the
impl UInt for u16 {}
switches from rule 2 to rule 1, explaining the behavior.The text was updated successfully, but these errors were encountered: