[Pitch] Light-weight same-type constraint syntax

I'm very against this. As @regexident says, this syntax is important for a major missing Swift feature, generic protocols.

This feature is used extensively in Rust, where it forms the backbone of generic conversion and operator overloading, and is one of the things I miss most when moving between the languages. I understand that the implementation in Swift is difficult and the idea has been passed over before for various reasons, but I remain convinced that this is an eventual necessity for Swift.

the obvious interpretation of this syntax is as a generic protocol, so this usage of this syntax will inherently be misleading. You can't actually use it how it looks:

extension Data: Collection<UInt8> { ... }
extension Data: Collection<UInt32> { ... }

is still an error due to overlapping conformances.

Likewise, you can't make any number of overlapping conformances:

protocol RawBytes { ... }
extension Data: Collection<T> where T: RawBytes { ... }

But also, it's lying about fundamental truths of Swift syntax. For better or worse, <T> means T is an input type, under the control of the programmer at that moment. With this proposal, <T> has an alternate meaning as a constraint of some invisible associated type to be equal to T.

I agree again with @regexident that an appropriate syntax for this is <.AssociatedType == T>, which

  • more or less matches Rust's syntax for the same purpose
  • is explicit about exactly which associated types are being constrained
  • doesn't require adding the concept of a "primary" associated type to the language
  • is flexible with regard to adding many constraints of many associated types (just use commas to separate the constraints)
  • is orthogonal to existing syntax rather than confusingly overlapping
  • allows us to keep the possibility of generic protocols open for the future (we're gonna want 'em)
17 Likes