* Ian Lance Taylor <i...@golang.org> [181016 17:59]: > The contract system in the generics design draft does not have this > problem. It may have other problems, but it doesn't have this one.
I haven't finished reading the draft (I got to the beginning of the Contract Details section when it was first mentioned on this list, but then got sidetracked) nor have I read any of the counter proposals, which is why I have not responded before now. I very much like the idea of contracts, but I do not like the specific design. Programming by example is a cool idea, but it is simply a bad practice when you want to clearly and unambiguously express the intended behavior. While the draft design allows the compiler to unambiguously determine the correct behavior, as has been pointed out many times in the various discussions on this list, the specific code which the programmer must use to convey his/her intent to the compiler is extremely ambiguous to the programmer, and the meaning conveyed by a previously written contract to a different programmer reading the code is at least equally ambiguous. The example in the draft document concerning value method vs. pointer method clearly demonstrates this. This is contrary to the stated goals of the Go language with respect to simplicity, scalability and maintainability. On the other hand, if you replace the body of the contract with a simple declarative syntax, you regain these core attributes of the Go language. contract Readable(T) { T: Implements(io.Reader) } contract ComparableInt(U) { U: EqualMethod(IsSame), ConvertableTo(int), ConvertableFrom(int) } contract SortableRecord(V, W) { V: IsSliceOf(W), Implements(sort.Interface) W: Implements(Record) } Each constraint that a programmer might want to use in a contract must be specified by the language as a contract keyword. These keywords are only special inside a contract body, so they do not interact in any way with other Go code (e.g. there can be a method Implements on a type). The contract syntax is such that it is unambiguous what tokens are contract keywords and what are other language identifiers, such as interfaces and types; e.g. in Implements(Implements) the first Implements is a contract keyword and the second is the name of an interface in the current package. Some contract keywords may not have any arguments. Comparable might mean that the type is comparable using the primitive == operator. contract Comparable(X) { X: Comparable } Whether to require, allow, or forbid the empty parentheses is a detail for later discussion. I.e. the constraint might be X: Comparable(). There would be no ambiguity between the above contract name Comparable and the contract keyword Comparable, since the contract keyword has no relevance outside the contract body. A contract keyword might have more than one argument, such as M: IsMapOf(K, V). In the above examples, the contract keyword Implements declares that type T must implement interface io.Reader. The keyword EqualMethod declares that type U must have a method IsSame that has an argument and result signature appropriate for overloading the == operator. I see no technical or readability reason to require, or even allow, the extra names on contract arguments, such as in the draft example contract convertible(_ To, f From). It is just as expressive to write contract convertible(To, From). The form in the draft was necessary to allow writing Go expressions that used both values and types to describe what was permissible. My declarative syntax does not need this. It might be convenient to have a predefined contract for each contract keyword that can be used without explicitly declaring the contract. The predefined contract would be shadowed by an explicit contract with the same name. The predefined contract will have an additional argument preceding the arguments required by the keyword. E.g. func Massage(type T ConvertableTo(T, int))(item T) error {...} would use the predefined ConvertableTo contract that is equivalent to a contract with one constraint, T: ConvertableTo(int). To switch gears to operator overloading, I would like to start by saying that one of the fundamental beauties of Go is that looking at a small piece of code out of context, a programmer can still glean a substantial amount of information about what the code does, _especially_ about when external code is called. I would very much dislike for the language to be changed such that looking at «a == b» would leave me wondering if some method of a or b would be called to evaluate that expression. I had thought of «a ==() b» as a possible syntax for "if the type of a has a method that overloads ==, use it, otherwise if the primitive == operator can be applied, use it, otherwise fail to compile." Someone else (was it Michael Jones?) suggested «a (==) b». I kind of like my syntax better because it has a "function call" look to it, but either works for me. The point is that possibly overloaded operators should be explicitly called out by the programmer writing the code. There should be no reason to use such syntax outside of functions that have type parameters. Whether that is allowed by the language (and just uses the primitive) or produces a compiler error is a detail to be considered. Now to get to EqualMethod above. This contract keyword specifies that the type has a method with the given name, and that the method takes one argument of the same type and returns a bool. Each operator that the language designers wish to allow to be overloaded will have a contract keyword that associates the operator with the method specified in the contract declaration. The method must have a signature given by the language specification for that particular operator. The type may forego implementing the method iff the compiler already knows how to apply the operator to the type. So, for «type MyInt int» the EqualMethod(IsSame) would be satisfied without any IsSame method, but if the type has the IsSame method, it must have the correct signature or the compiler will complain, and the method will be used in preference to the intrinsic == operator when the overloaded ==() is used in the code. Using the non-overloaded == will always ignore the IsSame method, producing a compiler error if the type is not comparable. A similar keyword will be defined for each operator that the language designers wish to allow to be overloaded. Rules such as in «a ==() b» which operand is used as the method receiver and which is used as the method argument are specified in the contract keyword definition. If it makes any sense (and I am not sure it does) to worry about commutativity or associativity, the behavior will be specified in the contract keyword definition. The specific list of contract keywords is left for discussion. This declarative syntax is much easier to write, read, and understand than the "declaration by example" used in the draft. It allows for a much larger variety of constraints than the "generics by interface" proposals that I have seen described on this list, and it allows for flexible operator overloading with only a trivial addition to the language syntax that does not invalidate the Go 1 Compatibility Promise while ensuring that non-overloaded operators will never be applied as overloaded operators (no hidden function calls). The biggest (only?) downside is that it requires a predetermined, but expandable, list of constraints. This is mitigated by the fact that adding a new constraint in a future version cannot break existing code. The fact that each constraint requires a keyword is, to me, a non-issue, since the contract keyword namespace is completely separate from all other identifier namespaces. ...Marvin -- You received this message because you are subscribed to the Google Groups "golang-nuts" group. To unsubscribe from this group and stop receiving emails from it, send an email to golang-nuts+unsubscr...@googlegroups.com. For more options, visit https://groups.google.com/d/optout.