1. Changelog
- 
     R4 - 
       Dropped the proposed default from the non-range version of replace_copy 
- 
       Fixed minor misspellings. 
- 
       Clarified the wording for some design decisions regarding the numeric algorithms (no design has actually changed). 
 
- 
       
- 
     R3 - 
       Fixed the default for the range version of replace_copy_if 
- 
       Discussed the problem with projections in range-based algorithms. 
 
- 
       
- 
     R2 - 
       Elaborated on the rationale. 
- 
       Elaborated on [algorithms.requirements] and on [SD-8]. 
- 
       Elaborated on defaulting value types for algorithms that use output iterators. 
- 
       Rebased on the top of the latest Standard. 
- 
       Added a link to a prototype implementation. 
- 
       Fixed the defaults for the * _scan 
 
- 
       
- 
     R1 - 
       Clarified some possible sources of ABI incompatibility, which were (guiltly) not discussed in R0. 
- 
       Added a reference and some discussion related to [P1997R1]. 
- 
       Added a reference to [algorithms.requirements]. 
- 
       Fixed misspellings / formatting. 
 
- 
       
- 
     R0 - 
       First submission 
 
- 
       
2. Tony Tables
| Before | After | 
|---|---|
| 
 | 
 | 
3. Motivation and Scope
List-initialization ([dcl.init.list]) has been introduced in C++11. When an initializer list is used as a function argument, overload resolution works in a specific way to find a suitable overload, possibly constructing a temporary object of the argument’s type (see [over.ics.list]).
This language feature allows to write code like this:
void f ( std :: complex < double > ); f ({ 1.23 , 4.56 }); // OK 
In case of function templates, the deduction rules do not allow deduction in the general case, because a type cannot be determined ([temp.deduct.call]). It is however allowed to use an initalizer list in case the corresponding template type parameter has a default:
template < typename T > void g ( T ); g ({ 1 , 2 }); // ERROR: cannot deduce T template < typename T = std :: complex < double >> void h ( T ); h ({ 1.0 , 2.0 }); // OK, the default is used for deducing T 
Default template type parameters have a somehow poor adoption in the
Standard Library. They are used for instance in 
namespace std { // [utility.exchange] template < class T , class U = T > constexpr T exchange ( T & obj , U && new_val ); } // thanks to U defaulted to T, we can write this: std :: complex < double > old_val ; auto new_val = std :: exchange ( old_val , { 3.0 , 4.0 }); // OK // old_vec is moved onto new_vec, and reset to a default-constructed state std :: vector < int > old_vec ; auto new_vec = std :: exchange ( old_vec , {}); // OK 
Unfortunately, they are not used in a number of other places, such as some standard algorithms, although reasonable defaults could still be determined based on other available template parameters.
Let’s consider C++20’s 
namespace std { // [vector.erasure] template < class T , class Allocator , class U > constexpr typename vector < T , Allocator >:: size_type erase ( vector < T , Allocator >& c , const U & value ); } std :: vector < std :: complex < double >> numbers ; numbers . push_back ({ 1.0 , 2.0 }); // OK erase ( numbers , { 1.0 , 2.0 }); // ERROR: cannot deduce U 
Since 
Similarly, an algorithm like 
namespace std { // [alg.find] template < class InputIterator , class T > constexpr InputIterator find ( InputIterator first , InputIterator last , const T & value ); } std :: vector < std :: complex < double >> numbers ; numbers . push_back ({ 1.0 , 2.0 }); // OK std :: find ( vector . begin (), vector . end (), { 1.0 , 2.0 }); // ERROR 
In this case 
In a nutshell: we are proposing to add a default template type parameter to the "algorithmic" function templates in the Standard Library:
- 
      which have a "value"-like function argument; and 
- 
      for which a reasonable default type can be inferred from the other template type parameters. 
This enables users to call these functions by using list-initialization for the "value"-like arguments.
4. Impact On The Standard
The proposed changes only affect existing Standard Library function templates. The impact is mostly positive: syntax that was ill-formed before becomes well-formed. There is a chance of incompatibility w.r.t. user code, but [SD-8] allows the kind of changes we are proposing.
This proposal does not depend on any other library extensions.
This proposal does not require any changes in the core language.
The first part of [P2218R0] is related to this proposal. [P2218R0] proposes to default the template type parameter of 
[P1997R1] proposes some changes to the core language in order to make array types copiable, assignable, etc. We are not proposing any changes to the language; however it’s interesting to note that, if [P1997R1] is adopted, then using arrays as value types in containers and algorithms may become more widespread. (Right now, array types are usable in containers/algorithms in extremely limited ways.) The present proposal would then make it possible to use list-initialization in order to create arrays as the arguments of algorithmic-like functions, for instance like this:
// in principle, possible under P1997R1 std :: vector < int [ 2 ] > v ( 42 ); v . push_back ({ 1 , 2 }); // in principle, possible under P1997R1 + this proposal // (T defaults to int[2], second parameter is deduced as const int(&)[2], // then aggregate initialization is selected) std :: ranges :: fill ( v , { 123 , 456 }); 
5. Design Decisions
5.1. What is the rationale for the default chosen in each case?
We have tried to keep the choice of the default as much straightforward as possible, by following this set of rules/guidelines.
In the following, 
5.1.1. Consistent Container Erasure
These would simply default 
5.1.2. Algorithms with an output iterator/range (where T 
   For those we are proposing to use the 
The algorithms in this category that we are proposing to change are:
- 
     fill_n 
- 
     replace_copy 
- 
     replace_copy_if 
5.1.3. Algorithms with a (mutable) input iterator/range (where T 
   Similarly, we’re proposing to default the value-like type to the 
The algorithms in this category that we are proposing to change are:
- 
     find 
- 
     count 
- 
     search_n 
- 
     replace 
- 
     replace_if 
- 
     replace_copy 
- 
     fill 
- 
     remove 
- 
     remove_copy 
- 
     lower_bound 
- 
     upper_bound 
- 
     equal_range 
- 
     binary_search 
5.1.4. Numeric algorithms
In general, numeric algorithms deserve some additional reasoning.
An exception is 
For 
// not proposed template < class InputIterator , class BinaryOperation , class T = invoke_result_t < BinaryOperation & , typename iterator_traits < InputIterator >:: value_type & , typename iterator_traits < InputIterator >:: value_type &>> constexpr T accumulate ( InputIterator first , InputIterator last , T init , BinaryOperation binary_op ); 
This however conflicts with the existing semantics.
Specifically, the version
of 
In other words: the already existing default does not aim for any sort of
complicated mathematical purity, but rather for a much more straightforward
definition. Based on this precedent, we are simply proposing to default 
This simplification is also applied to
- 
     exclusive_scan 
- 
     inclusive_scan 
- 
     transform_exclusive_scan 
- 
     transform_inclusive_scan 
for which 
Finally, the algorithms that work on multiple input sequences
(
what does SG6 think about this?
5.1.5. The problem of establishing a default for replace_copy 
   The non-range version of 
template < class InputIterator , class OutputIterator , class T > constexpr OutputIterator replace_copy ( InputIterator first , InputIterator last , OutputIterator result , const T & old_value , const T & new_value ); 
The issue with this signature is that there is only one template
type parameter for both the old value and the new value. (This is
probably an historical mistake; changing 
Now, if we were to pick a default for 
Consider this example, from a discussion on the LEWG reflector:
std :: vector < std :: pair < int , int >> intpairs ; std :: vector < std :: pair < float , float >> floatpairs ; std :: replace_copy ( intpairs . begin (), intpairs . end (), floatpairs . begin (), { 1 , 2 }, { 3.14 , 3.14 }); // <- 
If 
If 
std :: replace_copy ( floatpairs . begin (), floatpairs . end (), intpairs . begin (), { 3.14 , 3.14 }, // <- { 1 , 2 }); 
Now the fourth parameter will be converted to 
In conclusion: given no choice is strongly better than the other, and
given that ultimately the problem is due to an API flaw of 
The range version of 
5.2. Is this API compatible?
We are fairly certain that the proposed changes keep API compatibility.
Technically speaking, defaulting a template type parameter of a function template is a detectable change. Here’s an example (many thanks to of Arthur O’Dwyer, who provided a much simpler one than ours):
template < typename A , typename B /* = A */ > int f ( A , B ) { return 1 ; } int f ( int , int * ) { return 2 ; } int x = f ( 0 , { 0 }); 
In this code 
We do not think this is actually a meaningful problem. For this to affect client code, such code must be performing a function call that somehow is including a Standard Library function during its overload resolution, and then selecting some other function as the best candidate. If that’s the case, any change in the Standard Library (such as adding an overload) could affect the overload resolution and select a different candidate.
In a similar way, one can detect the presence of a defaulted template parameter via SFINAE. If we take a "prototype" version of an algorithmic function from the Standard Library that accepts two iterators and a "value":
template < typename I , typename T > I algorithm ( I , I , const T & ); 
then a call such as 
template < typename I , typename T = typename iterator_traits < I >:: value_type > I algorithm ( I , I , const T & ); 
then we are going to allow deduction for the corresponding function parameter, and the call before may become well-formed.
In principle, this will mean that code using SFINAE or other similar techniques may change behavior. We think that this is extremely unlikely to be happening in practice; if such code exists, it’s certainly a mistake, as it would always currently be failing deduction (given the current specification of the Standard Library algorithms).
In conclusion, although an API break is possible in principle, it’s extremely unlikely to be happening in practice. What’s more, the kind of proposed changes fall under what the Standard Library is allowed to do. [SD-8] in fact gives us the explicit permission to default template parameters for function templates in the Standard Library:
Primarily, the standard reserves the right to:
[...]
Add new default arguments to functions and templates
Based on this document, users are already not supposed to rely on the absence of such defaults.
5.3. Is this ABI compatible?
This proposal deals with two kind of changes to function templates.
5.3.1. Changes to consistent container erasure and non-range-based algorithms
Here we are simply proposing to default a template type parameter in the template parameter list of the related functions. This is entirely ABI compatibile (we’re dealing with function templates, and not class templates).
5.3.2. Changes to range-based algorithms
These changes require reordering the template argument list (see the discussion in § 5.4 Range-based algorithms). This is unfortunately not 100% ABI compatibile, as the instantiations of these functions would change their mangling.
With the reorderings proposed, we feel it’s very unlikely that this
incompatibility would cause a problem in practice. A user would have to call
range-based algorithm using the same type for the template parameters that got
reordered. For instance, for 
In the absence of such a type clash, then the problem would be averted: old code would keep using the old mangled name, and code compiled against the new version would use the new name. (The final binary may end up with multiple copies of the instantiated function, under different symbols.)
There would still be the chance of an ABI break for user code using explicit template instantiations of one of these functions. The TU defining the instantiation would then be incompatible with any other TU that uses a different version of these functions (and merely declares the instantation).
We are not sure that user code is actually allowed to use explicit instantiations of Standard Library functions. Such a usage seems to violate the [SD-8] directives, specifically the right of the Standard Library to:
Make changes to existing interfaces in a fashion that will be backward compatible, if those interfaces are solely used to instantiate types and invoke functions. Implementation details (the primary name of a type, the implementation details for a function callable) may not be depended upon.
For example, we may change implementation details for standard function templates so that those become callable function objects. If user code only invokes that callable, the behavior is unchanged.
Cf. also the discussion in the § 5.5 What about user-specified template parameters? section.
5.4. Range-based algorithms
The current specification in the Standard for range-based algorithms does not allow for a "straightforward" defaulting, like the non-range-based variants.
template < input_iterator I , sentinel_for < I > S , class T , class Proj = identity > requires indirect_binary_predicate < ranges :: equal_to , projected < I , Proj > , const T *> constexpr I find ( I first , S last , const T & value , Proj proj = {}); template < input_range R , class T , class Proj = identity > requires indirect_binary_predicate < ranges :: equal_to , projected < iterator_t < R > , Proj > , const T *> constexpr borrowed_iterator_t < R > find ( R && r , const T & value , Proj proj = {}); 
Both declarations do not allow for 
In order to fix this, we need to reorder the template parameters.
Some other algorithms need a slightly more elaborate change.
Let’s consider 
template < class T , output_iterator < const T &> O , sentinel_for < O > S > constexpr O fill ( O first , S last , const T & value ); template < class T , output_range < const T &> R > constexpr borrowed_iterator_t < R > fill ( R && r , const T & value ); 
Here the output iterator 
To summarize, the changes that we are proposing to the range-based algorithms go in this direction: they move template parameters and/or constraints in order to be able to give a meaningful default type to the "value"-like function parameters.
We believe that [SD-8] allows us to do so under the interpretation that the exact spelling or ordering of template type arguments and constraints is part of the "implementation details" that users cannot depend upon. There will be no changes for users who simply call these functions, letting the template arguments to be deduced.
5.5. What about user-specified template parameters?
For the non-algorithmic changes (that is, the ones to consistent container erasure), we are not changing the order nor the meaning of the template parameters. User code that is calling such functions specifying the template type parameters explicitly is therefore not going to be affected.
For the algorithms, users are already not allowed to rely on the number and the order of deducible template parameters. [algorithms.requirements] / 15 says:
The number and order of deducible template parameters for algorithm declarations are unspecified, except where explicitly stated otherwise.[Note 3: Consequently, an implementation can reject calls that specify an explicit template argument list. — end note]
Therefore, we do have the freedom not only of defaulting them, but also of reordering them when necessary.
Please note, however, that in spite of this paragraph in the Standard and of the SD-8 rules, the kind of changes we are proposing may be ABI incompatible. See § 5.3 Is this ABI compatible? for a discussion.
5.6. What about output iterators, whose value_type void 
   In this case, the changes proposed by this paper would not bring any effect; an user would still not be able to call an algorithm using an initializer list without also specifying the type to deduce.
5.7. The need of a custom projection type transformation
Many range-based algorithms feature a projection function. Let’s use 
template < input_range R , class T , class Proj = identity > requires indirect_binary_predicate < ranges :: equal_to , projected < iterator_t < R > , Proj > , const T *> constexpr borrowed_iterator_t < R > find ( R && r , const T & value , Proj proj = {}); 
For these algorithms 
struct point { int x ; int y ; }; struct tagged_point { std :: string tag ; point p ; }; std :: vector < tagged_point > v ; auto i = std :: ranges :: find ( v , { 1 , 2 }, // should be a point because of the projection & tagged_point :: p ); 
One could therefore conclude that the correct default for 
Unfortunately, as discussed by [P2322R5], this does not work in all cases; ranges yielding proxy iterators would end up using the wrong type as default.
If 
std :: vector < bool > v ; auto i = std :: ranges :: find ( v , { true}); // ERROR 
Another example (again brought forward by [P2322R5]) is the case of
of 
std :: vector < int > v1 , v2 ; auto i = std :: ranges :: find ( std :: views :: zip ( v1 , v2 ), { 3 , 4 }); // ERROR 
The problem here is that 
The default for a value-like parameter does not however need reference
semantics -- it needs value semantics, projecting a range’s value
type. Specifically, projecting 
In short, the correct default for 
With this, the default for a 
Technically speaking, in the general case, an projection may return
different types when invoked with a 
struct evil_projection { int operator ()( const bool & ) const ; double operator ()( bool && ) const ; }; 
used with a 
5.7.1. Providing the projected value as an API
While we could put the above solution to the projected value as an
exposition-only type transformation, we are actually proposing to
modify 
To this end, we are proposing to add a 
This allows us to reformulate the default for 
which is the form found in the proposed wording.
6. Implementation experience
A working prototype of the changes proposed by this paper, done on top of GCC 11, is available in this GCC branch on GitHub.
7. Technical Specifications
All the proposed changes are relative to [N4892].
7.1. Feature testing macro
Add to the list in [version.syn]:
#define __cpp_lib_default_template_type_for_algorithm_values YYYYMML // also in // <algorithm>, <ranges>, <numeric>, // <string>, <deque>, <list>, <forward_list>, <vector> 
with the value specified as usual (year and month of adoption).
7.2. Consistent Container Erasure
Modify [string.syn] and [string.erasure]:
template < class charT , class traits , class Allocator , class U = charT > constexpr typename basic_string < charT , traits , Allocator >:: size_type erase ( basic_string < charT , traits , Allocator >& c , const U & value ); 
Modify [deque.syn] and [deque.erasure]:
template < class T , class Allocator , class U = T > typename deque < T , Allocator >:: size_type erase ( deque < T , Allocator >& c , const U & value ); 
Modify [forwardlist.syn] and [forward.list.erasure]:
template < class T , class Allocator , class U = T > typename forward_list < T , Allocator >:: size_type erase ( forward_list < T , Allocator >& c , const U & value ); 
Modify [list.syn] and [list.erasure]:
template < class T , class Allocator , class U = T > typename list < T , Allocator >:: size_type erase ( list < T , Allocator >& c , const U & value ); 
Modify [vector.syn] and [vector.erasure]:
template < class T , class Allocator , class U = T > constexpr typename vector < T , Allocator >:: size_type erase ( vector < T , Allocator >& c , const U & value ); 
7.3. Iterators
Modify the 
template < class F , class ... Is > requires ( indirectly_ readable < Is > && ...) && invocable < F , iter_reference_t < Is > ... > using indirect_result_t = invoke_result_t < F , iter_reference_t < Is > ... > ; template < class F , class ... Is > requires ( indirectly_ readable < Is > && ...) && invocable < F , iter_value_t < Is >& ... > using indirect_value_result_t = invoke_result_t < F , iter_value_t < Is >& ... > ; // [projected], projected template < indirectly_ readable I , indirectly_ regular_ unary_ invocable < I > Proj > struct projected ; template < weakly_ incrementable I , class Proj > struct incrementable_traits < projected < I , Proj >> ; // [projected.value], projected value template < indirectly_ readable I , indirectly_ regular_ unary_ invocable < I > Proj > struct projected_value ; template < weakly_ incrementable I , class Proj > struct incrementable_traits < projected_value < I , Proj >> ; 
Add a new section at the end of [indirectcallable]:
??.??.??.?? Class template[projected.value]projected_value 
1. Class templateis used to constrain algorithms that accept callable objects and projections ([defns.projection]). It combines aprojected_value typeindirectly_ readable and a callable object typeI into a newProj type whoseindirectly_ readable type is the result of applyingreference to an lvalue reference to theProj ofiter_ value_ t .I namespace std { template < indirectly_ readable I , indirectly_ regular_ unary_ invocable < I > Proj > struct projected_value { using value_type = remove_cvref_t < indirect_value_result_t < Proj & , I >> ; indirect_value_result_t < Proj & , I > operator * () const ; // not defined }; template < weakly_ incrementable I , class Proj > struct incrementable_traits < projected_value < I , Proj >> { using difference_type = iter_difference_t < I > ; }; } 
7.4. Algorithms
Modify both the 
7.4.1. [alg.find]
template < lass InputIterator , class T = typename iterator_traits < InputIterator >:: value_type > constexpr InputIterator find ( InputIterator first , InputIterator last , const T & value ); template < class ExecutionPolicy , class ForwardIterator , class T = typename iterator_traits < InputIterator >:: value_type > ForwardIterator find ( ExecutionPolicy && exec , ForwardIterator first , ForwardIterator last , const T & value ); 
namespace ranges { template < input_iterator I , sentinel_for < I > S , class T , class Proj = identity , class T = typename projected_value < I , Proj >:: value_type > requires indirect_binary_predicate < ranges :: equal_to , projected < I , Proj > , const T *> constexpr I find ( I first , S last , const T & value , Proj proj = {}); template < input_range R , class T , class Proj = identity , class T = typename projected_value < iterator_t < R > , Proj >:: value_type > requires indirect_binary_predicate < ranges :: equal_to , projected < iterator_t < R > , Proj > , const T *> constexpr borrowed_iterator_t < R > find ( R && r , const T & value , Proj proj = {}); } 
7.4.2. [alg.count]
template < class InputIterator , class T = typename iterator_traits < InputIterator >:: value_type > constexpr typename iterator_traits < InputIterator >:: difference_type count ( InputIterator first , InputIterator last , const T & value ); template < class ExecutionPolicy , class ForwardIterator , class T = typename iterator_traits < InputIterator >:: value_type > typename iterator_traits < ForwardIterator >:: difference_type count ( ExecutionPolicy && exec , ForwardIterator first , ForwardIterator last , const T & value ); 
namespace ranges { template < input_iterator I , sentinel_for < I > S , class T , class Proj = identity , class T = typename projected_value < I , Proj >:: value_type > requires indirect_binary_predicate < ranges :: equal_to , projected < I , Proj > , const T *> constexpr iter_difference_t < I > count ( I first , S last , const T & value , Proj proj = {}); template < input_range R class T , class Proj = identity , class T = typename projected_value < iterator_t < R > , Proj >:: value_type > requires indirect_binary_predicate < ranges :: equal_to , projected < iterator_t < R > , Proj > , const T *> constexpr range_difference_t < R > count ( R && r , const T & value , Proj proj = {}); } 
7.4.3. [alg.search]
template < class ForwardIterator , class Size , class T = typename iterator_traits < ForwardIterator >:: value_type > constexpr ForwardIterator search_n ( ForwardIterator first , ForwardIterator last , Size count , const T & value ); template < class ForwardIterator , class Size , class T = typename iterator_traits < ForwardIterator >:: value_type , class BinaryPredicate > constexpr ForwardIterator search_n ( ForwardIterator first , ForwardIterator last , Size count , const T & value , BinaryPredicate pred ); template < class ExecutionPolicy , class ForwardIterator , class Size , class T = typename iterator_traits < ForwardIterator >:: value_type > ForwardIterator search_n ( ExecutionPolicy && exec , ForwardIterator first , ForwardIterator last , Size count , const T & value ); template < class ExecutionPolicy , class ForwardIterator , class Size , class T = typename iterator_traits < ForwardIterator >:: value_type , class BinaryPredicate > ForwardIterator search_n ( ExecutionPolicy && exec , ForwardIterator first , ForwardIterator last , Size count , const T & value , BinaryPredicate pred ); 
namespace ranges { template < forward_iterator I , sentinel_for < I > S , class T , class Pred = ranges :: equal_to , class Proj = identity , class T = typename projected_value < I , Proj >:: value_type > requires indirectly_comparable < I , const T * , Pred , Proj > constexpr subrange < I > search_n ( I first , S last , iter_difference_t < I > count , const T & value , Pred pred = {}, Proj proj = {}); template < forward_range R , class T , class Pred = ranges :: equal_to , class Proj = identity , class T = typename projected_value < iterator_t < R > , Proj >:: value_type > requires indirectly_comparable < iterator_t < R > , const T * , Pred , Proj > constexpr borrowed_subrange_t < R > search_n ( R && r , range_difference_t < R > count , const T & value , Pred pred = {}, Proj proj = {}); } 
7.4.4. [alg.replace]
template < class ForwardIterator , class T = typename iterator_traits < ForwardIterator >:: value_type > constexpr void replace ( ForwardIterator first , ForwardIterator last , const T & old_value , const T & new_value ); template < class ExecutionPolicy , class ForwardIterator , class T = typename iterator_traits < ForwardIterator >:: value_type > void replace ( ExecutionPolicy && exec , ForwardIterator first , ForwardIterator last , const T & old_value , const T & new_value ); template < class ForwardIterator , class Predicate , class T = typename iterator_traits < ForwardIterator >:: value_type > constexpr void replace_if ( ForwardIterator first , ForwardIterator last , Predicate pred , const T & new_value ); template < class ExecutionPolicy , class ForwardIterator , class Predicate , class T = typename iterator_traits < ForwardIterator >:: value_type > void replace_if ( ExecutionPolicy && exec , ForwardIterator first , ForwardIterator last , Predicate pred , const T & new_value ); 
namespace ranges { template < input_iterator I , sentinel_for < I > S , class T1 , class T2 , class Proj = identity , class T1 = typename projected_value < I , Proj >:: value_type , class T2 = T1 > requires indirectly_writable < I , const T2 &> && indirect_binary_predicate < ranges :: equal_to , projected < I , Proj > , const T1 *> constexpr I replace ( I first , S last , const T1 & old_value , const T2 & new_value , Proj proj = {}); template < input_range R , class T1 , class T2 , class Proj = identity , class T1 = typename projected_value < iterator_t < R > , Proj >:: value_type , class T2 = T1 > requires indirectly_writable < iterator_t < R > , const T2 &> && indirect_binary_predicate < ranges :: equal_to , projected < iterator_t < R > , Proj > , const T1 *> constexpr borrowed_iterator_t < R > replace ( R && r , const T1 & old_value , const T2 & new_value , Proj proj = {}); template < input_iterator I , sentinel_for < I > S , class T , class Proj = identity , class T = typename projected_value < I , Proj >:: value_type , indirect_unary_predicate < projected < I , Proj >> Pred > requires indirectly_writable < I , const T &> constexpr I replace_if ( I first , S last , Pred pred , const T & new_value , Proj proj = {}); template < input_range R , class T , class Proj = identity , class T = typename projected_value < iterator_t < R > , Proj >:: value_type , indirect_unary_predicate < projected < iterator_t < R > , Proj >> Pred > requires indirectly_writable < iterator_t < R > , const T &> constexpr borrowed_iterator_t < R > replace_if ( R && r , Pred pred , const T & new_value , Proj proj = {}); } 
template < class InputIterator , class OutputIterator , class Predicate , class T = typename iterator_traits < OutputIterator >:: value_type > constexpr OutputIterator replace_copy_if ( InputIterator first , InputIterator last , OutputIterator result , Predicate pred , const T & new_value ); template < class ExecutionPolicy , class ForwardIterator1 , class ForwardIterator2 , class Predicate , class T = typename iterator_traits < ForwardIterator2 >:: value_type > ForwardIterator2 replace_copy_if ( ExecutionPolicy && exec , ForwardIterator1 first , ForwardIterator1 last , ForwardIterator2 result , Predicate pred , const T & new_value ); 
namespace ranges { template < class I , class O > using replace_copy_result = in_out_result < I , O > ; template < input_iterator I , sentinel_for < I > S , class T1 , class T2 , output_iterator < const T2 &> O , class Proj = identity > template < input_iterator I , sentinel_for < I > S , class O , class Proj = identity , class T1 = typename projected_value < I , Proj >:: value_type , class T2 = iter_value_t < O >> requires indirectly_copyable < I , O > && indirect_binary_predicate < ranges :: equal_to , projected < I , Proj > , const T1 *> && output_iterator < O , const T2 &> constexpr replace_copy_result < I , O > replace_copy ( I first , S last , O result , const T1 & old_value , const T2 & new_value , Proj proj = {}); template < input_range R , class T1 , class T2 , output_iterator < const T2 &> O , class Proj = identity > template < input_range R , class O , class Proj = identity , class T1 = typename projected_value < iterator_t < R > , Proj >:: value_type , class T2 = iter_value_t < O >> requires indirectly_copyable < iterator_t < R > , O > && indirect_binary_predicate < ranges :: equal_to , projected < iterator_t < R > , Proj > , const T1 *> && output_iterator < O , const T2 &> constexpr replace_copy_result < borrowed_iterator_t < R > , O > replace_copy ( R && r , O result , const T1 & old_value , const T2 & new_value , Proj proj = {}); template < class I , class O > using replace_copy_if_result = in_out_result < I , O > ; template < input_iterator I , sentinel_for < I > S , class T , output_iterator < const T &> O , class O , class T = iter_value_t < O > , class Proj = identity , indirect_unary_predicate < projected < I , Proj >> Pred > requires indirectly_copyable < I , O > && output_iterator < O , const T &> constexpr replace_copy_if_result < I , O > replace_copy_if ( I first , S last , O result , Pred pred , const T & new_value , Proj proj = {}); template < input_range R , class T , output_iterator < const T &> O , class O , class T = iter_value_t < O > , class Proj = identity , indirect_unary_predicate < projected < iterator_t < R > , Proj >> Pred > requires indirectly_copyable < iterator_t < R > , O > && output_iterator < O , const T &> constexpr replace_copy_if_result < borrowed_iterator_t < R > , O > replace_copy_if ( R && r , O result , Pred pred , const T & new_value , Proj proj = {}); } 
7.4.5. [alg.fill]
template < class ForwardIterator , class T = typename iterator_traits < ForwardIterator >:: value_type > constexpr void fill ( ForwardIterator first , ForwardIterator last , const T & value ); template < class ExecutionPolicy , class ForwardIterator , class T = typename iterator_traits < ForwardIterator >:: value_type > void fill ( ExecutionPolicy && exec , ForwardIterator first , ForwardIterator last , const T & value ); template < class OutputIterator , class Size , class T = typename iterator_traits < OutputIterator >:: value_type > constexpr OutputIterator fill_n ( OutputIterator first , Size n , const T & value ); template < class ExecutionPolicy , class ForwardIterator , class Size , class T = typename iterator_traits < ForwardIterator >:: value_type > ForwardIterator fill_n ( ExecutionPolicy && exec , ForwardIterator first , Size n , const T & value ); 
namespace ranges { template < class T , output_iterator < const T &> O , sentinel_for < O > S > template < class O , sentinel_for < O > S , class T = iter_value_t < O >> requires output_iterator < O , const T &> constexpr O fill ( O first , S last , const T & value ); template < class T , output_range < const T &> R > template < class R , class T = range_value_t < R >> requires output_range < R , const T &> constexpr borrowed_iterator_t < R > fill ( R && r , const T & value ); template < class T , output_iterator < const T &> O > template < class O , class T = iter_value_t < O >> requires output_iterator < O , const T &> constexpr O fill_n ( O first , iter_difference_t < O > n , const T & value ); } 
7.4.6. [alg.remove]
template < class ForwardIterator , class T = typename iterator_traits < ForwardIterator >:: value_type > constexpr ForwardIterator remove ( ForwardIterator first , ForwardIterator last , const T & value ); template < class ExecutionPolicy , class ForwardIterator , class T = typename iterator_traits < ForwardIterator >:: value_type > ForwardIterator remove ( ExecutionPolicy && exec , ForwardIterator first , ForwardIterator last , const T & value ); 
namespace ranges { template < permutable I , sentinel_for < I > S , class T , class Proj = identity , class T = typename projected_value < I , Proj >:: value_type > requires indirect_binary_predicate < ranges :: equal_to , projected < I , Proj > , const T *> constexpr subrange < I > remove ( I first , S last , const T & value , Proj proj = {}); template < forward_range R , class T , class Proj = identity , class T = typename projected_value < iterator_t < R > , Proj >:: value_type > requires permutable < iterator_t < R >> && indirect_binary_predicate < ranges :: equal_to , projected < iterator_t < R > , Proj > , const T *> constexpr borrowed_subrange_t < R > remove ( R && r , const T & value , Proj proj = {}); } 
template < class InputIterator , class OutputIterator , class T = typename iterator_traits < InputIterator >:: value_type > constexpr OutputIterator remove_copy ( InputIterator first , InputIterator last , OutputIterator result , const T & value ); template < class ExecutionPolicy , class ForwardIterator1 , class ForwardIterator2 , class T = typename iterator_traits < ForwardIterator1 >:: value_type > ForwardIterator2 remove_copy ( ExecutionPolicy && exec , ForwardIterator1 first , ForwardIterator1 last , ForwardIterator2 result , const T & value ); 
namespace ranges { template < input_iterator I , sentinel_for < I > S , weakly_incrementable O , class T , class Proj = identity , class T = typename projected_value < I , Proj >:: value_type > requires indirectly_copyable < I , O > && indirect_binary_predicate < ranges :: equal_to , projected < I , Proj > , const T *> constexpr remove_copy_result < I , O > remove_copy ( I first , S last , O result , const T & value , Proj proj = {}); template < input_range R , weakly_incrementable O , class T , class Proj = identity , class T = typename projected_value < iterator_t < R > , Proj >:: value_type > requires indirectly_copyable < iterator_t < R > , O > && indirect_binary_predicate < ranges :: equal_to , projected < iterator_t < R > , Proj > , const T *> constexpr remove_copy_result < borrowed_iterator_t < R > , O > remove_copy ( R && r , O result , const T & value , Proj proj = {}); } 
7.4.7. [alg.binary.search]
template < class ForwardIterator , class T = typename iterator_traits < ForwardIterator >:: value_type > constexpr ForwardIterator lower_bound ( ForwardIterator first , ForwardIterator last , const T & value ); template < class ForwardIterator , class T = typename iterator_traits < ForwardIterator >:: value_type , class Compare > constexpr ForwardIterator lower_bound ( ForwardIterator first , ForwardIterator last , const T & value , Compare comp ); 
namespace ranges { template < forward_iterator I , sentinel_for < I > S , class T , class Proj = identity , class T = typename projected_value < I , Proj >:: value_type indirect_strict_weak_order < const T * , projected < I , Proj >> Comp = ranges :: less > constexpr I lower_bound ( I first , S last , const T & value , Comp comp = {}, Proj proj = {}); template < forward_range R , class T , class Proj = identity , class T = typename projected_value < iterator_t < R > , Proj >:: value_type indirect_strict_weak_order < const T * , projected < iterator_t < R > , Proj >> Comp = ranges :: less > constexpr borrowed_iterator_t < R > lower_bound ( R && r , const T & value , Comp comp = {}, Proj proj = {}); } 
template < class ForwardIterator , class T = typename iterator_traits < ForwardIterator >:: value_type > constexpr ForwardIterator upper_bound ( ForwardIterator first , ForwardIterator last , const T & value ); template < class ForwardIterator , class T = typename iterator_traits < ForwardIterator >:: value_type , class Compare > constexpr ForwardIterator upper_bound ( ForwardIterator first , ForwardIterator last , const T & value , Compare comp ); 
namespace ranges { template < forward_iterator I , sentinel_for < I > S , class T , class Proj = identity , class T = typename projected_value < I , Proj >:: value_type indirect_strict_weak_order < const T * , projected < I , Proj >> Comp = ranges :: less > constexpr I upper_bound ( I first , S last , const T & value , Comp comp = {}, Proj proj = {}); template < forward_range R , class T , class Proj = identity , class T = typename projected_value < iterator_t < R > , Proj >:: value_type indirect_strict_weak_order < const T * , projected < iterator_t < R > , Proj >> Comp = ranges :: less > constexpr borrowed_iterator_t < R > upper_bound ( R && r , const T & value , Comp comp = {}, Proj proj = {}); } 
template < class ForwardIterator , class T = typename iterator_traits < ForwardIterator >:: value_type > constexpr pair < ForwardIterator , ForwardIterator > equal_range ( ForwardIterator first , ForwardIterator last , const T & value ); template < class ForwardIterator , class T = typename iterator_traits < ForwardIterator >:: value_type , class Compare > constexpr pair < ForwardIterator , ForwardIterator > equal_range ( ForwardIterator first , ForwardIterator last , const T & value , Compare comp ); 
namespace ranges { template < forward_iterator I , sentinel_for < I > S , class T , class Proj = identity , class T = typename projected_value < I , Proj >:: value_type indirect_strict_weak_order < const T * , projected < I , Proj >> Comp = ranges :: less > constexpr subrange < I > equal_range ( I first , S last , const T & value , Comp comp = {}, Proj proj = {}); template < forward_range R , class T , class Proj = identity , class T = typename projected_value < iterator_t < R > , Proj >:: value_type indirect_strict_weak_order < const T * , projected < iterator_t < R > , Proj >> Comp = ranges :: less > constexpr borrowed_subrange_t < R > equal_range ( R && r , const T & value , Comp comp = {}, Proj proj = {}); } 
template < class ForwardIterator , class T = typename iterator_traits < ForwardIterator >:: value_type > constexpr bool binary_search ( ForwardIterator first , ForwardIterator last , const T & value ); template < class ForwardIterator , class T = typename iterator_traits < ForwardIterator >:: value_type , class Compare > constexpr bool binary_search ( ForwardIterator first , ForwardIterator last , const T & value , Compare comp ); 
namespace ranges { template < forward_iterator I , sentinel_for < I > S , class T , class Proj = identity , class T = typename projected_value < I , Proj >:: value_type indirect_strict_weak_order < const T * , projected < I , Proj >> Comp = ranges :: less > constexpr bool binary_search ( I first , S last , const T & value , Comp comp = {}, Proj proj = {}); template < forward_range R , class T , class Proj = identity , class T = typename projected_value < iterator_t < R > , Proj >:: value_type indirect_strict_weak_order < const T * , projected < iterator_t < R > , Proj >> Comp = ranges :: less > constexpr bool binary_search ( R && r , const T & value , Comp comp = {}, Proj proj = {}); } 
7.5. Numeric algorithms
Modify both the 
7.5.1. [accumulate]
template < class InputIterator , class T = typename iterator_traits < InputIterator >:: value_type > constexpr T accumulate ( InputIterator first , InputIterator last , T init ); template < class InputIterator , class T = typename iterator_traits < InputIterator >:: value_type , class BinaryOperation > constexpr T accumulate ( InputIterator first , InputIterator last , T init , BinaryOperation binary_op ); 
7.5.2. [reduce]
template < class InputIterator > constexpr typename iterator_traits < InputIterator >:: value_type reduce ( InputIterator first , InputIterator last ); template < class InputIterator , class T = typename iterator_traits < InputIterator >:: value_type > constexpr T reduce ( InputIterator first , InputIterator last , T init ); template < class InputIterator , class T = typename iterator_traits < InputIterator >:: value_type , class BinaryOperation > constexpr T reduce ( InputIterator first , InputIterator last , T init , BinaryOperation binary_op ); template < class ExecutionPolicy , class ForwardIterator > typename iterator_traits < ForwardIterator >:: value_type reduce ( ExecutionPolicy && exec , ForwardIterator first , ForwardIterator last ); template < class ExecutionPolicy , class ForwardIterator , class T = typename iterator_traits < ForwardIterator >:: value_type > T reduce ( ExecutionPolicy && exec , ForwardIterator first , ForwardIterator last , T init ); template < class ExecutionPolicy , class ForwardIterator , class T = typename iterator_traits < ForwardIterator >:: value_type , class BinaryOperation > T reduce ( ExecutionPolicy && exec , ForwardIterator first , ForwardIterator last , T init , BinaryOperation binary_op ); 
7.5.3. [exclusive.scan]
template < class InputIterator , class OutputIterator , class T = typename iterator_traits < InputIterator >:: value_type > constexpr OutputIterator exclusive_scan ( InputIterator first , InputIterator last , OutputIterator result , T init ); template < class InputIterator , class OutputIterator , class T = typename iterator_traits < InputIterator >:: value_type , class BinaryOperation > constexpr OutputIterator exclusive_scan ( InputIterator first , InputIterator last , OutputIterator result , T init , BinaryOperation binary_op ); template < class ExecutionPolicy , class ForwardIterator1 , class ForwardIterator2 , class T = typename iterator_traits < ForwardIterator1 >:: value_type > ForwardIterator2 exclusive_scan ( ExecutionPolicy && exec , ForwardIterator1 first , ForwardIterator1 last , ForwardIterator2 result , T init ); template < class ExecutionPolicy , class ForwardIterator1 , class ForwardIterator2 , class T = typename iterator_traits < ForwardIterator1 >:: value_type , class BinaryOperation > ForwardIterator2 exclusive_scan ( ExecutionPolicy && exec , ForwardIterator1 first , ForwardIterator1 last , ForwardIterator2 result , T init , BinaryOperation binary_op ); 
7.5.4. [inclusive.scan]
template < class InputIterator , class OutputIterator > constexpr OutputIterator inclusive_scan ( InputIterator first , InputIterator last , OutputIterator result ); template < class InputIterator , class OutputIterator , class BinaryOperation > constexpr OutputIterator inclusive_scan ( InputIterator first , InputIterator last , OutputIterator result , BinaryOperation binary_op ); template < class InputIterator , class OutputIterator , class BinaryOperation , class T = typename iterator_traits < InputIterator >:: value_type > constexpr OutputIterator inclusive_scan ( InputIterator first , InputIterator last , OutputIterator result , BinaryOperation binary_op , T init ); template < class ExecutionPolicy , class ForwardIterator1 , class ForwardIterator2 > ForwardIterator2 inclusive_scan ( ExecutionPolicy && exec , ForwardIterator1 first , ForwardIterator1 last , ForwardIterator2 result ); template < class ExecutionPolicy , class ForwardIterator1 , class ForwardIterator2 , class BinaryOperation > ForwardIterator2 inclusive_scan ( ExecutionPolicy && exec , ForwardIterator1 first , ForwardIterator1 last , ForwardIterator2 result , BinaryOperation binary_op ); template < class ExecutionPolicy , class ForwardIterator1 , class ForwardIterator2 , class BinaryOperation , class T = typename iterator_traits < ForwardIterator1 >:: value_type > ForwardIterator2 inclusive_scan ( ExecutionPolicy && exec , ForwardIterator1 first , ForwardIterator1 last , ForwardIterator2 result , BinaryOperation binary_op , T init ); 
7.5.5. [transform.exclusive.scan]
template < class InputIterator , class OutputIterator , class T = typename iterator_traits < InputIterator >:: value_type , class BinaryOperation , class UnaryOperation > constexpr OutputIterator transform_exclusive_scan ( InputIterator first , InputIterator last , OutputIterator result , T init , BinaryOperation binary_op , UnaryOperation unary_op ); template < class ExecutionPolicy , class ForwardIterator1 , class ForwardIterator2 , class T = typename iterator_traits < ForwardIterator1 >:: value_type , class BinaryOperation , class UnaryOperation > ForwardIterator2 transform_exclusive_scan ( ExecutionPolicy && exec , ForwardIterator1 first , ForwardIterator1 last , ForwardIterator2 result , T init , BinaryOperation binary_op , UnaryOperation unary_op ); 
7.5.6. [transform.inclusive.scan]
template < class InputIterator , class OutputIterator , class BinaryOperation , class UnaryOperation > constexpr OutputIterator transform_inclusive_scan ( InputIterator first , InputIterator last , OutputIterator result , BinaryOperation binary_op , UnaryOperation unary_op ); template < class InputIterator , class OutputIterator , class BinaryOperation , class UnaryOperation , class T = typename iterator_traits < InputIterator >:: value_type > constexpr OutputIterator transform_inclusive_scan ( InputIterator first , InputIterator last , OutputIterator result , BinaryOperation binary_op , UnaryOperation unary_op , T init ); template < class ExecutionPolicy , class ForwardIterator1 , class ForwardIterator2 , class BinaryOperation , class UnaryOperation > ForwardIterator2 transform_inclusive_scan ( ExecutionPolicy && exec , ForwardIterator1 first , ForwardIterator1 last , ForwardIterator2 result , BinaryOperation binary_op , UnaryOperation unary_op ); template < class ExecutionPolicy , class ForwardIterator1 , class ForwardIterator2 , class BinaryOperation , class UnaryOperation , class T = typename iterator_traits < ForwardIterator1 >:: value_type > ForwardIterator2 transform_inclusive_scan ( ExecutionPolicy && exec , ForwardIterator1 first , ForwardIterator1 last , ForwardIterator2 result , BinaryOperation binary_op , UnaryOperation unary_op , T init ); 
7.5.7. [numeric.iota]
template < class ForwardIterator , class T = typename iterator_traits < ForwardIterator >:: value_type > constexpr void iota ( ForwardIterator first , ForwardIterator last , T value ); 
8. Acknowledgements
Thanks to KDAB for supporting this work.
Special thanks to Stephan T. Lavavej for confirming that the omission
of a default argument from the 
All remaining errors are ours and ours only.