-
Notifications
You must be signed in to change notification settings - Fork 4
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
[DRAFT] RFC: Associated type defaults and default groups #11
Conversation
text/0000-assoc-default-groups.md
Outdated
|
||
As Rust traits are a form of type classes, | ||
we naturally look for prior art from were they first were introduced. | ||
That language, being Haskell, permits a user to specify associated type defaults. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
While traits are type classes, it is also comparable to OOP interfaces; specialization in particular is entirely inspired from C++.
So I'd suggest adding the following comparison to prior arts as well:
- C++ (all diagnostic is done post-monomorphization so probably not an interesting example, but still...)
- Swift (Rust trait ≈ Swift protocols. Protocols can be inherited and can have associated types. But protocol constants and methods cannot be defaulted.)
- Perhaps also check what Scala is doing 🙃
|
||
fn arbitrary() -> Self::Strategy { | ||
Self::arbitrary_with(Default::default()) | ||
} |
This comment was marked as off-topic.
This comment was marked as off-topic.
Sorry, something went wrong.
This comment was marked as off-topic.
This comment was marked as off-topic.
Sorry, something went wrong.
text/0000-assoc-default-groups.md
Outdated
|
||
[proptest]: https://altsysrq.github.io/rustdoc/proptest/latest/proptest/arbitrary/trait.Arbitrary.html | ||
|
||
For example, we may provide an API (due to [proptest]): |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It might read better to write this line as:
For example, we could change [proptest]'s API to be:
text/0000-assoc-default-groups.md
Outdated
just like before, that `x: u8`. The reason why is much the same as | ||
we have previously discussed in the [background]. | ||
|
||
Once place where this proposal does diverge from what is currently implemented |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Once -> One; does diverge -> diverges
text/0000-assoc-default-groups.md
Outdated
what do I do then?"*. Don't worry; We've got you covered. | ||
|
||
To be able to assume that `Self::Bar` is truly `u8` in snippets (2) and (5), | ||
you may henceforth use `default { .. }` to group items into atomic units of |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Might want to explicitly state associated items
here.
text/0000-assoc-default-groups.md
Outdated
} | ||
} | ||
|
||
struct Alan; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
At RustConf, misdreavious's slides included descriptions of what was being referred to. It was a great thing to include. I feel like we should do the same for RFCs, and link to these peoples' Wikipedia pages.
text/0000-assoc-default-groups.md
Outdated
} | ||
} | ||
|
||
struct Alan; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Can we change two of these to be women who've contributed to computer science, to maintain a gender balance?
text/0000-assoc-default-groups.md
Outdated
|
||
In the current implementation, (6) is rejected because the compiler will not | ||
let you assume that `x` is of type `usize`. But in this proposal, you would be | ||
allowed to assume this. To permit this is not a problem because `Foo for ()` |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
()
=> Vec<T>
text/0000-assoc-default-groups.md
Outdated
|
||
Idris has a concept it calls [`interface`s][idris_interface]. | ||
These resemble type classes in Haskell, and by extension traits in Rust. | ||
However, unlike Haskell and Rust, these `interface`s are incoherent and will |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Incoherent is a strongly negative word. Would be good to rephrase this to be less negative?
text/0000-assoc-default-groups.md
Outdated
```rust | ||
trait ComputerScientist { | ||
default { | ||
type Bar = u8; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
"Bar, QUUX, wibble" keep confusing me. Could these be changed to terms in the domain of being a Computer Scientist?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I like the base RFC, mostly editorial stuff, plus some corrections on the C++.
protocol Foo { | ||
associatedtype Bar = Int | ||
|
||
func append() -> Bar { return 0 } |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The syntax highlighter seems to dislike this, maybe split it?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The syntax highlighter dislikes it because it's literally not allowed.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ohhh, okay
text/0000-assoc-default-groups.md
Outdated
``` | ||
|
||
This shows that templates in C++ are fundamentally different from the type of | ||
parametric polymorphism (generics) that Rust employs. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I would cut out this part, from the "do note" on, and rewrite the above paragraph.
text/0000-assoc-default-groups.md
Outdated
One thing to note here is that C++ allows us to assume in both the unspecialized | ||
variant of `foo` as well as the specialized version that `bar` is the underlying | ||
type we said it was in the `typedef`. This is unlike the default in this RFC | ||
but the same as when we use `default { .. }`. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'd rewrite this comment quite heavily, to something along the following lines:
You will note that C++ allows us to assume in both the base template class, as well as the specialization, that
bar
is equal to the underlying type. This is because one cannot specialize any part of a class without specializing the whole of it. It's equivalent to one atomicdefault { ... }
block.
text/0000-assoc-default-groups.md
Outdated
}; | ||
|
||
template<typename T> struct foo<wrap<T>> { // Partial specialization. | ||
typedef std::string bar; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
You should use using
here instead - using bar = std::string;
. It's just more readable.
text/0000-assoc-default-groups.md
Outdated
template<typename T> struct wrap {}; | ||
|
||
template<typename T> struct foo { // Unspecialized. | ||
typedef int bar; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
See above comment on using
.
text/0000-assoc-default-groups.md
Outdated
} | ||
``` | ||
|
||
However, please note that for use of `default { .. }` inside implementations, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is worth having at the top of the section, not the bottom.
text/0000-assoc-default-groups.md
Outdated
use by specialization and for `default impl`. | ||
Therefore, the syntax is only partially new. | ||
|
||
2. secondarily, if you have implementations where it is commonly needed |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
"secondarily" is already said by the number "2".
I have an alternative in mind. So, a caveat here. I have not thought about how this interacts with specialization. So everything here talks from the perspective of writing defaults in traits. The short of it is that in this alternative, default blocks let you specialize default implementations when the implementing type's type aliases match the specified type aliases in the default block. And furthermore, the default block does not allow declaration of new associated items. It can only provide new defaults that can make use of the specified type. For an example of this alternative, we'll use this trait and this impl: trait Trait {
type Type;
const Const: Type;
}
impl Trait for UStruct {
type Type = u8;
const Const: Type = 0u8;
} Without using a default block, we cannot give a default to trait Trait {
type Type;
const Const: Type;
default {
type Type = u8;
const Const: u8 = 0u8;
}
}
impl Trait for UStruct {
type Type = u8;
// No need to specify `Const` as it uses the default given.
} But then, we decide that if the implementer chooses trait Trait {
type Type;
const Const: Type;
default {
type Type = u8;
const Const: u8 = 0u8;
}
default {
type Type = i8;
const Const: i8 = 1i8;
}
}
impl Trait for IStruct {
type Type = i8;
// `Const` defaults to 1i8
} To pick out which default applies, we look at what the trait Trait {
type Type = u8;
const Const: Type;
default {
type Type = u8;
const Const: u8 = 0u8;
}
default {
type Type = i8;
const Const: i8 = 1i8;
}
}
impl Trait for UStruct {
// type defaults to `u8`.
// `Const` defaults to 0u8 because `Type` is `u8`.
} And this all works great in simple cases. It gets complex though, if there's more than one type. You could have... trait Trait {
type TypeA;
type TypeB;
const Const: u8;
default {
type TypeA = ();
const Const: u8= 0u8;
}
default {
type TypeB = ();
const Const: u8= 1u8;
}
} ...and then you're not sure which default applies. That can be fixed by requiring defaulted items use all type aliases in the default block. Similarly, you can have specialization issues if you allow defaults where the types are generic. |
In RFC rust-lang#2500 I've got this trait: trait Needle<H: Haystack>: Sized {
type Searcher: Searcher<H::Target>;
type Consumer: Consumer<H::Target>;
fn into_searcher(self) -> Self::Searcher;
fn into_consumer(self) -> Self::Consumer;
} usually the associated types Consumer and Searcher are equivalent, so I'd like to default it: trait Needle<H: Haystack>: Sized {
type Searcher: Searcher<H::Target>;
fn into_searcher(self) -> Self::Searcher;
default {
type Consumer: Consumer<H::Target> = Self::Searcher;
fn into_consumer(self) -> Self::Consumer { self.into_searcher() }
}
} However, a Searcher may not implement |
I haven't looked at others' comments, but this seems like a nice, straightforward approach to defaults to me. type Foo = impl Bar; is the opaque type that implements This could be left unaddressed unless the trait alias |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Nice work. Overall, definitely in favour of this RFC. I must say, I didn't fully appreciate the motivation until I read the Guide section (not so much the Motivation section, hah). I was weighing up the pros and cons of the behaviour of associated type defaults in provided methods in this RFC vs. existing behaviour, and came to essentially the same conclusions as you. The granularity of this approach offers a big advantage, and the way methods interact with default associated types just seems more intuitive this way.
``` | ||
|
||
By allowing defaults, we can transition to this more flexible API without | ||
breaking any consumers by simply saying: |
This comment was marked as resolved.
This comment was marked as resolved.
Sorry, something went wrong.
|
||
[specialization]: /~https://github.com/rust-lang/rfcs/pull/1210 | ||
|
||
With respect to [specialization], the behaviour is the same. |
This comment was marked as resolved.
This comment was marked as resolved.
Sorry, something went wrong.
text/0000-assoc-default-groups.md
Outdated
is with respect to the following example (6): | ||
|
||
```rust | ||
#![feature(associated_type_defaults)] |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Slightly confusing feature name, given we already have "associated type defaults"? Maybe "associated_type_defaults_v2" or such?
text/0000-assoc-default-groups.md
Outdated
|
||
the underlying type of `Assoc` stays the same for all implementations which | ||
do not change the default of `Assoc`. The same applies to specializations. | ||
With respect to type visibility, it is the same as that of `existential type`. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
"Type opacity" (rather than "visibility"), you mean? I think that's clearer.
text/0000-assoc-default-groups.md
Outdated
|
||
There applies no restriction on the nesting of groups. | ||
This means that you may nest them arbitrarily. | ||
When nesting does occur, the atomicity applies as if the nesting was flattened. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
s/was flattened/were flattened/.
Wouldn't it be better just to disallow nesting? I thought of this before I read this section, and came to the conclusion that such variations just add needless complexity and potentially confusion.
|
||
Note: Everything in this section assumes actual support for [specialization]. | ||
|
||
Now, you might be thinking: - *"Well, what if I __do__ need to assume that |
This comment was marked as resolved.
This comment was marked as resolved.
Sorry, something went wrong.
Thanks all for chipping in! |
This is a draft version of an RFC for you to review, before a formal proposal is made for consideration.
Summary
Resolve the design of associated type defaults, first introduced in RFC 192, such that provided methods and other items may not assume type defaults. This applies equally to
default
with respect to specialization.Introduce the concept of
default { .. }
groups in traits and their implementations which may be used to introduce atomic units of specialization (if anything in the group is specialized, everything must be).