string_view.hpp 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670
  1. /*
  2. © Copyright (c) Marshall Clow 2012-2015.
  3. © Copyright Beman Dawes 2015
  4. Distributed under the Boost Software License, Version 1.0. (See accompanying
  5. file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  6. For more information, see http://www.boost.org
  7. Based on the StringRef implementation in LLVM (http://llvm.org) and
  8. N3422 by Jeffrey Yasskin
  9. http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2012/n3442.html
  10. Updated July 2015 to reflect the Library Fundamentals TS
  11. http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2015/n4480.html
  12. */
  13. #ifndef BOOST_STRING_VIEW_HPP
  14. #define BOOST_STRING_VIEW_HPP
  15. #include <boost/config.hpp>
  16. #include <boost/detail/workaround.hpp>
  17. #include <boost/utility/string_view_fwd.hpp>
  18. #include <boost/throw_exception.hpp>
  19. #include <cstddef>
  20. #include <stdexcept>
  21. #include <algorithm>
  22. #include <iterator>
  23. #include <string>
  24. #include <cstring>
  25. #include <iosfwd>
  26. namespace boost {
  27. namespace detail {
  28. // A helper functor because sometimes we don't have lambdas
  29. template <typename charT, typename traits>
  30. class string_view_traits_eq {
  31. public:
  32. string_view_traits_eq ( charT ch ) : ch_(ch) {}
  33. bool operator()( charT val ) const { return traits::eq (ch_, val); }
  34. charT ch_;
  35. };
  36. }
  37. template<typename charT, typename traits> // traits defaulted in string_view_fwd.hpp
  38. class basic_string_view {
  39. public:
  40. // types
  41. typedef traits traits_type;
  42. typedef charT value_type;
  43. typedef charT* pointer;
  44. typedef const charT* const_pointer;
  45. typedef charT& reference;
  46. typedef const charT& const_reference;
  47. typedef const_pointer const_iterator; // impl-defined
  48. typedef const_iterator iterator;
  49. typedef std::reverse_iterator<const_iterator> const_reverse_iterator;
  50. typedef const_reverse_iterator reverse_iterator;
  51. typedef std::size_t size_type;
  52. typedef std::ptrdiff_t difference_type;
  53. static BOOST_CONSTEXPR_OR_CONST size_type npos = size_type(-1);
  54. // construct/copy
  55. BOOST_CONSTEXPR basic_string_view() BOOST_NOEXCEPT
  56. : ptr_(NULL), len_(0) {}
  57. BOOST_CONSTEXPR basic_string_view(const basic_string_view &rhs) BOOST_NOEXCEPT
  58. : ptr_(rhs.ptr_), len_(rhs.len_) {}
  59. basic_string_view& operator=(const basic_string_view &rhs) BOOST_NOEXCEPT {
  60. ptr_ = rhs.ptr_;
  61. len_ = rhs.len_;
  62. return *this;
  63. }
  64. template<typename Allocator>
  65. basic_string_view(const std::basic_string<charT, traits,
  66. Allocator>& str) BOOST_NOEXCEPT
  67. : ptr_(str.data()), len_(str.length()) {}
  68. BOOST_CONSTEXPR basic_string_view(const charT* str)
  69. : ptr_(str), len_(traits::length(str)) {}
  70. BOOST_CONSTEXPR basic_string_view(const charT* str, size_type len)
  71. : ptr_(str), len_(len) {}
  72. // iterators
  73. BOOST_CONSTEXPR const_iterator begin() const BOOST_NOEXCEPT { return ptr_; }
  74. BOOST_CONSTEXPR const_iterator cbegin() const BOOST_NOEXCEPT { return ptr_; }
  75. BOOST_CONSTEXPR const_iterator end() const BOOST_NOEXCEPT { return ptr_ + len_; }
  76. BOOST_CONSTEXPR const_iterator cend() const BOOST_NOEXCEPT { return ptr_ + len_; }
  77. const_reverse_iterator rbegin() const BOOST_NOEXCEPT { return const_reverse_iterator(end()); }
  78. const_reverse_iterator crbegin() const BOOST_NOEXCEPT { return const_reverse_iterator(end()); }
  79. const_reverse_iterator rend() const BOOST_NOEXCEPT { return const_reverse_iterator(begin()); }
  80. const_reverse_iterator crend() const BOOST_NOEXCEPT { return const_reverse_iterator(begin()); }
  81. // capacity
  82. BOOST_CONSTEXPR size_type size() const BOOST_NOEXCEPT { return len_; }
  83. BOOST_CONSTEXPR size_type length() const BOOST_NOEXCEPT { return len_; }
  84. BOOST_CONSTEXPR size_type max_size() const BOOST_NOEXCEPT { return len_; }
  85. BOOST_CONSTEXPR bool empty() const BOOST_NOEXCEPT { return len_ == 0; }
  86. // element access
  87. BOOST_CONSTEXPR const_reference operator[](size_type pos) const BOOST_NOEXCEPT { return ptr_[pos]; }
  88. BOOST_CONSTEXPR const_reference at(size_t pos) const {
  89. return pos >= len_ ? BOOST_THROW_EXCEPTION(std::out_of_range("boost::string_view::at")) : ptr_[pos];
  90. // if ( pos >= len_ )
  91. // BOOST_THROW_EXCEPTION( std::out_of_range ( "boost::string_view::at" ) );
  92. // return ptr_[pos];
  93. }
  94. BOOST_CONSTEXPR const_reference front() const { return ptr_[0]; }
  95. BOOST_CONSTEXPR const_reference back() const { return ptr_[len_-1]; }
  96. BOOST_CONSTEXPR const_pointer data() const BOOST_NOEXCEPT { return ptr_; }
  97. // modifiers
  98. void clear() BOOST_NOEXCEPT { len_ = 0; } // Boost extension
  99. BOOST_CXX14_CONSTEXPR void remove_prefix(size_type n) {
  100. if ( n > len_ )
  101. n = len_;
  102. ptr_ += n;
  103. len_ -= n;
  104. }
  105. BOOST_CXX14_CONSTEXPR void remove_suffix(size_type n) {
  106. if ( n > len_ )
  107. n = len_;
  108. len_ -= n;
  109. }
  110. BOOST_CXX14_CONSTEXPR void swap(basic_string_view& s) BOOST_NOEXCEPT {
  111. std::swap(ptr_, s.ptr_);
  112. std::swap(len_, s.len_);
  113. }
  114. // basic_string_view string operations
  115. #ifndef BOOST_NO_CXX11_EXPLICIT_CONVERSION_OPERATORS
  116. template<typename Allocator>
  117. explicit operator std::basic_string<charT, traits, Allocator>() const {
  118. return std::basic_string<charT, traits, Allocator>(begin(), end());
  119. }
  120. #endif
  121. #ifndef BOOST_NO_CXX11_FUNCTION_TEMPLATE_DEFAULT_ARGS
  122. template<typename Allocator = std::allocator<charT> >
  123. std::basic_string<charT, traits> to_string(const Allocator& a = Allocator()) const {
  124. return std::basic_string<charT, traits, Allocator>(begin(), end(), a);
  125. }
  126. #else
  127. std::basic_string<charT, traits> to_string() const {
  128. return std::basic_string<charT, traits>(begin(), end());
  129. }
  130. #endif
  131. size_type copy(charT* s, size_type n, size_type pos=0) const {
  132. if (pos > size())
  133. BOOST_THROW_EXCEPTION(std::out_of_range("string_view::copy" ));
  134. size_type rlen = (std::min)(n, len_ - pos);
  135. // use std::copy(begin() + pos, begin() + pos + rlen, s) rather than
  136. // std::copy_n(begin() + pos, rlen, s) to support pre-C++11 standard libraries
  137. std::copy(begin() + pos, begin() + pos + rlen, s);
  138. return rlen;
  139. }
  140. BOOST_CXX14_CONSTEXPR basic_string_view substr(size_type pos, size_type n=npos) const {
  141. if ( pos > size())
  142. BOOST_THROW_EXCEPTION( std::out_of_range ( "string_view::substr" ) );
  143. if (n == npos || pos + n > size())
  144. n = size () - pos;
  145. return basic_string_view(data() + pos, n);
  146. }
  147. BOOST_CXX14_CONSTEXPR int compare(basic_string_view x) const BOOST_NOEXCEPT {
  148. const int cmp = traits::compare(ptr_, x.ptr_, (std::min)(len_, x.len_));
  149. return cmp != 0 ? cmp : (len_ == x.len_ ? 0 : len_ < x.len_ ? -1 : 1);
  150. }
  151. BOOST_CXX14_CONSTEXPR int compare(size_type pos1, size_type n1, basic_string_view x)
  152. const BOOST_NOEXCEPT {
  153. return substr(pos1, n1).compare(x);
  154. }
  155. BOOST_CXX14_CONSTEXPR int compare(size_type pos1, size_type n1,
  156. basic_string_view x, size_type pos2, size_type n2) const {
  157. return substr(pos1, n1).compare(x.substr(pos2, n2));
  158. }
  159. BOOST_CXX14_CONSTEXPR int compare(const charT* x) const {
  160. return compare(basic_string_view(x));
  161. }
  162. BOOST_CXX14_CONSTEXPR int compare(size_type pos1, size_type n1, const charT* x) const {
  163. return substr(pos1, n1).compare(basic_string_view(x));
  164. }
  165. BOOST_CXX14_CONSTEXPR int compare(size_type pos1, size_type n1,
  166. const charT* x, size_type n2) const {
  167. return substr(pos1, n1).compare(basic_string_view(x, n2));
  168. }
  169. // Searches
  170. BOOST_CONSTEXPR bool starts_with(charT c) const BOOST_NOEXCEPT { // Boost extension
  171. return !empty() && traits::eq(c, front());
  172. }
  173. BOOST_CONSTEXPR bool starts_with(basic_string_view x) const BOOST_NOEXCEPT { // Boost extension
  174. return len_ >= x.len_ && traits::compare(ptr_, x.ptr_, x.len_) == 0;
  175. }
  176. BOOST_CONSTEXPR bool ends_with(charT c) const BOOST_NOEXCEPT { // Boost extension
  177. return !empty() && traits::eq(c, back());
  178. }
  179. BOOST_CONSTEXPR bool ends_with(basic_string_view x) const BOOST_NOEXCEPT { // Boost extension
  180. return len_ >= x.len_ &&
  181. traits::compare(ptr_ + len_ - x.len_, x.ptr_, x.len_) == 0;
  182. }
  183. // find
  184. BOOST_CXX14_CONSTEXPR size_type find(basic_string_view s, size_type pos = 0) const BOOST_NOEXCEPT {
  185. if (pos > size())
  186. return npos;
  187. if (s.empty())
  188. return pos;
  189. const_iterator iter = std::search(this->cbegin() + pos, this->cend(),
  190. s.cbegin (), s.cend (), traits::eq);
  191. return iter == this->cend () ? npos : std::distance(this->cbegin (), iter);
  192. }
  193. BOOST_CXX14_CONSTEXPR size_type find(charT c, size_type pos = 0) const BOOST_NOEXCEPT
  194. { return find(basic_string_view(&c, 1), pos); }
  195. BOOST_CXX14_CONSTEXPR size_type find(const charT* s, size_type pos, size_type n) const BOOST_NOEXCEPT
  196. { return find(basic_string_view(s, n), pos); }
  197. BOOST_CXX14_CONSTEXPR size_type find(const charT* s, size_type pos = 0) const BOOST_NOEXCEPT
  198. { return find(basic_string_view(s), pos); }
  199. // rfind
  200. BOOST_CXX14_CONSTEXPR size_type rfind(basic_string_view s, size_type pos = npos) const BOOST_NOEXCEPT {
  201. if (len_ < s.len_)
  202. return npos;
  203. if (pos > len_ - s.len_)
  204. pos = len_ - s.len_;
  205. if (s.len_ == 0u) // an empty string is always found
  206. return pos;
  207. for (const charT* cur = ptr_ + pos;; --cur) {
  208. if (traits::compare(cur, s.ptr_, s.len_) == 0)
  209. return cur - ptr_;
  210. if (cur == ptr_)
  211. return npos;
  212. };
  213. }
  214. BOOST_CXX14_CONSTEXPR size_type rfind(charT c, size_type pos = npos) const BOOST_NOEXCEPT
  215. { return rfind(basic_string_view(&c, 1), pos); }
  216. BOOST_CXX14_CONSTEXPR size_type rfind(const charT* s, size_type pos, size_type n) const BOOST_NOEXCEPT
  217. { return rfind(basic_string_view(s, n), pos); }
  218. BOOST_CXX14_CONSTEXPR size_type rfind(const charT* s, size_type pos = npos) const BOOST_NOEXCEPT
  219. { return rfind(basic_string_view(s), pos); }
  220. // find_first_of
  221. BOOST_CXX14_CONSTEXPR size_type find_first_of(basic_string_view s, size_type pos = 0) const BOOST_NOEXCEPT {
  222. if (pos >= len_ || s.len_ == 0)
  223. return npos;
  224. const_iterator iter = std::find_first_of
  225. (this->cbegin () + pos, this->cend (), s.cbegin (), s.cend (), traits::eq);
  226. return iter == this->cend () ? npos : std::distance ( this->cbegin (), iter );
  227. }
  228. BOOST_CXX14_CONSTEXPR size_type find_first_of(charT c, size_type pos = 0) const BOOST_NOEXCEPT
  229. { return find_first_of(basic_string_view(&c, 1), pos); }
  230. BOOST_CXX14_CONSTEXPR size_type find_first_of(const charT* s, size_type pos, size_type n) const BOOST_NOEXCEPT
  231. { return find_first_of(basic_string_view(s, n), pos); }
  232. BOOST_CXX14_CONSTEXPR size_type find_first_of(const charT* s, size_type pos = 0) const BOOST_NOEXCEPT
  233. { return find_first_of(basic_string_view(s), pos); }
  234. // find_last_of
  235. BOOST_CXX14_CONSTEXPR size_type find_last_of(basic_string_view s, size_type pos = npos) const BOOST_NOEXCEPT {
  236. if (s.len_ == 0u)
  237. return npos;
  238. if (pos >= len_)
  239. pos = 0;
  240. else
  241. pos = len_ - (pos+1);
  242. const_reverse_iterator iter = std::find_first_of
  243. ( this->crbegin () + pos, this->crend (), s.cbegin (), s.cend (), traits::eq );
  244. return iter == this->crend () ? npos : reverse_distance ( this->crbegin (), iter);
  245. }
  246. BOOST_CXX14_CONSTEXPR size_type find_last_of(charT c, size_type pos = npos) const BOOST_NOEXCEPT
  247. { return find_last_of(basic_string_view(&c, 1), pos); }
  248. BOOST_CXX14_CONSTEXPR size_type find_last_of(const charT* s, size_type pos, size_type n) const BOOST_NOEXCEPT
  249. { return find_last_of(basic_string_view(s, n), pos); }
  250. BOOST_CXX14_CONSTEXPR size_type find_last_of(const charT* s, size_type pos = npos) const BOOST_NOEXCEPT
  251. { return find_last_of(basic_string_view(s), pos); }
  252. // find_first_not_of
  253. BOOST_CXX14_CONSTEXPR size_type find_first_not_of(basic_string_view s, size_type pos = 0) const BOOST_NOEXCEPT {
  254. if (pos >= len_)
  255. return npos;
  256. if (s.len_ == 0)
  257. return pos;
  258. const_iterator iter = find_not_of ( this->cbegin () + pos, this->cend (), s );
  259. return iter == this->cend () ? npos : std::distance ( this->cbegin (), iter );
  260. }
  261. BOOST_CXX14_CONSTEXPR size_type find_first_not_of(charT c, size_type pos = 0) const BOOST_NOEXCEPT
  262. { return find_first_not_of(basic_string_view(&c, 1), pos); }
  263. BOOST_CXX14_CONSTEXPR size_type find_first_not_of(const charT* s, size_type pos, size_type n) const BOOST_NOEXCEPT
  264. { return find_first_not_of(basic_string_view(s, n), pos); }
  265. BOOST_CXX14_CONSTEXPR size_type find_first_not_of(const charT* s, size_type pos = 0) const BOOST_NOEXCEPT
  266. { return find_first_not_of(basic_string_view(s), pos); }
  267. // find_last_not_of
  268. BOOST_CXX14_CONSTEXPR size_type find_last_not_of(basic_string_view s, size_type pos = npos) const BOOST_NOEXCEPT {
  269. if (pos >= len_)
  270. pos = len_ - 1;;
  271. if (s.len_ == 0u)
  272. return pos;
  273. pos = len_ - (pos+1);
  274. const_reverse_iterator iter = find_not_of ( this->crbegin () + pos, this->crend (), s );
  275. return iter == this->crend () ? npos : reverse_distance ( this->crbegin (), iter );
  276. }
  277. BOOST_CXX14_CONSTEXPR size_type find_last_not_of(charT c, size_type pos = npos) const BOOST_NOEXCEPT
  278. { return find_last_not_of(basic_string_view(&c, 1), pos); }
  279. BOOST_CXX14_CONSTEXPR size_type find_last_not_of(const charT* s, size_type pos, size_type n) const BOOST_NOEXCEPT
  280. { return find_last_not_of(basic_string_view(s, n), pos); }
  281. BOOST_CXX14_CONSTEXPR size_type find_last_not_of(const charT* s, size_type pos = npos) const BOOST_NOEXCEPT
  282. { return find_last_not_of(basic_string_view(s), pos); }
  283. private:
  284. template <typename r_iter>
  285. size_type reverse_distance(r_iter first, r_iter last) const BOOST_NOEXCEPT {
  286. // Portability note here: std::distance is not NOEXCEPT, but calling it with a string_view::reverse_iterator will not throw.
  287. return len_ - 1 - std::distance ( first, last );
  288. }
  289. template <typename Iterator>
  290. Iterator find_not_of(Iterator first, Iterator last, basic_string_view s) const BOOST_NOEXCEPT {
  291. for (; first != last ; ++first)
  292. if ( 0 == traits::find(s.ptr_, s.len_, *first))
  293. return first;
  294. return last;
  295. }
  296. const charT *ptr_;
  297. std::size_t len_;
  298. };
  299. // Comparison operators
  300. // Equality
  301. template<typename charT, typename traits>
  302. inline bool operator==(basic_string_view<charT, traits> x,
  303. basic_string_view<charT, traits> y) BOOST_NOEXCEPT {
  304. if (x.size () != y.size ()) return false;
  305. return x.compare(y) == 0;
  306. }
  307. // Inequality
  308. template<typename charT, typename traits>
  309. inline bool operator!=(basic_string_view<charT, traits> x,
  310. basic_string_view<charT, traits> y) BOOST_NOEXCEPT {
  311. if ( x.size () != y.size ()) return true;
  312. return x.compare(y) != 0;
  313. }
  314. // Less than
  315. template<typename charT, typename traits>
  316. inline bool operator<(basic_string_view<charT, traits> x,
  317. basic_string_view<charT, traits> y) BOOST_NOEXCEPT {
  318. return x.compare(y) < 0;
  319. }
  320. // Greater than
  321. template<typename charT, typename traits>
  322. inline bool operator>(basic_string_view<charT, traits> x,
  323. basic_string_view<charT, traits> y) BOOST_NOEXCEPT {
  324. return x.compare(y) > 0;
  325. }
  326. // Less than or equal to
  327. template<typename charT, typename traits>
  328. inline bool operator<=(basic_string_view<charT, traits> x,
  329. basic_string_view<charT, traits> y) BOOST_NOEXCEPT {
  330. return x.compare(y) <= 0;
  331. }
  332. // Greater than or equal to
  333. template<typename charT, typename traits>
  334. inline bool operator>=(basic_string_view<charT, traits> x,
  335. basic_string_view<charT, traits> y) BOOST_NOEXCEPT {
  336. return x.compare(y) >= 0;
  337. }
  338. // "sufficient additional overloads of comparison functions"
  339. template<typename charT, typename traits, typename Allocator>
  340. inline bool operator==(basic_string_view<charT, traits> x,
  341. const std::basic_string<charT, traits, Allocator> & y) BOOST_NOEXCEPT {
  342. return x == basic_string_view<charT, traits>(y);
  343. }
  344. template<typename charT, typename traits, typename Allocator>
  345. inline bool operator==(const std::basic_string<charT, traits, Allocator> & x,
  346. basic_string_view<charT, traits> y) BOOST_NOEXCEPT {
  347. return basic_string_view<charT, traits>(x) == y;
  348. }
  349. template<typename charT, typename traits>
  350. inline bool operator==(basic_string_view<charT, traits> x,
  351. const charT * y) BOOST_NOEXCEPT {
  352. return x == basic_string_view<charT, traits>(y);
  353. }
  354. template<typename charT, typename traits>
  355. inline bool operator==(const charT * x,
  356. basic_string_view<charT, traits> y) BOOST_NOEXCEPT {
  357. return basic_string_view<charT, traits>(x) == y;
  358. }
  359. template<typename charT, typename traits, typename Allocator>
  360. inline bool operator!=(basic_string_view<charT, traits> x,
  361. const std::basic_string<charT, traits, Allocator> & y) BOOST_NOEXCEPT {
  362. return x != basic_string_view<charT, traits>(y);
  363. }
  364. template<typename charT, typename traits, typename Allocator>
  365. inline bool operator!=(const std::basic_string<charT, traits, Allocator> & x,
  366. basic_string_view<charT, traits> y) BOOST_NOEXCEPT {
  367. return basic_string_view<charT, traits>(x) != y;
  368. }
  369. template<typename charT, typename traits>
  370. inline bool operator!=(basic_string_view<charT, traits> x,
  371. const charT * y) BOOST_NOEXCEPT {
  372. return x != basic_string_view<charT, traits>(y);
  373. }
  374. template<typename charT, typename traits>
  375. inline bool operator!=(const charT * x,
  376. basic_string_view<charT, traits> y) BOOST_NOEXCEPT {
  377. return basic_string_view<charT, traits>(x) != y;
  378. }
  379. template<typename charT, typename traits, typename Allocator>
  380. inline bool operator<(basic_string_view<charT, traits> x,
  381. const std::basic_string<charT, traits, Allocator> & y) BOOST_NOEXCEPT {
  382. return x < basic_string_view<charT, traits>(y);
  383. }
  384. template<typename charT, typename traits, typename Allocator>
  385. inline bool operator<(const std::basic_string<charT, traits, Allocator> & x,
  386. basic_string_view<charT, traits> y) BOOST_NOEXCEPT {
  387. return basic_string_view<charT, traits>(x) < y;
  388. }
  389. template<typename charT, typename traits>
  390. inline bool operator<(basic_string_view<charT, traits> x,
  391. const charT * y) BOOST_NOEXCEPT {
  392. return x < basic_string_view<charT, traits>(y);
  393. }
  394. template<typename charT, typename traits>
  395. inline bool operator<(const charT * x,
  396. basic_string_view<charT, traits> y) BOOST_NOEXCEPT {
  397. return basic_string_view<charT, traits>(x) < y;
  398. }
  399. template<typename charT, typename traits, typename Allocator>
  400. inline bool operator>(basic_string_view<charT, traits> x,
  401. const std::basic_string<charT, traits, Allocator> & y) BOOST_NOEXCEPT {
  402. return x > basic_string_view<charT, traits>(y);
  403. }
  404. template<typename charT, typename traits, typename Allocator>
  405. inline bool operator>(const std::basic_string<charT, traits, Allocator> & x,
  406. basic_string_view<charT, traits> y) BOOST_NOEXCEPT {
  407. return basic_string_view<charT, traits>(x) > y;
  408. }
  409. template<typename charT, typename traits>
  410. inline bool operator>(basic_string_view<charT, traits> x,
  411. const charT * y) BOOST_NOEXCEPT {
  412. return x > basic_string_view<charT, traits>(y);
  413. }
  414. template<typename charT, typename traits>
  415. inline bool operator>(const charT * x,
  416. basic_string_view<charT, traits> y) BOOST_NOEXCEPT {
  417. return basic_string_view<charT, traits>(x) > y;
  418. }
  419. template<typename charT, typename traits, typename Allocator>
  420. inline bool operator<=(basic_string_view<charT, traits> x,
  421. const std::basic_string<charT, traits, Allocator> & y) BOOST_NOEXCEPT {
  422. return x <= basic_string_view<charT, traits>(y);
  423. }
  424. template<typename charT, typename traits, typename Allocator>
  425. inline bool operator<=(const std::basic_string<charT, traits, Allocator> & x,
  426. basic_string_view<charT, traits> y) BOOST_NOEXCEPT {
  427. return basic_string_view<charT, traits>(x) <= y;
  428. }
  429. template<typename charT, typename traits>
  430. inline bool operator<=(basic_string_view<charT, traits> x,
  431. const charT * y) BOOST_NOEXCEPT {
  432. return x <= basic_string_view<charT, traits>(y);
  433. }
  434. template<typename charT, typename traits>
  435. inline bool operator<=(const charT * x,
  436. basic_string_view<charT, traits> y) BOOST_NOEXCEPT {
  437. return basic_string_view<charT, traits>(x) <= y;
  438. }
  439. template<typename charT, typename traits, typename Allocator>
  440. inline bool operator>=(basic_string_view<charT, traits> x,
  441. const std::basic_string<charT, traits, Allocator> & y) BOOST_NOEXCEPT {
  442. return x >= basic_string_view<charT, traits>(y);
  443. }
  444. template<typename charT, typename traits, typename Allocator>
  445. inline bool operator>=(const std::basic_string<charT, traits, Allocator> & x,
  446. basic_string_view<charT, traits> y) BOOST_NOEXCEPT {
  447. return basic_string_view<charT, traits>(x) >= y;
  448. }
  449. template<typename charT, typename traits>
  450. inline bool operator>=(basic_string_view<charT, traits> x,
  451. const charT * y) BOOST_NOEXCEPT {
  452. return x >= basic_string_view<charT, traits>(y);
  453. }
  454. template<typename charT, typename traits>
  455. inline bool operator>=(const charT * x,
  456. basic_string_view<charT, traits> y) BOOST_NOEXCEPT {
  457. return basic_string_view<charT, traits>(x) >= y;
  458. }
  459. namespace detail {
  460. template<class charT, class traits>
  461. inline void insert_fill_chars(std::basic_ostream<charT, traits>& os, std::size_t n) {
  462. enum { chunk_size = 8 };
  463. charT fill_chars[chunk_size];
  464. std::fill_n(fill_chars, static_cast< std::size_t >(chunk_size), os.fill());
  465. for (; n >= chunk_size && os.good(); n -= chunk_size)
  466. os.write(fill_chars, static_cast< std::size_t >(chunk_size));
  467. if (n > 0 && os.good())
  468. os.write(fill_chars, n);
  469. }
  470. template<class charT, class traits>
  471. void insert_aligned(std::basic_ostream<charT, traits>& os, const basic_string_view<charT,traits>& str) {
  472. const std::size_t size = str.size();
  473. const std::size_t alignment_size = static_cast< std::size_t >(os.width()) - size;
  474. const bool align_left = (os.flags() & std::basic_ostream<charT, traits>::adjustfield) == std::basic_ostream<charT, traits>::left;
  475. if (!align_left) {
  476. detail::insert_fill_chars(os, alignment_size);
  477. if (os.good())
  478. os.write(str.data(), size);
  479. }
  480. else {
  481. os.write(str.data(), size);
  482. if (os.good())
  483. detail::insert_fill_chars(os, alignment_size);
  484. }
  485. }
  486. } // namespace detail
  487. // Inserter
  488. template<class charT, class traits>
  489. inline std::basic_ostream<charT, traits>&
  490. operator<<(std::basic_ostream<charT, traits>& os,
  491. const basic_string_view<charT,traits>& str) {
  492. if (os.good()) {
  493. const std::size_t size = str.size();
  494. const std::size_t w = static_cast< std::size_t >(os.width());
  495. if (w <= size)
  496. os.write(str.data(), size);
  497. else
  498. detail::insert_aligned(os, str);
  499. os.width(0);
  500. }
  501. return os;
  502. }
  503. #if 0
  504. // numeric conversions
  505. //
  506. // These are short-term implementations.
  507. // In a production environment, I would rather avoid the copying.
  508. //
  509. inline int stoi (string_view str, size_t* idx=0, int base=10) {
  510. return std::stoi ( std::string(str), idx, base );
  511. }
  512. inline long stol (string_view str, size_t* idx=0, int base=10) {
  513. return std::stol ( std::string(str), idx, base );
  514. }
  515. inline unsigned long stoul (string_view str, size_t* idx=0, int base=10) {
  516. return std::stoul ( std::string(str), idx, base );
  517. }
  518. inline long long stoll (string_view str, size_t* idx=0, int base=10) {
  519. return std::stoll ( std::string(str), idx, base );
  520. }
  521. inline unsigned long long stoull (string_view str, size_t* idx=0, int base=10) {
  522. return std::stoull ( std::string(str), idx, base );
  523. }
  524. inline float stof (string_view str, size_t* idx=0) {
  525. return std::stof ( std::string(str), idx );
  526. }
  527. inline double stod (string_view str, size_t* idx=0) {
  528. return std::stod ( std::string(str), idx );
  529. }
  530. inline long double stold (string_view str, size_t* idx=0) {
  531. return std::stold ( std::string(str), idx );
  532. }
  533. inline int stoi (wstring_view str, size_t* idx=0, int base=10) {
  534. return std::stoi ( std::wstring(str), idx, base );
  535. }
  536. inline long stol (wstring_view str, size_t* idx=0, int base=10) {
  537. return std::stol ( std::wstring(str), idx, base );
  538. }
  539. inline unsigned long stoul (wstring_view str, size_t* idx=0, int base=10) {
  540. return std::stoul ( std::wstring(str), idx, base );
  541. }
  542. inline long long stoll (wstring_view str, size_t* idx=0, int base=10) {
  543. return std::stoll ( std::wstring(str), idx, base );
  544. }
  545. inline unsigned long long stoull (wstring_view str, size_t* idx=0, int base=10) {
  546. return std::stoull ( std::wstring(str), idx, base );
  547. }
  548. inline float stof (wstring_view str, size_t* idx=0) {
  549. return std::stof ( std::wstring(str), idx );
  550. }
  551. inline double stod (wstring_view str, size_t* idx=0) {
  552. return std::stod ( std::wstring(str), idx );
  553. }
  554. inline long double stold (wstring_view str, size_t* idx=0) {
  555. return std::stold ( std::wstring(str), idx );
  556. }
  557. #endif
  558. }
  559. #if 0
  560. namespace std {
  561. // Hashing
  562. template<> struct hash<boost::string_view>;
  563. template<> struct hash<boost::u16string_view>;
  564. template<> struct hash<boost::u32string_view>;
  565. template<> struct hash<boost::wstring_view>;
  566. }
  567. #endif
  568. #endif