11 #ifndef NONSTD_SV_LITE_H_INCLUDED
12 #define NONSTD_SV_LITE_H_INCLUDED
14 #define string_view_lite_MAJOR 1
15 #define string_view_lite_MINOR 4
16 #define string_view_lite_PATCH 0
18 #define string_view_lite_VERSION nssv_STRINGIFY(string_view_lite_MAJOR) "." nssv_STRINGIFY(string_view_lite_MINOR) "." nssv_STRINGIFY(string_view_lite_PATCH)
20 #define nssv_STRINGIFY( x ) nssv_STRINGIFY_( x )
21 #define nssv_STRINGIFY_( x ) #x
25 #define nssv_STRING_VIEW_DEFAULT 0
26 #define nssv_STRING_VIEW_NONSTD 1
27 #define nssv_STRING_VIEW_STD 2
30 #define nssv_CONFIG_SELECT_STRING_VIEW nssv_STRING_VIEW_NONSTD
36 #if defined( nssv_CONFIG_SELECT_STD_STRING_VIEW ) || defined( nssv_CONFIG_SELECT_NONSTD_STRING_VIEW )
37 # error nssv_CONFIG_SELECT_STD_STRING_VIEW and nssv_CONFIG_SELECT_NONSTD_STRING_VIEW are deprecated and removed, please use nssv_CONFIG_SELECT_STRING_VIEW=nssv_STRING_VIEW_...
40 #ifndef nssv_CONFIG_STD_SV_OPERATOR
41 # define nssv_CONFIG_STD_SV_OPERATOR 0
44 #ifndef nssv_CONFIG_USR_SV_OPERATOR
45 # define nssv_CONFIG_USR_SV_OPERATOR 1
48 #ifdef nssv_CONFIG_CONVERSION_STD_STRING
49 # define nssv_CONFIG_CONVERSION_STD_STRING_CLASS_METHODS nssv_CONFIG_CONVERSION_STD_STRING
50 # define nssv_CONFIG_CONVERSION_STD_STRING_FREE_FUNCTIONS nssv_CONFIG_CONVERSION_STD_STRING
53 #ifndef nssv_CONFIG_CONVERSION_STD_STRING_CLASS_METHODS
54 # define nssv_CONFIG_CONVERSION_STD_STRING_CLASS_METHODS 1
57 #ifndef nssv_CONFIG_CONVERSION_STD_STRING_FREE_FUNCTIONS
58 # define nssv_CONFIG_CONVERSION_STD_STRING_FREE_FUNCTIONS 1
63 #ifndef nssv_CONFIG_NO_EXCEPTIONS
64 # if defined(__cpp_exceptions) || defined(__EXCEPTIONS) || defined(_CPPUNWIND)
65 # define nssv_CONFIG_NO_EXCEPTIONS 0
67 # define nssv_CONFIG_NO_EXCEPTIONS 1
74 #ifndef nssv_CPLUSPLUS
75 # if defined(_MSVC_LANG ) && !defined(__clang__)
76 # define nssv_CPLUSPLUS (_MSC_VER == 1900 ? 201103L : _MSVC_LANG )
78 # define nssv_CPLUSPLUS __cplusplus
82 #define nssv_CPP98_OR_GREATER ( nssv_CPLUSPLUS >= 199711L )
83 #define nssv_CPP11_OR_GREATER ( nssv_CPLUSPLUS >= 201103L )
84 #define nssv_CPP11_OR_GREATER_ ( nssv_CPLUSPLUS >= 201103L )
85 #define nssv_CPP14_OR_GREATER ( nssv_CPLUSPLUS >= 201402L )
86 #define nssv_CPP17_OR_GREATER ( nssv_CPLUSPLUS >= 201703L )
87 #define nssv_CPP20_OR_GREATER ( nssv_CPLUSPLUS >= 202000L )
91 #if nssv_CPP17_OR_GREATER && defined(__has_include )
92 # if __has_include( <string_view> )
93 # define nssv_HAVE_STD_STRING_VIEW 1
95 # define nssv_HAVE_STD_STRING_VIEW 0
98 # define nssv_HAVE_STD_STRING_VIEW 0
101 #define nssv_USES_STD_STRING_VIEW ( (nssv_CONFIG_SELECT_STRING_VIEW == nssv_STRING_VIEW_STD) || ((nssv_CONFIG_SELECT_STRING_VIEW == nssv_STRING_VIEW_DEFAULT) && nssv_HAVE_STD_STRING_VIEW) )
103 #define nssv_HAVE_STARTS_WITH ( nssv_CPP20_OR_GREATER || !nssv_USES_STD_STRING_VIEW )
104 #define nssv_HAVE_ENDS_WITH nssv_HAVE_STARTS_WITH
110 #if nssv_USES_STD_STRING_VIEW
112 #include <string_view>
116 #if nssv_CONFIG_CONVERSION_STD_STRING_FREE_FUNCTIONS
120 template<
class CharT,
class Traits,
class Allocator = std::allocator<CharT> >
121 std::basic_string<CharT, Traits, Allocator>
122 to_string( std::basic_string_view<CharT, Traits> v, Allocator
const & a = Allocator() )
124 return std::basic_string<CharT,Traits, Allocator>( v.begin(), v.end(), a );
127 template<
class CharT,
class Traits,
class Allocator >
128 std::basic_string_view<CharT, Traits>
129 to_string_view( std::basic_string<CharT, Traits, Allocator>
const & s )
131 return std::basic_string_view<CharT, Traits>( s.data(), s.size() );
136 #if nssv_CONFIG_STD_SV_OPERATOR
138 using namespace std::literals::string_view_literals;
142 #if nssv_CONFIG_USR_SV_OPERATOR
144 inline namespace literals {
145 inline namespace string_view_literals {
148 constexpr std::string_view
operator "" _sv(
const char* str,
size_t len ) noexcept
150 return std::string_view{ str, len };
153 constexpr std::u16string_view
operator "" _sv(
const char16_t* str,
size_t len ) noexcept
155 return std::u16string_view{ str, len };
158 constexpr std::u32string_view
operator "" _sv(
const char32_t* str,
size_t len ) noexcept
160 return std::u32string_view{ str, len };
163 constexpr std::wstring_view
operator "" _sv(
const wchar_t* str,
size_t len ) noexcept
165 return std::wstring_view{ str, len };
170 #endif // nssv_CONFIG_USR_SV_OPERATOR
174 #endif // nssv_CONFIG_CONVERSION_STD_STRING_FREE_FUNCTIONS
178 using std::string_view;
179 using std::wstring_view;
180 using std::u16string_view;
181 using std::u32string_view;
182 using std::basic_string_view;
186 using std::operator==;
187 using std::operator!=;
188 using std::operator<;
189 using std::operator<=;
190 using std::operator>;
191 using std::operator>=;
193 using std::operator<<;
197 #else // nssv_HAVE_STD_STRING_VIEW
217 #if defined(_MSC_VER ) && !defined(__clang__)
218 # define nssv_COMPILER_MSVC_VER (_MSC_VER )
219 # define nssv_COMPILER_MSVC_VERSION (_MSC_VER / 10 - 10 * ( 5 + (_MSC_VER < 1900 ) ) )
221 # define nssv_COMPILER_MSVC_VER 0
222 # define nssv_COMPILER_MSVC_VERSION 0
225 #define nssv_COMPILER_VERSION( major, minor, patch ) ( 10 * ( 10 * (major) + (minor) ) + (patch) )
227 #if defined(__clang__)
228 # define nssv_COMPILER_CLANG_VERSION nssv_COMPILER_VERSION(__clang_major__, __clang_minor__, __clang_patchlevel__)
230 # define nssv_COMPILER_CLANG_VERSION 0
233 #if defined(__GNUC__) && !defined(__clang__)
234 # define nssv_COMPILER_GNUC_VERSION nssv_COMPILER_VERSION(__GNUC__, __GNUC_MINOR__, __GNUC_PATCHLEVEL__)
236 # define nssv_COMPILER_GNUC_VERSION 0
240 #define nssv_BETWEEN( v, lo, hi ) ( (lo) <= (v) && (v) < (hi) )
245 # define nssv_HAS_CPP0X _HAS_CPP0X
247 # define nssv_HAS_CPP0X 0
252 #if nssv_COMPILER_MSVC_VER >= 1900
253 # undef nssv_CPP11_OR_GREATER
254 # define nssv_CPP11_OR_GREATER 1
257 #define nssv_CPP11_90 (nssv_CPP11_OR_GREATER_ || nssv_COMPILER_MSVC_VER >= 1500)
258 #define nssv_CPP11_100 (nssv_CPP11_OR_GREATER_ || nssv_COMPILER_MSVC_VER >= 1600)
259 #define nssv_CPP11_110 (nssv_CPP11_OR_GREATER_ || nssv_COMPILER_MSVC_VER >= 1700)
260 #define nssv_CPP11_120 (nssv_CPP11_OR_GREATER_ || nssv_COMPILER_MSVC_VER >= 1800)
261 #define nssv_CPP11_140 (nssv_CPP11_OR_GREATER_ || nssv_COMPILER_MSVC_VER >= 1900)
262 #define nssv_CPP11_141 (nssv_CPP11_OR_GREATER_ || nssv_COMPILER_MSVC_VER >= 1910)
264 #define nssv_CPP14_000 (nssv_CPP14_OR_GREATER)
265 #define nssv_CPP17_000 (nssv_CPP17_OR_GREATER)
269 #define nssv_HAVE_CONSTEXPR_11 nssv_CPP11_140
270 #define nssv_HAVE_EXPLICIT_CONVERSION nssv_CPP11_140
271 #define nssv_HAVE_INLINE_NAMESPACE nssv_CPP11_140
272 #define nssv_HAVE_NOEXCEPT nssv_CPP11_140
273 #define nssv_HAVE_NULLPTR nssv_CPP11_100
274 #define nssv_HAVE_REF_QUALIFIER nssv_CPP11_140
275 #define nssv_HAVE_UNICODE_LITERALS nssv_CPP11_140
276 #define nssv_HAVE_USER_DEFINED_LITERALS nssv_CPP11_140
277 #define nssv_HAVE_WCHAR16_T nssv_CPP11_100
278 #define nssv_HAVE_WCHAR32_T nssv_CPP11_100
280 #if ! ( ( nssv_CPP11_OR_GREATER && nssv_COMPILER_CLANG_VERSION ) || nssv_BETWEEN( nssv_COMPILER_CLANG_VERSION, 300, 400 ) )
281 # define nssv_HAVE_STD_DEFINED_LITERALS nssv_CPP11_140
283 # define nssv_HAVE_STD_DEFINED_LITERALS 0
288 #define nssv_HAVE_CONSTEXPR_14 nssv_CPP14_000
292 #define nssv_HAVE_NODISCARD nssv_CPP17_000
296 #define nssv_HAVE_STD_HASH nssv_CPP11_120
300 #if nssv_HAVE_CONSTEXPR_11
301 # define nssv_constexpr constexpr
303 # define nssv_constexpr
306 #if nssv_HAVE_CONSTEXPR_14
307 # define nssv_constexpr14 constexpr
309 # define nssv_constexpr14
312 #if nssv_HAVE_EXPLICIT_CONVERSION
313 # define nssv_explicit explicit
315 # define nssv_explicit
318 #if nssv_HAVE_INLINE_NAMESPACE
319 # define nssv_inline_ns inline
321 # define nssv_inline_ns
324 #if nssv_HAVE_NOEXCEPT
325 # define nssv_noexcept noexcept
327 # define nssv_noexcept
338 #if nssv_HAVE_NULLPTR
339 # define nssv_nullptr nullptr
341 # define nssv_nullptr NULL
344 #if nssv_HAVE_NODISCARD
345 # define nssv_nodiscard [[nodiscard]]
347 # define nssv_nodiscard
359 #if ! nssv_CONFIG_NO_EXCEPTIONS
360 # include <stdexcept>
363 #if nssv_CPP11_OR_GREATER
364 # include <type_traits>
369 #if defined(__clang__)
370 # pragma clang diagnostic ignored "-Wreserved-user-defined-literal"
371 # pragma clang diagnostic push
372 # pragma clang diagnostic ignored "-Wuser-defined-literals"
373 #elif defined(__GNUC__)
374 # pragma GCC diagnostic push
375 # pragma GCC diagnostic ignored "-Wliteral-suffix"
378 #if nssv_COMPILER_MSVC_VERSION >= 140
379 # define nssv_SUPPRESS_MSGSL_WARNING(expr) [[gsl::suppress(expr)]]
380 # define nssv_SUPPRESS_MSVC_WARNING(code, descr) __pragma(warning(suppress: code) )
381 # define nssv_DISABLE_MSVC_WARNINGS(codes) __pragma(warning(push)) __pragma(warning(disable: codes))
383 # define nssv_SUPPRESS_MSGSL_WARNING(expr)
384 # define nssv_SUPPRESS_MSVC_WARNING(code, descr)
385 # define nssv_DISABLE_MSVC_WARNINGS(codes)
388 #if defined(__clang__)
389 # define nssv_RESTORE_WARNINGS() _Pragma("clang diagnostic pop")
390 #elif defined(__GNUC__)
391 # define nssv_RESTORE_WARNINGS() _Pragma("GCC diagnostic pop")
392 #elif nssv_COMPILER_MSVC_VERSION >= 140
393 # define nssv_RESTORE_WARNINGS() __pragma(warning(pop ))
395 # define nssv_RESTORE_WARNINGS()
409 namespace
nonstd {
namespace sv_lite {
411 #if nssv_CPP11_OR_GREATER
415 #if nssv_CPP14_OR_GREATER
417 template<
typename CharT >
418 inline constexpr std::size_t length( CharT * s, std::size_t result = 0 )
421 std::size_t r = result;
422 while ( *v !=
'\0' ) {
429 #else // nssv_CPP14_OR_GREATER
433 template<
typename CharT >
434 inline constexpr std::size_t length( CharT * s, std::size_t result = 0 )
436 return *s ==
'\0' ? result : length( s + 1, result + 1 );
439 #endif // nssv_CPP14_OR_GREATER
443 #endif // nssv_CPP11_OR_GREATER
448 class Traits = std::char_traits<CharT>
450 class basic_string_view;
461 class basic_string_view
466 typedef Traits traits_type;
467 typedef CharT value_type;
469 typedef CharT * pointer;
470 typedef CharT
const * const_pointer;
471 typedef CharT & reference;
472 typedef CharT
const & const_reference;
474 typedef const_pointer iterator;
475 typedef const_pointer const_iterator;
476 typedef std::reverse_iterator< const_iterator > reverse_iterator;
477 typedef std::reverse_iterator< const_iterator > const_reverse_iterator;
479 typedef std::size_t size_type;
480 typedef std::ptrdiff_t difference_type;
489 #if nssv_CPP11_OR_GREATER
493 : data_( other.data_)
494 , size_( other.size_)
505 #if nssv_CPP17_OR_GREATER
506 , size_( Traits::length(s) )
507 #elif nssv_CPP11_OR_GREATER
508 , size_( detail::length(s) )
510 , size_( Traits::length(s) )
516 #if nssv_CPP11_OR_GREATER
557 return data_at( pos );
562 #if nssv_CONFIG_NO_EXCEPTIONS
563 assert( pos < size() );
567 throw std::out_of_range(
"nonstd::string_view::at()");
570 return data_at( pos );
573 nssv_constexpr const_reference front()
const {
return data_at( 0 ); }
574 nssv_constexpr const_reference back()
const {
return data_at( size() - 1 ); }
582 assert( n <= size() );
589 assert( n <= size() );
596 swap( data_, other.data_ );
597 swap( size_, other.size_ );
602 size_type copy( CharT * dest, size_type n, size_type pos = 0 )
const
604 #if nssv_CONFIG_NO_EXCEPTIONS
605 assert( pos <= size() );
609 throw std::out_of_range(
"nonstd::string_view::copy()");
612 const size_type rlen = (std::min)( n, size() - pos );
614 (void) Traits::copy( dest,
data() + pos, rlen );
619 nssv_constexpr14 basic_string_view substr( size_type pos = 0, size_type n = npos )
const
621 #if nssv_CONFIG_NO_EXCEPTIONS
622 assert( pos <= size() );
626 throw std::out_of_range(
"nonstd::string_view::substr()");
629 return basic_string_view(
data() + pos, (std::min)( n, size() - pos ) );
636 if (
const int result = Traits::compare(
data(), other.data(), (std::min)( size(), other.size() ) ) )
641 return size() == other.size() ? 0 : size() < other.size() ? -1 : 1;
644 nssv_constexpr int compare( size_type pos1, size_type n1, basic_string_view other )
const
646 return substr( pos1, n1 ).compare( other );
649 nssv_constexpr int compare( size_type pos1, size_type n1, basic_string_view other, size_type pos2, size_type n2 )
const
651 return substr( pos1, n1 ).compare( other.substr( pos2, n2 ) );
656 return compare( basic_string_view( s ) );
659 nssv_constexpr int compare( size_type pos1, size_type n1, CharT
const * s )
const
661 return substr( pos1, n1 ).compare( basic_string_view( s ) );
664 nssv_constexpr int compare( size_type pos1, size_type n1, CharT
const * s, size_type n2 )
const
666 return substr( pos1, n1 ).compare( basic_string_view( s, n2 ) );
675 return size() >= v.size() && compare( 0, v.size(), v ) == 0;
680 return starts_with( basic_string_view( &c, 1 ) );
685 return starts_with( basic_string_view( s ) );
692 return size() >= v.size() && compare( size() - v.size(), npos, v ) == 0;
697 return ends_with( basic_string_view( &c, 1 ) );
702 return ends_with( basic_string_view( s ) );
709 return assert( v.size() == 0 || v.data() !=
nssv_nullptr )
712 : to_pos( std::search( cbegin() + pos, cend(), v.cbegin(), v.cend(), Traits::eq ) );
717 return find( basic_string_view( &c, 1 ), pos );
720 nssv_constexpr14 size_type find( CharT
const * s, size_type pos, size_type n )
const
722 return find( basic_string_view( s, n ), pos );
727 return find( basic_string_view( s ), pos );
734 if ( size() < v.size() )
741 return (std::min)( size(), pos );
744 const_iterator last = cbegin() + (std::min)( size() - v.size(), pos ) + v.size();
745 const_iterator result = std::find_end( cbegin(), last, v.cbegin(), v.cend(), Traits::eq );
747 return result != last ? size_type( result - cbegin() ) : npos;
752 return rfind( basic_string_view( &c, 1 ), pos );
755 nssv_constexpr14 size_type rfind( CharT
const * s, size_type pos, size_type n )
const
757 return rfind( basic_string_view( s, n ), pos );
760 nssv_constexpr14 size_type rfind( CharT
const * s, size_type pos = npos )
const
762 return rfind( basic_string_view( s ), pos );
771 : to_pos( std::find_first_of( cbegin() + pos, cend(), v.cbegin(), v.cend(), Traits::eq ) );
776 return find_first_of( basic_string_view( &c, 1 ), pos );
779 nssv_constexpr size_type find_first_of( CharT
const * s, size_type pos, size_type n )
const
781 return find_first_of( basic_string_view( s, n ), pos );
784 nssv_constexpr size_type find_first_of( CharT
const * s, size_type pos = 0 )
const
786 return find_first_of( basic_string_view( s ), pos );
796 ? find_last_of( v, size() - 1 )
797 : to_pos( std::find_first_of( const_reverse_iterator( cbegin() + pos + 1 ), crend(), v.cbegin(), v.cend(), Traits::eq ) );
802 return find_last_of( basic_string_view( &c, 1 ), pos );
805 nssv_constexpr size_type find_last_of( CharT
const * s, size_type pos, size_type
count )
const
807 return find_last_of( basic_string_view( s,
count ), pos );
810 nssv_constexpr size_type find_last_of( CharT
const * s, size_type pos = npos )
const
812 return find_last_of( basic_string_view( s ), pos );
821 : to_pos( std::find_if( cbegin() + pos, cend(), not_in_view( v ) ) );
826 return find_first_not_of( basic_string_view( &c, 1 ), pos );
829 nssv_constexpr size_type find_first_not_of( CharT
const * s, size_type pos, size_type
count )
const
831 return find_first_not_of( basic_string_view( s,
count ), pos );
834 nssv_constexpr size_type find_first_not_of( CharT
const * s, size_type pos = 0 )
const
836 return find_first_not_of( basic_string_view( s ), pos );
846 ? find_last_not_of( v, size() - 1 )
847 : to_pos( std::find_if( const_reverse_iterator( cbegin() + pos + 1 ), crend(), not_in_view( v ) ) );
852 return find_last_not_of( basic_string_view( &c, 1 ), pos );
855 nssv_constexpr size_type find_last_not_of( CharT
const * s, size_type pos, size_type
count )
const
857 return find_last_not_of( basic_string_view( s,
count ), pos );
860 nssv_constexpr size_type find_last_not_of( CharT
const * s, size_type pos = npos )
const
862 return find_last_not_of( basic_string_view( s ), pos );
867 #if nssv_CPP17_OR_GREATER
869 #elif nssv_CPP11_OR_GREATER
870 enum : size_type { npos = size_type(-1) };
872 enum { npos = size_type(-1) };
878 const basic_string_view v;
880 nssv_constexpr explicit not_in_view( basic_string_view view ) : v( view ) {}
884 return npos == v.find_first_of( c );
890 return it == cend() ? npos : size_type( it - cbegin() );
893 nssv_constexpr size_type to_pos( const_reverse_iterator it )
const
895 return it == crend() ? npos : size_type( crend() - it - 1 );
900 #if nssv_BETWEEN( nssv_COMPILER_GNUC_VERSION, 1, 500 )
903 return assert( pos < size() ), data_[pos];
912 #if nssv_CONFIG_CONVERSION_STD_STRING_CLASS_METHODS
914 template<
class Allocator >
915 basic_string_view( std::basic_string<CharT, Traits, Allocator>
const & s )
nssv_noexcept
920 #if nssv_HAVE_EXPLICIT_CONVERSION
922 template<
class Allocator >
923 explicit operator std::basic_string<CharT, Traits, Allocator>()
const
928 #endif // nssv_HAVE_EXPLICIT_CONVERSION
930 #if nssv_CPP11_OR_GREATER
932 template<
class Allocator = std::allocator<CharT> >
933 std::basic_string<CharT, Traits, Allocator>
934 to_string( Allocator
const & a = Allocator() )
const
936 return std::basic_string<CharT, Traits, Allocator>( begin(), end(), a );
941 std::basic_string<CharT, Traits>
944 return std::basic_string<CharT, Traits>( begin(), end() );
947 template<
class Allocator >
948 std::basic_string<CharT, Traits, Allocator>
951 return std::basic_string<CharT, Traits, Allocator>( begin(), end(), a );
954 #endif // nssv_CPP11_OR_GREATER
956 #endif // nssv_CONFIG_CONVERSION_STD_STRING_CLASS_METHODS
966 template<
class CharT,
class Traits >
968 basic_string_view <CharT, Traits> lhs,
970 {
return lhs.compare( rhs ) == 0 ; }
972 template<
class CharT,
class Traits >
974 basic_string_view <CharT, Traits> lhs,
976 {
return lhs.compare( rhs ) != 0 ; }
978 template<
class CharT,
class Traits >
980 basic_string_view <CharT, Traits> lhs,
982 {
return lhs.compare( rhs ) < 0 ; }
984 template<
class CharT,
class Traits >
986 basic_string_view <CharT, Traits> lhs,
988 {
return lhs.compare( rhs ) <= 0 ; }
990 template<
class CharT,
class Traits >
992 basic_string_view <CharT, Traits> lhs,
994 {
return lhs.compare( rhs ) > 0 ; }
996 template<
class CharT,
class Traits >
998 basic_string_view <CharT, Traits> lhs,
1000 {
return lhs.compare( rhs ) >= 0 ; }
1007 #if ! nssv_CPP11_OR_GREATER || nssv_BETWEEN( nssv_COMPILER_MSVC_VERSION, 100, 141 )
1013 template<
class CharT,
class Traits>
1015 basic_string_view<CharT, Traits> lhs,
1017 {
return lhs.compare( rhs ) == 0; }
1019 template<
class CharT,
class Traits>
1023 {
return rhs.compare( lhs ) == 0; }
1025 template<
class CharT,
class Traits>
1027 basic_string_view<CharT, Traits> lhs,
1029 {
return lhs.size() == rhs.size() && lhs.compare( rhs ) == 0; }
1031 template<
class CharT,
class Traits>
1033 std::basic_string<CharT, Traits> rhs,
1035 {
return lhs.size() == rhs.size() && lhs.compare( rhs ) == 0; }
1039 template<
class CharT,
class Traits>
1041 basic_string_view<CharT, Traits> lhs,
1043 {
return lhs.compare( rhs ) != 0; }
1045 template<
class CharT,
class Traits>
1049 {
return rhs.compare( lhs ) != 0; }
1051 template<
class CharT,
class Traits>
1053 basic_string_view<CharT, Traits> lhs,
1055 {
return lhs.size() != rhs.size() && lhs.compare( rhs ) != 0; }
1057 template<
class CharT,
class Traits>
1059 std::basic_string<CharT, Traits> rhs,
1061 {
return lhs.size() != rhs.size() || rhs.compare( lhs ) != 0; }
1065 template<
class CharT,
class Traits>
1067 basic_string_view<CharT, Traits> lhs,
1069 {
return lhs.compare( rhs ) < 0; }
1071 template<
class CharT,
class Traits>
1075 {
return rhs.compare( lhs ) > 0; }
1077 template<
class CharT,
class Traits>
1079 basic_string_view<CharT, Traits> lhs,
1081 {
return lhs.compare( rhs ) < 0; }
1083 template<
class CharT,
class Traits>
1085 std::basic_string<CharT, Traits> rhs,
1087 {
return rhs.compare( lhs ) > 0; }
1091 template<
class CharT,
class Traits>
1093 basic_string_view<CharT, Traits> lhs,
1095 {
return lhs.compare( rhs ) <= 0; }
1097 template<
class CharT,
class Traits>
1101 {
return rhs.compare( lhs ) >= 0; }
1103 template<
class CharT,
class Traits>
1105 basic_string_view<CharT, Traits> lhs,
1107 {
return lhs.compare( rhs ) <= 0; }
1109 template<
class CharT,
class Traits>
1111 std::basic_string<CharT, Traits> rhs,
1113 {
return rhs.compare( lhs ) >= 0; }
1117 template<
class CharT,
class Traits>
1119 basic_string_view<CharT, Traits> lhs,
1121 {
return lhs.compare( rhs ) > 0; }
1123 template<
class CharT,
class Traits>
1127 {
return rhs.compare( lhs ) < 0; }
1129 template<
class CharT,
class Traits>
1131 basic_string_view<CharT, Traits> lhs,
1133 {
return lhs.compare( rhs ) > 0; }
1135 template<
class CharT,
class Traits>
1137 std::basic_string<CharT, Traits> rhs,
1139 {
return rhs.compare( lhs ) < 0; }
1143 template<
class CharT,
class Traits>
1145 basic_string_view<CharT, Traits> lhs,
1147 {
return lhs.compare( rhs ) >= 0; }
1149 template<
class CharT,
class Traits>
1153 {
return rhs.compare( lhs ) <= 0; }
1155 template<
class CharT,
class Traits>
1157 basic_string_view<CharT, Traits> lhs,
1159 {
return lhs.compare( rhs ) >= 0; }
1161 template<
class CharT,
class Traits>
1163 std::basic_string<CharT, Traits> rhs,
1165 {
return rhs.compare( lhs ) <= 0; }
1167 #else // newer compilers:
1169 #define nssv_BASIC_STRING_VIEW_I(T,U) typename std::decay< basic_string_view<T,U> >::type
1171 #if nssv_BETWEEN( nssv_COMPILER_MSVC_VERSION, 140, 150 )
1172 # define nssv_MSVC_ORDER(x) , int=x
1174 # define nssv_MSVC_ORDER(x)
1179 template<
class CharT,
class Traits nssv_MSVC_ORDER(1) >
1181 basic_string_view <CharT, Traits> lhs,
1183 {
return lhs.compare( rhs ) == 0; }
1185 template<
class CharT,
class Traits nssv_MSVC_ORDER(2) >
1187 nssv_BASIC_STRING_VIEW_I(CharT, Traits) lhs,
1189 {
return lhs.size() == rhs.size() && lhs.compare( rhs ) == 0; }
1193 template<
class CharT,
class Traits nssv_MSVC_ORDER(1) >
1195 basic_string_view < CharT, Traits > lhs,
1196 nssv_BASIC_STRING_VIEW_I( CharT, Traits ) rhs )
nssv_noexcept
1197 {
return lhs.size() != rhs.size() || lhs.compare( rhs ) != 0 ; }
1199 template<
class CharT,
class Traits nssv_MSVC_ORDER(2) >
1201 nssv_BASIC_STRING_VIEW_I( CharT, Traits ) lhs,
1203 {
return lhs.compare( rhs ) != 0 ; }
1207 template<
class CharT,
class Traits nssv_MSVC_ORDER(1) >
1209 basic_string_view < CharT, Traits > lhs,
1210 nssv_BASIC_STRING_VIEW_I( CharT, Traits ) rhs )
nssv_noexcept
1211 {
return lhs.compare( rhs ) < 0 ; }
1213 template<
class CharT,
class Traits nssv_MSVC_ORDER(2) >
1215 nssv_BASIC_STRING_VIEW_I( CharT, Traits ) lhs,
1217 {
return lhs.compare( rhs ) < 0 ; }
1221 template<
class CharT,
class Traits nssv_MSVC_ORDER(1) >
1223 basic_string_view < CharT, Traits > lhs,
1224 nssv_BASIC_STRING_VIEW_I( CharT, Traits ) rhs )
nssv_noexcept
1225 {
return lhs.compare( rhs ) <= 0 ; }
1227 template<
class CharT,
class Traits nssv_MSVC_ORDER(2) >
1229 nssv_BASIC_STRING_VIEW_I( CharT, Traits ) lhs,
1231 {
return lhs.compare( rhs ) <= 0 ; }
1235 template<
class CharT,
class Traits nssv_MSVC_ORDER(1) >
1237 basic_string_view < CharT, Traits > lhs,
1238 nssv_BASIC_STRING_VIEW_I( CharT, Traits ) rhs )
nssv_noexcept
1239 {
return lhs.compare( rhs ) > 0 ; }
1241 template<
class CharT,
class Traits nssv_MSVC_ORDER(2) >
1243 nssv_BASIC_STRING_VIEW_I( CharT, Traits ) lhs,
1245 {
return lhs.compare( rhs ) > 0 ; }
1249 template<
class CharT,
class Traits nssv_MSVC_ORDER(1) >
1251 basic_string_view < CharT, Traits > lhs,
1252 nssv_BASIC_STRING_VIEW_I( CharT, Traits ) rhs )
nssv_noexcept
1253 {
return lhs.compare( rhs ) >= 0 ; }
1255 template<
class CharT,
class Traits nssv_MSVC_ORDER(2) >
1257 nssv_BASIC_STRING_VIEW_I( CharT, Traits ) lhs,
1259 {
return lhs.compare( rhs ) >= 0 ; }
1261 #undef nssv_MSVC_ORDER
1262 #undef nssv_BASIC_STRING_VIEW_I
1264 #endif // compiler-dependent approach to comparisons
1270 template<
class Stream >
1271 void write_padding( Stream & os, std::streamsize n )
1273 for ( std::streamsize i = 0; i < n; ++i )
1274 os.rdbuf()->sputc( os.fill() );
1277 template<
class Stream,
class View >
1278 Stream & write_to_stream( Stream & os, View
const & sv )
1280 typename Stream::sentry sentry( os );
1285 const std::streamsize length =
static_cast<std::streamsize
>( sv.length() );
1288 const bool pad = ( length < os.width() );
1289 const bool left_pad = pad && ( os.flags() & std::ios_base::adjustfield ) == std::ios_base::right;
1292 write_padding( os, os.width() - length );
1295 os.rdbuf()->sputn( sv.begin(), length );
1297 if ( pad && !left_pad )
1298 write_padding( os, os.width() - length );
1308 template<
class CharT,
class Traits >
1309 std::basic_ostream<CharT, Traits> &
1311 std::basic_ostream<CharT, Traits>& os,
1312 basic_string_view <CharT, Traits> sv )
1314 return detail::write_to_stream( os, sv );
1319 typedef basic_string_view<char> string_view;
1320 typedef basic_string_view<wchar_t> wstring_view;
1321 #if nssv_HAVE_WCHAR16_T
1322 typedef basic_string_view<char16_t> u16string_view;
1323 typedef basic_string_view<char32_t> u32string_view;
1332 #if nssv_HAVE_USER_DEFINED_LITERALS
1338 #if nssv_CONFIG_STD_SV_OPERATOR && nssv_HAVE_STD_DEFINED_LITERALS
1342 return nonstd::sv_lite::string_view{ str, len };
1347 return nonstd::sv_lite::u16string_view{ str, len };
1352 return nonstd::sv_lite::u32string_view{ str, len };
1357 return nonstd::sv_lite::wstring_view{ str, len };
1360 #endif // nssv_CONFIG_STD_SV_OPERATOR && nssv_HAVE_STD_DEFINED_LITERALS
1362 #if nssv_CONFIG_USR_SV_OPERATOR
1366 return nonstd::sv_lite::string_view{ str, len };
1371 return nonstd::sv_lite::u16string_view{ str, len };
1376 return nonstd::sv_lite::u32string_view{ str, len };
1381 return nonstd::sv_lite::wstring_view{ str, len };
1384 #endif // nssv_CONFIG_USR_SV_OPERATOR
1394 #if nssv_CONFIG_CONVERSION_STD_STRING_FREE_FUNCTIONS
1401 #if nssv_CPP11_OR_GREATER && nssv_COMPILER_MSVC_VERSION != 140
1403 template<
class CharT,
class Traits,
class Allocator = std::allocator<CharT> >
1404 std::basic_string<CharT, Traits, Allocator>
1405 to_string( basic_string_view<CharT, Traits> v, Allocator
const & a = Allocator() )
1407 return std::basic_string<CharT,Traits, Allocator>( v.begin(), v.end(), a );
1412 template<
class CharT,
class Traits >
1413 std::basic_string<CharT, Traits>
1416 return std::basic_string<CharT, Traits>( v.begin(), v.end() );
1419 template<
class CharT,
class Traits,
class Allocator >
1420 std::basic_string<CharT, Traits, Allocator>
1421 to_string( basic_string_view<CharT, Traits> v, Allocator
const & a )
1423 return std::basic_string<CharT, Traits, Allocator>( v.begin(), v.end(), a );
1426 #endif // nssv_CPP11_OR_GREATER
1428 template<
class CharT,
class Traits,
class Allocator >
1429 basic_string_view<CharT, Traits>
1432 return basic_string_view<CharT, Traits>( s.data(), s.size() );
1437 #endif // nssv_CONFIG_CONVERSION_STD_STRING_FREE_FUNCTIONS
1445 using sv_lite::basic_string_view;
1446 using sv_lite::string_view;
1447 using sv_lite::wstring_view;
1449 #if nssv_HAVE_WCHAR16_T
1450 using sv_lite::u16string_view;
1452 #if nssv_HAVE_WCHAR32_T
1453 using sv_lite::u32string_view;
1458 using sv_lite::operator==;
1459 using sv_lite::operator!=;
1460 using sv_lite::operator<;
1461 using sv_lite::operator<=;
1462 using sv_lite::operator>;
1463 using sv_lite::operator>=;
1465 using sv_lite::operator<<;
1467 #if nssv_CONFIG_CONVERSION_STD_STRING_FREE_FUNCTIONS
1479 #if nssv_HAVE_STD_HASH
1481 #include <functional>
1486 struct hash<
nonstd::string_view >
1489 std::size_t operator()( nonstd::string_view v )
const nssv_noexcept
1491 return std::hash<std::string>()( std::string( v.data(), v.size() ) );
1496 struct hash<
nonstd::wstring_view >
1499 std::size_t operator()( nonstd::wstring_view v )
const nssv_noexcept
1501 return std::hash<std::wstring>()( std::wstring( v.data(), v.size() ) );
1506 struct hash<
nonstd::u16string_view >
1509 std::size_t operator()( nonstd::u16string_view v )
const nssv_noexcept
1511 return std::hash<std::u16string>()( std::u16string( v.data(), v.size() ) );
1516 struct hash<
nonstd::u32string_view >
1519 std::size_t operator()( nonstd::u32string_view v )
const nssv_noexcept
1521 return std::hash<std::u32string>()( std::u32string( v.data(), v.size() ) );
1527 #endif // nssv_HAVE_STD_HASH
1531 #endif // nssv_HAVE_STD_STRING_VIEW
1532 #endif // NONSTD_SV_LITE_H_INCLUDED