> > I get that it avoids introducing those properties directly into the > > language but it also locks the language into those properties. You can > > never change https://github.com/golang/go/issues/19113 after > > introducing contracts because there could be a contract somewhere that > > uses 1 << u instead of contracts.Unsigned. > > This is an interesting point but I'm not yet convinced. The effect of > adopting issue 19113 would be that the contract would accept types > that it previously did not accept. But any polymorphic function using > that contract would still compile. And any call to that polymorphic > function would still compile and work correctly. So what you are > arguing is that we could not make a change that would cause a contract > to accept additional types. I've been questioning why it is important > for a contract to reject types, and I'm still questioning that.
That's true, it would be somewhat analogous to removing a method from an interface. It still accepts all the types it previously accepted and accepts more. The analogy breaks down because code calling the removed method would break if it attempted to call the now missing method. Generic code satisfying the contract that relied on the property that m - n > 0 for all m, n would continue to run in the absence of this now missing property when instantiated with a signed number. Though it may not do so correctly. Unlike the missing method, the missing property is harder to detect. It's easy to detect 1 << u in contracts but unless you know that it was written previous to the version of Go that allowed signed shifts you can't update it to contracts.Unsigned(u). If it was written later it could just mean "any integral number and I'm using it as the amount to shift by". To avoid the polysemy, you have to use the contracts package from the start. You avoid adding the notion of unsigned-ness into the language directly but by creating a mechanism that, for the built in types, can effectively only be used by the standard library. Contracts would definitely disallow a change like https://github.com/golang/go/issues/24529 That's a backwards-incompatible change now but it's rarely used and easy to detect and to automatically fix. After contracts it would likely be as rarely used but much harder to detect and automatically fix. > I guess that by special rules you mean the fact that == permits !=, > and that < permits other comparison operators. Are there other > special rules you mean? These rules seem to confuse people and I'm > increasingly inclined to think that we should just drop them. If you drop them that simplifies things but also creates further reliance on the contracts package since no one is going to want to enumerate all comparisons used when they can just say contracts.Ordered(T). It also means that you can have contracts like contract Less(t T) { t < t } contract Greater(t T) { t > t } While these accept exactly the same types, they accept completely different code in the implementation. > It is like solving a puzzle because you are trying to figure out how > to exclude some types. Why are you doing that? Why not just describe > the types you accept, and not worry about other types? Just as we do > for interface types? > To me contracts aren't implicit or roundabout at all, at least not > when it comes to which types they permit. They are very direct and > straightforward, and they really could not be simpler. But clearly > many people disagree. Perhaps we need to make them more complex by > turning them into a list of explicit constraints. This is an artificial example that I would hope never comes up in practice, but if you bear with me on it I think it illustrates several points: contract complex(s S, t T) { s.F[t] > <-t.G(s) } This is clearly written in code golf style. It can be written more explicitly as: contract complex(s S, t T) { var fieldF = s.F var indexResult = fieldF[t] var selG = t.G var invocationResult = selG(s) var channelResult = <-invocationResult indexResult > channelResult } To be fair, I said more explicitly—not more clearly! You can't use the contracts package to simplify this because indexResult and channelResult aren't types, they're values of an unspecified type that is entirely implicit in the contract definition. So if < doesn't imply the other operators on ordered types you have to do contract complex(s S, t T) { var a = s.F[t] var b = <-t.G(s) a == b a != b a < b // etc. } You could introduce a third type parameter like contract complex(s S, t T, _ U) { var a U = s.F[t] var b U = <-t.G(s) contracts.Comparable(U) } But in addition to having an extra, arguably useless type parameter to carry around, this is a different contract. It only asserts that s.F[t] and <-t.G(s) are assignable to U whereas the original asserted they were identical types (arguably fine). You could special case contracts-in-contracts to allow expressions of a type to be used as that type when composing contracts. Then you could write something like contract complex(s S, t T) { var a = s.F[t] contracts.Comparable(a) contracts.IdenticalType(a, <-t.G(s)) } That does give rise to meta-contracts like IdenticalType that exist only to be used with this special case, though. (Just so no one thinks I've cheated and made up an impossible contract, here are some examples that satisfy it: type S1 struct { F []T } type T1 int func (T1) G(s S1) chan T1 type S2 struct { F map[*T2]float64 } type T2 struct { G func(S2) chan float64 } The contract is satisfied by the pairs (S1, T1), (*S1, T1), (S1, *T1), (*S1, *T1), (S2, *T2), and (*S2, *T2).) All of that said I do think contracts would work and I agree that they are simple. They're simple to implement (assuming a sub-exponential type inference algorithm exists). They keep the language simple in that it adds few things to the spec. It's relatively mechanical to extract a contract from an existing non-generic implementation. My problem with them is that they are the wrong kind of simple. They are not simple in the sense of being simple to reason about in general. They are not simple in the sense that they are not orthogonal to interfaces. They are not simple in that let you get overly specific about some things that no one should care about but give no means whatsoever to express other things that one would care about. I mean you could say that about all code but this is a metatype system not regular code, despite appearances. Maybe these are curmudgeonly concerns. It's not a technical argument, but it still just doesn't sit right with me. -- 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.