1. Changelog
- 
     R0 - 
       First submission. 
 
- 
       
2. Motivation and Scope
The Standard is currently lacking support for concatenating strings and string views by means of operator+ :
std :: string calculate ( std :: string_view prefix ) { return prefix + get_string (); // ERROR } 
This constitutes a major asymmetry when considering the rest of 
In general, this makes the concatenation APIs between string and string views have a poor usability experience:
std :: string str ; std :: string view ; // Appending str + view ; // ERROR str + std :: string ( view ); // OK, but inefficient str + view . data (); // Compiles, but BUG! std :: string copy = str ; copy += view ; // OK, but tedious to write (requires explicit copy) copy . append ( view ); // OK, ditto // Prepending view + str ; // ERROR std :: string copy = str ; str . insert ( 0 , view ); // OK, but tedious and inefficient 
Similarly, the current situation is asymmetric when considering concatenation against raw pointers:
std :: string str ; str + "hello" ; // OK str + "hello" sv ; // ERROR "hello" + str ; // OK "hello" sv + str ; // ERROR 
All of this is just bad ergonomics; the lack of 
Now, as shown above, there are workarounds available either in terms
of named functions (
std :: string calculate ( std :: string_view prefix ) { return std :: string ( prefix ) + get_string (); // inefficient } 
And it may even open the door to subtle bugs:
std :: string result1 = str + view ; // ERROR. <Sigh>, ok, let me rewrite as... std :: string result2 = str + view . data (); // Now compiles; but not semantically equivalent 
The last line behaves differently in case 
This paper proposes to fix these API flaws by adding suitable 
2.1. Why are those overloads missing in the first place?
[N3685] ("
I also omitted
because LLVM returns a lightweight object from this overload and only performs the concatenation lazily. If we define this overload, we’ll have a hard time introducing that lightweight concatenation later.operator + ( basic_string , basic_string_view ) 
Subsequent revisions of the paper no longer have this paragraph.
There is a couple of considerations that we think are important here.
- 
     string_view operator + 
- 
     We strongly feel that overloading operator + str + "hello" sv str + "hello" strA + strB + strC operator + operator % operator + operator % 
In short: we do not see any reason to further withhold the proposed additions.
3. Impact On The Standard
This proposal is a pure library extension.
This proposal does not depend on any other library extensions.
This proposal does not require any changes in the core language.
4. Design Decisions
The proposed wording builds on top / reuses of the existing one for 
The proposed overloads are constrained in the same way as the other
string concatenation APIs (e.g. 
5. Implementation experience
A working prototype of the changes proposed by this paper, done on top of GCC 12.1, is available in this GCC branch on GitHub.
6. Technical Specifications
All the proposed changes are relative to [N4910].
6.1. Feature testing macro
In [version.syn], modify
#define __cpp_lib_string_view 201803L YYYYMML // also in <string>, <string_view> 
with the value specified as usual (year and month of adoption of the present proposal).
6.2. Proposed wording
Modify [string.syn] as shown:
[...] // 23.4.3, basic_string [...] template < class charT , class traits , class Allocator > constexpr basic_string < charT , traits , Allocator > operator + ( charT lhs , const basic_string < charT , traits , Allocator >& rhs ); template < class charT , class traits , class Allocator > constexpr basic_string < charT , traits , Allocator > operator + ( charT lhs , basic_string < charT , traits , Allocator >&& rhs ); template < class T , class charT , class traits , class Allocator > constexpr basic_string < charT , traits , Allocator > operator + ( const T & lhs , const basic_string < charT , traits , Allocator >& rhs ); template < class T , class charT , class traits , class Allocator > constexpr basic_string < charT , traits , Allocator > operator + ( const T & lhs , basic_string < charT , traits , Allocator >&& rhs ); [...] template < class charT , class traits , class Allocator > constexpr basic_string < charT , traits , Allocator > operator + ( const basic_string < charT , traits , Allocator >& lhs , charT rhs ); template < class charT , class traits , class Allocator > constexpr basic_string < charT , traits , Allocator > operator + ( basic_string < charT , traits , Allocator >&& lhs , charT rhs ); template < class charT , class traits , class Allocator , class T > constexpr basic_string < charT , traits , Allocator > operator + ( const basic_string < charT , traits , Allocator >& lhs , const T & rhs ); template < class charT , class traits , class Allocator > constexpr basic_string < charT , traits , Allocator > operator + ( basic_string < charT , traits , Allocator >&& lhs , const T & rhs ); 
Append the following at the end of [string.op.plus]:
template < class charT , class traits , class Allocator , class T > constexpr basic_string < charT , traits , Allocator > operator + ( const basic_string < charT , traits , Allocator >& lhs , const T & rhs ); ??? Constraints:
isis_convertible_v < const T & , basic_string_view < charT , traits >> trueand
isis_convertible_v < const T & , const charT *> false.??? Effects: Equivalent to:
basic_string < charT , traits , Allocator > r = lhs ; r . append ( rhs ); return r ; 
template < class charT , class traits , class Allocator , class T > constexpr basic_string < charT , traits , Allocator > operator + ( basic_string < charT , traits , Allocator >&& lhs , const T & rhs ); ??? Constraints:
isis_convertible_v < const T & , basic_string_view < charT , traits >> trueand
isis_convertible_v < const T & , const charT *> false.??? Effects: Equivalent to:
lhs . append ( rhs ); return std :: move ( lhs ); 
template < class T , class charT , class traits , class Allocator > constexpr basic_string < charT , traits , Allocator > operator + ( const T & lhs , const basic_string < charT , traits , Allocator >& rhs ); ??? Constraints:
isis_convertible_v < const T & , basic_string_view < charT , traits >> trueand
isis_convertible_v < const T & , const charT *> false.??? Effects: Equivalent to:
basic_string < charT , traits , Allocator > r = rhs ; r . insert ( 0 , lhs ); return r ; 
template < class T , class charT , class traits , class Allocator > constexpr basic_string < charT , traits , Allocator > operator + ( const T & lhs , basic_string < charT , traits , Allocator >&& rhs ); ??? Constraints:
isis_convertible_v < const T & , basic_string_view < charT , traits >> trueand
isis_convertible_v < const T & , const charT *> false.??? Effects: Equivalent to:
rhs . insert ( 0 , lhs ); return std :: move ( rhs ); 
7. Acknowledgements
Thanks to KDAB for supporting this work.
All remaining errors are ours and ours only.