Likewise, too many people in this discussion are going from “this doesn’t work how I expected/wanted it to work” to “this is a bug”. It’s not a bug. It works exactly as it was intended to work. It wasn’t a mistake or an oversight, and it wasn’t due to not thinking it through. It was almost certainly implemented this way because other similar pppular languages work the same way.
There may be many things you don’t like about a piece of software, but not everything you dislike is a bug.
And please don’t lose sight of the fact that not everyone agrees with your expectations. I will repeat: based on my experiences with other languages this feature behaves exactly as I would expect, and if it didn’t then I would be surprised. You can’t justify changing this just because you expected something else. You have to make a stronger case than that. And considering this would be a breaking change I think you need a very strong case, which I have yet to see here.
Default function parameters behave the same way in C++ as in Swift.
It's not necessarily desirable, but it's implementable, implemented, and already shipping.
Keeping all of C++'s gotchas in mind is why I prefer not to use it anymore. (I started programming in C++ before it had templates.)
Back to Swift: imagine this scenario you compile against a base class, calling a method f() which has a default argument. Your code doesn't instantiate subclasses, you're calling a factory. Now you link against modules defining subclasses and those subclasses are what your code is using at runtime. You don't create instances of those subclasses yourself, that's hidden by the factory. Now how are you going to know that some subclasses override the default argument's value and/or type? When you wrote the code, you could have explicitly passed in the same value as the default value. How are the subclasses going to know whether you passed in that value, or that value was a default value?
There are reasons default argument values are bound at the call-site.
Is that a joke? You are claiming that a default value = 3 is somehow a subtype of a default value = 12?
Also, how is the default value shown at the call site (since you claim it isn't internal)? Sure, it affects the signature (by not having to call one of the parameters), but that is very different than the value itself being a part of the signature.
My original point that one way or another it is a bug stands. Either it is a bug in the way @Nevin proposed, or it is a bug because the compiler isn't balking at the redefinition of the default value (as you try to override it with a different signature).
No, there's no subtyping relationship between default values.
A default value in an overriding method is not required to be the same as the default value in the overridden method. It is one other way in which signatures can be different between the two methods, just as covariant return types are another. This is not a bug.
The default value is not shown to you at the call site because, well, it's a default. It is, however, baked into your compiled binary.
"That is how it works right now" does not mean that something is not a bug. If that were a valid argument, you could never have any bugs, because the program would always be working in the way it is working. Everything is just a tautology.
If subclasses can't change signatures, then why is the default value allowed to change if the default value is part of the signature? You simply seem to say "because it can" without offering any sort of justification for why that is, or offering a use case where it could be helpful.
With contravariant parameter types, the parameter passed will always fit within the shape of the expected type. With covariant return types, the returned value will always fit within the shape of whatever is receiving the value. An Int will always fit in an Int?. That is why they are allowed.
Default values are something else entirely. They are values and not types. 12 will never fit in 3. You never said why signatures are allowed to have different default values and still be considered equivalent. Just that the compiler currently doesn't complain. The complier could act differently and be more consistent with the behavior of multiple functions which match the signatures created by default values.
Finally, just so we are on the same page, what is your exact definition of signature?
Jon, I refer you back to the examples given earlier. This thread exists precisely because the default value of the superclass method doesn't change after a subclass overrides the implementation.
Indeed, I am not writing about all the designs that could be. As I said earlier, I am writing to clarify the design that is because it's not been correctly described. One cannot call a design inconsistent without knowing what that design is.
Good point; there's not one consistent usage, I suppose. I'm using it to refer to all parts of the method definition that form part of the contract between the author and the user.
Now it's morning, I'll elaborate on my previous post:
The behaviour of default arguments is currently consistent with the shadowing behaviour of protocol extension methods, a larger issue which has similarly been labelled as a potential pitfall for newcomers. Static features often tend to look confusing and unexpected when mixed with dynamicism.
The way to get dynamic (overridable) behaviour of a protocol extension, is to declare the method in the protocol. It's fair to say we could do with some equivalent feature for default arguments. A strawman syntax:
As for the reasons to keep the current behaviour as default:
It's lightweight. A method with 10 arguments, all with defaults, doesn't generate an exponential number of methods internally. It's just one method, with arguments which can be filled in by the compiler.
It's resilient. A user may have an expectation of what a default argument is, and rely on it (it may appear in the signature in docs, etc…), and a library update isn't going to break that. Having a default argument behaves exactly like the user wrote that argument out explicitly, which is reasonable mental model for what's going on. If the library could change the default from under you, people would have to start writing out the argument anyway, for predictable behaviour.
By changing the default, we'd be exhanging pitfalls. Currently, worst case is the user expects dynamicism, and finds there is none. With this change, there'd be a performance hit from hidden method calls, and the default could change out from under you, both being much harder problems to spot.
I think it makes most sense to try accomodating both use cases with the dynamic annotation suggested above.
I wonder - would open be a more appropriate annotation?
For the non-dynamic/non-open case, given the potential for confusion over what it does, would you then mark an attempt to override the default as an error (or a warning, at least?).
Perhaps, but I'm not sure it's harmful enough. We'd have to justify the breaking change. We also currently allow shadowing of extension methods, and I think we should be consistent in both cases as to whether shadowing should be permitted.
I'm not sure about open. Aside from the access control connotations, this could apply to protocols, too. dynamic also has an existing meaning, though it's more rarely used. I can't immediately think of a name I'm completely happy with.
Ah. Under my definition func foo(x : Int = 12) actually provides two separate call signatures:
func foo(x : Int)
func foo()
When calling the second form, the caller shouldn't need to know or care about the default value. The form itself foo() should communicate everything needed to the caller, since the default is an internal implementation detail. Since they have the same signature, the implementation should be able to change to define the two cases separately without causing problems (given that the implementations still do the same thing).
That last bit currently fails, and that is one of the inconsistencies I am calling out here. The subclass currently has to know how the superclass implemented these signatures internally, because the override will behave differently based on whether it was implemented as a default value or as two separate methods.
Suppose we implement this by generating methods for every combination of parameters. With two defaulted arguments, this would be 4 signatures. With three, suddenly one simple method is generating 8 different methods. For a method with n arguments, all with defaults, you end up with 2^n total methods generated, all of which must be overridable for subclasses. Exponential systems like this is generally cause problems.
A better alternative would be to internally implement this with an overridable method for each argument. You would end up with generated calls along the lines of:
object.foo(x: 1)
object.foo(x: object.#foo_x_arg1_default()) //a generated method hidden to the user
This would support dynamic overriding of default parameters, and would fit in nicely with the dynamic modifier I proposed above. Given this, the compiler should be able to generate final methods for protocol conformance as-needed, which can forward to these dynamic methods.
As for changing the default, do the reasons I listed in my post above influence your opinion at all? Aside from whether it's intuitive, I'd consider what we have now the safest default.
Again, “is” and “should be” need to be distinguished. What Swift does today is emphatically not what you describe.
The default value is emitted at the call site during compilation. If a later version of a dynamically linked library changes its default, your compiled binary is unaffected. What you say “actually” happens is entirely not what happens.
Just because the compiler transforms foo() to foo(x: 12) behind the scenes doesn't mean that the call signature wasn't foo() to the writer of the call. The fact that 12 is passed to x is an implementation detail that the caller shouldn't have to know/reason about.
We are talking in circles here. Several people are saying "We need to change it because of ___" to which you keep replying "But it currently works _____ way. We want to change the way it works. That is why it is a pitch...
Instead of marking when we want dynamic behavior, can we mark when we want to allow the default to be inlined? This would make it more consistent with other times the compiler inlines across module boundaries.
If I was going to "fix this bug", I'd make it more static: if a subclass overrides a method that has default parameters, I'd require it to have the exact same default parameters. That would likely make some existing code start failing to compile, with gnashing of teeth and wailing.
If you want to stop the circle-talking, please stop calling this behavior a bug.
A bug doesn't need a pitch to Swift-evolution to fix. Swift-development is the better forum to fix the implementation as long as you aren't changing the behavior of what was decided or expected in Evolution or by the core team. e.g. if all default Int values were using a value of 0 instead. That's a bug since it's not doing what has been agreed on as correct behavior.
It does seem that you want to make a Pitch to change the expected and documented behavior, so Swift-evolution is the right place to debate such changes. But don't say it is a bug; it's clearly not, and if it is you're in the wrong forum to fix it. Point out that it's an inconsistency in the language, or that it creates confusion that can lead to bugs among users. But it is not a broken implementation of what was decided should be correct behavior; it's a behavior that you don't agree with.
No harm or foul intended here, just hoping you can see why the words of your pitch may be undermining the pitch itself.
I consider this a bug, and in fact I have one assigned to me for fixing this. In addition to capturing 'self', for default arguments of local functions it also makes sense to capture bindings from outer scopes.
I was talking about self, (instance itself) sorry if it was not clear. Of course, anything global (including static members and globally declared constants or variables that hold anything) work now.
I often run into cases that the sensible default for a method parameter is part of the object itself (one of its properties or instance functions).