The concept that the default value is part of the signature is just one way of handling default arguments, and most people in this discussion seem to be saying not a good way.
Most languages, all the dynamic languages (obviously) and some static languages like Scala, would behave as though the default was not part of the signature and instead be a shorthand for defining multiple methods. And as numerous example have pointed out this is a superior interpretation/way.
A key part of this argument, rather than saying that is how I interpret them because I am used to languages that do it that way, how about why is that interpretation superior to what the majority of languages do and what the majority of people expect.
It does! Both of these hold true! I think you're misunderstanding the example here. Let's write it out more clearly:
class C {
func f(_ i: Int = 42) {
print(i, "c")
}
init() { }
}
class D : C {
override func f(_ i: Int = 21) {
print(i, "d")
}
}
let c = C()
c.f() // 42 c
let d = D()
d.f() // 21 d
let dAsC = d as C
dAsC.f() // 42 d
Yes, of course there is that other consistent rule--we have that currently. Override implementations actually do override the superclass implementation. But it seems this is consistent and confusing. Hence, there's the alternative: remove the feature, and then there's no more confusion while still preserving consistency.
Great example of the current inconsistency, I am to treat default values as part of the signature yet I can define them in a protocol, but can in an extension! The default case is statically dispatched but the non default isn't!
If the interpretation where shorthand for multiple methods then:
Which is what most people would expect and it would also be consistent with protocols not having definitions since the protocol like now doesn't include the default, but this is no longer confusing since the default isn't part of the protocol.
I disagree that this is a matter of opinion. The signature is what it is. It's visible and made up of text; among that text are default values. You see it--it's literally (in every sense of literally) there. You can't just will that away.
Because default values are part of the signature, so too is its absence. Since printTest() without an argument isn't part of the protocol requirements (and currently can't be--we can make that additive change though), printTest() without an argument is statically dispatched and can only be shadowed, not overridden.
Yes, this is confusing. We can definitely improve the diagnostics and clarify these rules.
By that argument then the following shouldn't be allowed:
class B {
var ex = 1
}
class D: B {
override init() {
super.init()
ex = 2
}
}
let dAsB: B = D()
print(dAsB.ex) // 2
Since ex = 1 is part of the signature, "You see itāitās literally (in every sense of literally) there. You canāt just will that away.", then the above shouldn't be allowed. Your argument makes no sense, in functions the = xxx is part of the signature but in the rest of the language that very same syntax isn't!
That's clearly a faulty argument, because in making it you've willed away the other part of the signature: var.
var tells you everything you need to know about why ex can be some other value. Swift lets you change it because, in the contract that's the signature, you literally said that you reserved the right to do so.
But we are not talking about a final function which would be the nearest equivalent of a let, var is closest to an overridable function. The analogy isn't perfect, non are they are after all analogies. The point is you are happy for a variable not to be tied to its default value but not an argument, which is after all very similar.
I've acknowledged this option's existence, but it's certainly not consistent: subclasses cannot change the signature of superclasses, as I've already said.
Yes I know. But "moo" comes from the derived class just like the default argument should even though the Cow object has been type cast to Animal. (Actually there is a mistake in the code but we all know what you mean.)
The value of an instance variable is not part of the API of the type; if it's a let binding, then arguably it could be, but with a var binding it cannot be for obvious reasons. The signature of a method, of which default values are a part, is a part of the API of the type.
The code runs as-is and produces the result indicated (Xcode 9.3b4).
"moo" comes from the body of the method; the type comes from its signature. Default arguments are in the signature. I don't know what we're disagreeing about anymore at this point.
It is entirely consistent, just a different set of rules.
If you have reasons for why one set of rules is preferable, in the context of designing a language for use by actual human programmers, by all means please share those reasons.
Merely stating ādefault arguments are part of the signatureā is not effective, because the proposal is exactly to *change* how that works. If you think it should stay working as it does today, then please explain *why* you prefer the current behavior, rather than just repeatedly stating what the current behavior is.
I wasn't aware that you were proposing to change the rules.
You called the behavior [edit for clarity: behavior of override functions] a bug, and I explained why in fact it's not: it's consistent with the current design of Swift [edit for clarity: the current design of Swift wherein default values are part of the method signature]. I have no opinion whatsoever about whether the current design is good (in fact, I agree that it can be confusing); I'm just explaining over and over again what it is because there seems to be pervasive confusion about it.
If you understand what it is today and want to change it, then it's up to you to explain why such a change would be superior. Unless I'm mistaken, supporting default values as part of the signature is part of this whole business of moving away from tuples-as-argument-lists that also removed implicit tuple splatting, argument shuffle, etc. If I'm right, then it's a deliberate recent change, and reverting it isn't in the cards.