P0091R0
Revision of N4471
2015-09-24
Mike Spertus, Symantec
mike_spertus@symantec.com
Richard Smith
richard@metafoo.co.uk
This paper proposes extending template parameter deduction for functions to constructors of template classes and incorporates feedback from the EWG review of N4471.
Currently, if we want to construct template classes, we need to specify the template parameters.
For example, N4498 on Variadic Lock Guards
gives an example of acquiring a lock_guard on two mutexes inside an operator= to properly lock both the
source and destination of the assignment. Expanding typedefs, the locks are acquired by the following statement
(See the paper for details and rationale).
The sections below first spell out the problem in more detail and then makes precise what “like we expect from other functions and methods” means in this context.
To simplify the examples below, suppose the following definitions are in place.
In the current standard, the following objects would be constructed as shown
We propose two techniques
These techniques work well together as will be explained below. They can also be adopted separately (E.g., as a result of the discussion on Code Compatibility below).We propose to allow a template name referring to a class template as a simple-type-specifier in two contexts:
In the case of a function-notation type conversion (e.g., "tuple(1, 2.0, false)") or a direct parenthesized or braced initialization, the initialization is resolved as follows. First, constructors and constructor templates declared in the named template are enumerated. Let Ci be such a constructor or constructor template; together they form an overload set. A parallel overload set F of function templates is then created as follows: For each Ci a function template is constructed with template parameters that include both those of the named class template and if Ci is a constructor template, those of that template (default arguments are included too) -- the function parameters are the constructor parameters, and the return type is void Deduction and overload resolution is then performed for a synthesized call to F with the parenthesized or braced expressions used as arguments. If that call doesn't yield a "best viable function", the program is ill-formed. Otherwise, the template name is treated as the class-name that is obtained from the named class template with the deduced arguments corresponding to that template's parameters.
Let's look at an example:
Note that after the deduction process described above the initialization may still end up being ill-formed. For example, a selected constructor might be inaccessible or deleted, or the selected template instance might have been specialized or partially specialized in such a way that the candidate constructors will not match the initializer.
The case of a simple-declaration with copy-initialization syntax is treated similarly to the approach described above, except that explicit constructors and constructor templates are ignored, and the initializer expression is used as the single call argument during the deduction process.While the above procedure generates many useful deducible constructors, some constructors that we would like to be deducible are not. For example, one could imaging a function make_vector defined as follows:
Although there is no constructor in vector from which we can deduce the type
of the vector from two iterators, one would like to be able to deduce the type of
the vector from the value type of the two iterators.
For example, some implementations of the STL define their value_type
typedef as follows
We suggest a notation to allow constructors
to specify their template parameters by either explicitly declaring the signatures for
any further needed constructor deductions outside the class
In effect, this allows any pure factory function to be specified with any deduction rules that are specifiable by any function with a standard first-class name and no boilerplate code in the body. It also allows us to suppress a standard deduction from the above process via “= delete;”
Alternatively, we could use “declaration notation”
Note that it is only necessary to declare canonical factory functions. Giving a definition is not allowed as they just construct their return type from their arguments according to normal rules. This restriction makes it instantly visible that any use of a constructor as a factory simply constructs the class and does not have any obscure semantics.
The focus on this paper is on simplifying the interface of a class for its clients. Within a class, one may need to explicitly specify the parameters as before due to the injected class name:
Suppose I produce a library and I'm under license to preserve source compatibility across all 1.x upgrades, and I have this class template in version 1.0:
... and in version 1.1 I rewrite it as this:
If one of my users upgrades to C++17, with this change in the language, I am no longer complying with the terms of my licensing. Likewise, if this language change happens between me releasing 1.0 and 1.1, I can no longer release version 1.1 because it might break some of my existing customers.
The point is: current code does not express any intent about whether class template parameters are deducible, based on whether they use the version 1.0 code or the version 1.1 code. But this change makes that implicit property into part of the de facto interface of the code.
So why include auto-deduction at all in this proposal as canonical factory functions could be used to make explicit all of the auto-deduced constructors? Basically, having to manually specify what is obviously expected has an insidious cost as any (honest) Java programmer can tell you so we would like to discuss with the committee whether this will be an actual problem in practice before jettisoning a useful feature and make programmers manually insert the “obvious” canonical factory functions for all of their existing classes.