qxLib
string.inl
Go to the documentation of this file.
1 /**
2 
3  @file string.inl
4  @author Khrapov
5  @date 4.09.2019
6  @copyright © Nick Khrapov, 2021. All right reserved.
7 
8 **/
9 
10 namespace qx
11 {
12 
13 template<class char_t, class traits_t>
14 inline basic_string<char_t, traits_t>::basic_string(size_type nSymbols, value_type chSymbol) noexcept
15 {
16  assign(nSymbols, chSymbol);
17 }
18 
19 template<class char_t, class traits_t>
20 inline basic_string<char_t, traits_t>::basic_string(const_pointer pszSource, size_type nSymbols) noexcept
21 {
22  assign(pszSource, nSymbols);
23 }
24 
25 template<class char_t, class traits_t>
26 inline basic_string<char_t, traits_t>::basic_string(const_pointer pszSource) noexcept
27 {
28  assign(pszSource);
29 }
30 
31 template<class char_t, class traits_t>
33 {
34  assign(std::move(sAnother));
35 }
36 
37 template<class char_t, class traits_t>
39 {
40  assign(sAnother);
41 }
42 
43 template<class char_t, class traits_t>
44 template<class fwd_it_t>
45 inline basic_string<char_t, traits_t>::basic_string(fwd_it_t itFirst, fwd_it_t itLast) noexcept
46 {
47  assign(itFirst, itLast);
48 }
49 
50 template<class char_t, class traits_t>
51 template<range_of_t_c<char_t> string_t>
52 inline basic_string<char_t, traits_t>::basic_string(const string_t& sAnother) noexcept
53 {
54  assign(sAnother);
55 }
56 
57 template<class char_t, class traits_t>
59 {
60  free();
61 }
62 
63 template<class char_t, class traits_t>
64 inline void basic_string<char_t, traits_t>::assign(size_type nSymbols, value_type chSymbol) noexcept
65 {
66  if (_resize(nSymbols))
67  std::fill(begin(), end(), chSymbol);
68 }
69 
70 template<class char_t, class traits_t>
71 inline void basic_string<char_t, traits_t>::assign(const_pointer pszSource, size_type nSymbols) noexcept
72 {
73  if (pszSource && _resize(nSymbols))
74  std::memmove(data(), pszSource, nSymbols * sizeof(value_type));
75 }
76 
77 template<class char_t, class traits_t>
78 inline void basic_string<char_t, traits_t>::assign(const_pointer pszSource) noexcept
79 {
80  if (pszSource != data())
81  assign(pszSource, traits_t::length(pszSource));
82 }
83 
84 template<class char_t, class traits_t>
85 inline void basic_string<char_t, traits_t>::assign(basic_string&& sAnother) noexcept
86 {
87  std::swap(m_Data, sAnother.m_Data);
88 }
89 
90 template<class char_t, class traits_t>
91 inline void basic_string<char_t, traits_t>::assign(const basic_string& sAnother) noexcept
92 {
93  if (sAnother.data() != data())
94  assign(sAnother.data());
95 }
96 
97 template<class char_t, class traits_t>
98 template<class fwd_it_t>
99 inline void basic_string<char_t, traits_t>::assign(fwd_it_t itFirst, fwd_it_t itLast) noexcept
100 {
101  if (_resize(std::distance(itFirst, itLast)))
102  {
103  fwd_it_t itOther = itFirst;
104  iterator itThis = begin();
105 
106  while (itOther != itLast)
107  {
108  *itThis = *itOther;
109  ++itThis;
110  ++itOther;
111  }
112  }
113 }
114 
115 template<class char_t, class traits_t>
116 template<range_of_t_c<char_t> string_t>
117 inline void basic_string<char_t, traits_t>::assign(const string_t& sAnother) noexcept
118 {
119  assign(sAnother.cbegin(), sAnother.cend());
120 }
121 
122 template<class char_t, class traits_t>
123 template<class... args_t>
124  requires format_acceptable_args<char_t, args_t...>
125 void basic_string<char_t, traits_t>::format(format_string_type<args_t...> sFormat, const args_t&... args)
126 {
127  vformat(sFormat.get(), args...);
128 }
129 
130 template<class char_t, class traits_t>
131 template<class... args_t>
132  requires format_acceptable_args<char_t, args_t...>
134  format_string_type<args_t...> sFormat,
135  const args_t&... args)
136 {
137  return static_vformat(sFormat.get(), args...);
138 }
139 
140 template<class char_t, class traits_t>
141 template<class... args_t>
142  requires format_acceptable_args<char_t, args_t...>
143 void basic_string<char_t, traits_t>::append_format(format_string_type<args_t...> sFormat, const args_t&... args)
144 {
145  append_vformat(sFormat.get(), args...);
146 }
147 
148 template<class char_t, class traits_t>
149 template<class... args_t>
150  requires format_acceptable_args<char_t, args_t...>
151 void basic_string<char_t, traits_t>::vformat(string_view svFormat, const args_t&... args)
152 {
153  clear();
154  append_vformat(svFormat, args...);
155 }
156 
157 template<class char_t, class traits_t>
158 template<class... args_t>
159  requires format_acceptable_args<char_t, args_t...>
161  string_view svFormat,
162  const args_t&... args)
163 {
164  basic_string sTemp;
165  sTemp.vformat(svFormat, args...);
166  return sTemp;
167 }
168 
169 template<class char_t, class traits_t>
170 template<class... args_t>
171  requires format_acceptable_args<char_t, args_t...>
172 inline void basic_string<char_t, traits_t>::append_vformat(string_view svFormat, const args_t&... args)
173 {
174  if (!svFormat.empty())
175  traits_t::format_to(std::back_inserter(*this), svFormat, args...);
176 }
177 
178 template<class char_t, class traits_t>
180 {
181  std::swap(m_Data, sOther.m_Data);
182 }
183 
184 template<class char_t, class traits_t>
185 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::reserve(
186  size_type nCapacity) noexcept
187 {
188  if (nCapacity > capacity())
189  _resize(nCapacity, string_resize_type::reserve);
190 
191  return capacity();
192 }
193 
194 template<class char_t, class traits_t>
196 {
197  if (!m_Data.is_small() && capacity() > size())
198  _resize(size(), string_resize_type::shrink_to_fit);
199 }
200 
201 template<class char_t, class traits_t>
203 {
204  m_Data.free();
205 }
206 
207 template<class char_t, class traits_t>
208 inline typename basic_string<char_t, traits_t>::string_view basic_string<char_t, traits_t>::substr(
209  size_type nPos,
210  size_type nSymbols) const noexcept
211 {
212  return string_view(data() + nPos, nSymbols != npos ? nSymbols : size() - nPos);
213 }
214 
215 template<class char_t, class traits_t>
217 {
218  for (value_type& ch : *this)
219  ch = traits_t::to_lower(ch);
220 }
221 
222 template<class char_t, class traits_t>
224 {
225  for (value_type& ch : *this)
226  ch = traits_t::to_upper(ch);
227 }
228 
229 template<class char_t, class traits_t>
230 inline typename basic_string<char_t, traits_t>::value_type basic_string<char_t, traits_t>::front() const noexcept
231 {
232  return at(0);
233 }
234 
235 template<class char_t, class traits_t>
236 inline typename basic_string<char_t, traits_t>::value_type basic_string<char_t, traits_t>::back() const noexcept
237 {
238  return at(size() - 1);
239 }
240 
241 template<class char_t, class traits_t>
242 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::length() const noexcept
243 {
244  return size();
245 }
246 
247 template<class char_t, class traits_t>
248 inline typename basic_string<char_t, traits_t>::const_pointer basic_string<char_t, traits_t>::c_str() const noexcept
249 {
250  return data();
251 }
252 
253 template<class char_t, class traits_t>
254 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::capacity() const noexcept
255 {
256  return m_Data.capacity();
257 }
258 
259 template<class char_t, class traits_t>
260 constexpr typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::max_size() noexcept
261 {
262  return std::numeric_limits<size_type>::max() - 1 // npos reserved
263  - 1; // null terminator
264 }
265 
266 template<class char_t, class traits_t>
267 template<class to_t>
268 inline std::optional<to_t> basic_string<char_t, traits_t>::to(const_pointer pszFormat) const noexcept
269 {
270  std::optional<to_t> optResult = std::nullopt;
271 
272  if constexpr (
273  std::is_trivial_v<to_t> && std::is_standard_layout_v<to_t> || std::is_pointer_v<to_t>
274  || std::is_same_v<to_t, std::nullptr_t>)
275  {
276  if constexpr (std::is_same_v<to_t, std::nullptr_t>)
277  {
278  if (compare(QX_STR_PREFIX(typename traits_t::value_type, "nullptr")) == 0)
279  {
280  optResult = nullptr;
281  }
282  }
283  else if constexpr (std::is_same_v<to_t, bool>)
284  {
285  if (compare(QX_STR_PREFIX(typename traits_t::value_type, "true")) == 0)
286  {
287  optResult = true;
288  }
289  else if (compare(QX_STR_PREFIX(typename traits_t::value_type, "false")) == 0)
290  {
291  optResult = false;
292  }
293  }
294  else if (const auto pszSelectedFormat = pszFormat ? pszFormat : get_format_specifier<value_type, to_t>())
295  {
296  to_t result;
297  const int nConvertedArgs = traits_t::sscanf(data(), pszSelectedFormat, &result);
298 
299  if (nConvertedArgs == 1)
300  optResult = result;
301  }
302  }
303  else
304  {
305  to_t result;
306  sstream_type ss(data());
307  ss >> result;
308  optResult = result;
309  }
310 
311  return optResult;
312 }
313 
314 template<class char_t, class traits_t>
315 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::copy(
316  pointer pDest,
317  size_type nCount,
318  size_type nPos) const noexcept
319 {
320  size_type nCharsToCopy = 0;
321 
322  if (pDest && nCount > 0 && nPos < size())
323  {
324  nCharsToCopy = std::min(nPos + nCount, size()) - nPos;
325  std::memcpy(pDest, data() + nPos, nCharsToCopy * sizeof(value_type));
326  }
327 
328  return nCharsToCopy;
329 }
330 
331 template<class char_t, class traits_t>
332 template<class from_t>
333 inline void basic_string<char_t, traits_t>::from(const from_t& data)
334 {
335  if constexpr (
336  std::is_trivial_v<from_t> && std::is_standard_layout_v<from_t> || std::is_pointer_v<from_t>
337  || std::is_same_v<from_t, std::nullptr_t>)
338  {
339  if constexpr (std::is_same_v<from_t, std::nullptr_t>)
340  {
341  assign(QX_STR_PREFIX(typename traits_t::value_type, "nullptr"));
342  }
343  else if constexpr (std::is_same_v<from_t, bool>)
344  {
345  assign(
346  data ? QX_STR_PREFIX(typename traits_t::value_type, "true")
347  : QX_STR_PREFIX(typename traits_t::value_type, "false"));
348  }
349  else
350  {
351  format(QX_STR_PREFIX(typename traits_t::value_type, "{}"), data);
352  }
353  }
354  else
355  {
356  sstream_type ss;
357  ss << data;
358  assign(ss.str());
359  }
360 }
361 
362 template<class char_t, class traits_t>
363 template<class from_t>
364 inline basic_string<char_t, traits_t> basic_string<char_t, traits_t>::static_from(const from_t& data)
365 {
366  basic_string sTemp;
367  sTemp.from(data);
368  return std::move(sTemp);
369 }
370 
371 template<class char_t, class traits_t>
372 inline void basic_string<char_t, traits_t>::append(value_type chSymbol) noexcept
373 {
374  append(&chSymbol, 1);
375 }
376 
377 template<class char_t, class traits_t>
378 inline void basic_string<char_t, traits_t>::append(const_pointer pszStr, size_type nSymbols) noexcept
379 {
380  if (pszStr)
381  {
382  const size_type nSize = size();
383  const size_type nSizeSource = nSymbols == npos ? traits_t::length(pszStr) : nSymbols;
384 
385  if (_resize(nSize + nSizeSource))
386  std::memcpy(data() + nSize, pszStr, nSizeSource * sizeof(value_type));
387  }
388 }
389 
390 template<class char_t, class traits_t>
391 inline void basic_string<char_t, traits_t>::append(const basic_string& sStr) noexcept
392 {
393  append(sStr.data(), sStr.size());
394 }
395 
396 template<class char_t, class traits_t>
397 template<class fwd_it_t>
398 inline void basic_string<char_t, traits_t>::append(fwd_it_t itBegin, fwd_it_t itEnd) noexcept
399 {
400  for (auto it = itBegin; it != itEnd; ++it)
401  push_back(*it);
402 }
403 
404 template<class char_t, class traits_t>
405 template<range_of_t_c<char_t> string_t>
406 inline void basic_string<char_t, traits_t>::append(const string_t& sStr) noexcept
407 {
408  append(sStr.cbegin(), sStr.cend());
409 }
410 
411 template<class char_t, class traits_t>
412 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::insert(
413  size_type nPos,
414  value_type chSymbol) noexcept
415 {
416  return insert(nPos, &chSymbol, 1);
417 }
418 
419 template<class char_t, class traits_t>
420 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::insert(
421  size_type nPos,
422  const_pointer pszWhat,
423  size_type nSymbols) noexcept
424 {
425  if (pszWhat)
426  {
427  const size_type nSize = size();
428  const size_type nSizeSource = nSymbols == npos ? traits_t::length(pszWhat) : nSymbols;
429 
430  if (nSizeSource > 0 && _resize(nSize + nSizeSource))
431  {
432  std::memmove(data() + nPos + nSizeSource, data() + nPos, (nSize - nPos) * sizeof(value_type));
433  std::memcpy(data() + nPos, pszWhat, nSizeSource * sizeof(value_type));
434  return nPos + nSizeSource;
435  }
436  }
437 
438  return npos;
439 }
440 
441 template<class char_t, class traits_t>
442 template<class fwd_it_t>
443 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::insert(
444  size_type nPos,
445  fwd_it_t itWhatBegin,
446  fwd_it_t itWhatEnd) noexcept
447 {
448  if constexpr (is_random_access_iterator<fwd_it_t>)
449  {
450  return insert(nPos, itWhatBegin.operator->(), static_cast<size_type>(itWhatEnd - itWhatBegin));
451  }
452  else
453  {
454  size_type nWhatSize = 0;
455  for (auto it = itWhatBegin; it != itWhatEnd; ++it)
456  ++nWhatSize;
457 
458  size_type nStartSymbols = size();
459  if (nWhatSize > 0 && _resize(nStartSymbols + nWhatSize))
460  {
461  std::memmove(data() + nPos + nWhatSize, data() + nPos, (nStartSymbols - nPos) * sizeof(value_type));
462 
463  size_type nWhatPos = 0;
464  for (auto it = itWhatBegin; it != itWhatEnd; ++it)
465  {
466  at(nPos + nWhatPos) = *it;
467  ++nWhatPos;
468  }
469 
470  return nPos + nWhatSize;
471  }
472  else
473  {
474  return npos;
475  }
476  }
477 }
478 
479 template<class char_t, class traits_t>
480 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::insert(
481  size_type nPos,
482  const basic_string& sWhat) noexcept
483 {
484  return insert(nPos, sWhat.data(), sWhat.size());
485 }
486 
487 template<class char_t, class traits_t>
488 template<range_of_t_c<char_t> string_t>
489 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::insert(
490  size_type nPos,
491  string_t sWhat) noexcept
492 {
493  return insert(static_cast<size_type>(nPos), sWhat.cbegin(), sWhat.cend());
494 }
495 
496 template<class char_t, class traits_t>
497 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::insert(
498  const_iterator itPos,
499  value_type chSymbol) noexcept
500 {
501  return insert(static_cast<size_type>(itPos - cbegin()), chSymbol);
502 }
503 
504 template<class char_t, class traits_t>
505 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::insert(
506  const_iterator itPos,
507  const_pointer pszWhat,
508  size_type nSymbols) noexcept
509 {
510  return insert(static_cast<size_type>(itPos - cbegin()), pszWhat, nSymbols);
511 }
512 
513 template<class char_t, class traits_t>
514 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::insert(
515  const_iterator itPos,
516  const basic_string& sWhat) noexcept
517 {
518  return insert(static_cast<size_type>(itPos - cbegin()), sWhat.data(), sWhat.size());
519 }
520 
521 template<class char_t, class traits_t>
522 template<class fwd_it_t>
523 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::insert(
524  const_iterator itPos,
525  fwd_it_t itWhatBegin,
526  fwd_it_t itWhatEnd) noexcept
527 {
528  return insert(static_cast<size_type>(itPos - begin()), itWhatBegin, itWhatEnd);
529 }
530 
531 template<class char_t, class traits_t>
532 template<range_of_t_c<char_t> string_t>
533 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::insert(
534  const_iterator itPos,
535  string_t sWhat) noexcept
536 {
537  return insert(static_cast<size_type>(itPos - begin()), sWhat.cbegin(), sWhat.cend());
538 }
539 
540 template<class char_t, class traits_t>
541 inline void basic_string<char_t, traits_t>::push_back(value_type chSymbol) noexcept
542 {
543  insert(size(), &chSymbol, 1);
544 }
545 
546 template<class char_t, class traits_t>
547 inline void basic_string<char_t, traits_t>::push_front(value_type chSymbol) noexcept
548 {
549  insert(0, &chSymbol, 1);
550 }
551 
552 template<class char_t, class traits_t>
553 inline void basic_string<char_t, traits_t>::erase(iterator itFirst, iterator itLast) noexcept
554 {
555  if (const typename iterator::difference_type nCharsToErase = itLast - itFirst; nCharsToErase > 0)
556  {
557  const size_type nStartSize = size();
558  const size_type nSymbolsToCopy = itLast != end() ? traits_t::length(itLast.operator->()) : 0;
559 
560  if (nSymbolsToCopy > 0)
561  {
562  std::memmove(itFirst.operator->(), itLast.operator->(), nSymbolsToCopy * sizeof(value_type));
563  }
564 
565  if (static_cast<typename iterator::difference_type>(nStartSize) >= nCharsToErase)
566  {
567  _resize(nStartSize - nCharsToErase);
568  }
569  }
570 }
571 
572 template<class char_t, class traits_t>
574 {
575  erase(itPos, itPos + 1);
576 }
577 
578 template<class char_t, class traits_t>
579 inline void basic_string<char_t, traits_t>::erase(size_type nPos) noexcept
580 {
581  erase(iterator(this, nPos), iterator(this, nPos + 1));
582 }
583 
584 template<class char_t, class traits_t>
585 inline void basic_string<char_t, traits_t>::erase(size_type nPos, size_type nSymbols) noexcept
586 {
587  erase(iterator(this, nPos), iterator(this, nPos + nSymbols));
588 }
589 
590 template<class char_t, class traits_t>
591 inline typename basic_string<char_t, traits_t>::value_type basic_string<char_t, traits_t>::pop_back() noexcept
592 {
593  value_type chRet = back();
594  erase(size() - 1);
595  return chRet;
596 }
597 
598 template<class char_t, class traits_t>
599 inline typename basic_string<char_t, traits_t>::value_type basic_string<char_t, traits_t>::pop_front() noexcept
600 {
601  value_type chRet = front();
602  erase(0);
603  return chRet;
604 }
605 
606 template<class char_t, class traits_t>
607 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::trim_left() noexcept
608 {
609  return _trim_left(
610  [](value_type ch)
611  {
612  return traits_t::is_space(ch);
613  });
614 }
615 
616 template<class char_t, class traits_t>
617 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::trim_left(
618  value_type chSymbol) noexcept
619 {
620  return _trim_left(
621  [chSymbol](value_type ch)
622  {
623  return ch == chSymbol;
624  });
625 }
626 
627 template<class char_t, class traits_t>
628 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::trim_left(
629  const_pointer pszStr) noexcept
630 {
631  if (pszStr)
632  {
633  return _trim_left(
634  [pszStr](value_type ch)
635  {
636  for (size_type j = 0; pszStr[j] != QX_CHAR_PREFIX(value_type, '\0'); ++j)
637  {
638  if (pszStr[j] == ch)
639  return true;
640  }
641 
642  return false;
643  });
644  }
645  else
646  {
647  return 0;
648  }
649 }
650 
651 template<class char_t, class traits_t>
652 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::trim_left(
653  const_pointer pszStr,
654  size_type nStrSize) noexcept
655 {
656  if (pszStr)
657  {
658  return _trim_left(
659  [pszStr, nStrSize](value_type ch)
660  {
661  for (size_type j = 0; j < nStrSize; ++j)
662  if (pszStr[j] == ch)
663  return true;
664 
665  return false;
666  });
667  }
668  else
669  {
670  return 0;
671  }
672 }
673 
674 template<class char_t, class traits_t>
675 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::trim_left(
676  const basic_string& sStr) noexcept
677 {
678  return trim_left(sStr.data(), sStr.size());
679 }
680 
681 template<class char_t, class traits_t>
682 template<class fwd_it_t>
683 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::trim_left(
684  fwd_it_t itBegin,
685  fwd_it_t itEnd) noexcept
686 {
687  return _trim_left(
688  [itBegin, itEnd](auto ch)
689  {
690  for (auto it = itBegin; it != itEnd; ++it)
691  if (*it == ch)
692  return true;
693 
694  return false;
695  });
696 }
697 
698 template<class char_t, class traits_t>
699 template<range_of_t_c<char_t> string_t>
700 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::trim_left(
701  const string_t& sStr) noexcept
702 {
703  return trim_left(sStr.cbegin(), sStr.cend());
704 }
705 
706 template<class char_t, class traits_t>
707 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::trim_right() noexcept
708 {
709  return _trim_right(
710  [](value_type ch)
711  {
712  return traits_t::is_space(ch);
713  });
714 }
715 
716 template<class char_t, class traits_t>
717 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::trim_right(
718  value_type chSymbol) noexcept
719 {
720  return _trim_right(
721  [chSymbol](value_type ch)
722  {
723  return ch == chSymbol;
724  });
725 }
726 
727 template<class char_t, class traits_t>
728 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::trim_right(
729  const_pointer pszStr) noexcept
730 {
731  if (pszStr)
732  {
733  return _trim_right(
734  [pszStr](value_type ch)
735  {
736  for (size_type j = 0; pszStr[j] != QX_CHAR_PREFIX(value_type, '\0'); ++j)
737  {
738  if (pszStr[j] == ch)
739  return true;
740  }
741 
742  return false;
743  });
744  }
745  else
746  {
747  return 0;
748  }
749 }
750 
751 template<class char_t, class traits_t>
752 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::trim_right(
753  const_pointer pszStr,
754  size_type nStrSize) noexcept
755 {
756  if (pszStr)
757  {
758  return _trim_right(
759  [pszStr, nStrSize](value_type ch)
760  {
761  for (size_type j = 0; j < nStrSize; ++j)
762  if (pszStr[j] == ch)
763  return true;
764 
765  return false;
766  });
767  }
768  else
769  {
770  return 0;
771  }
772 }
773 
774 template<class char_t, class traits_t>
775 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::trim_right(
776  const basic_string& sStr) noexcept
777 {
778  return trim_right(sStr.data(), sStr.size());
779 }
780 
781 template<class char_t, class traits_t>
782 template<class fwd_it_t>
783 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::trim_right(
784  fwd_it_t itBegin,
785  fwd_it_t itEnd) noexcept
786 {
787  return _trim_right(
788  [itBegin, itEnd](auto ch)
789  {
790  for (auto it = itBegin; it != itEnd; ++it)
791  if (*it == ch)
792  return true;
793 
794  return false;
795  });
796 }
797 
798 template<class char_t, class traits_t>
799 template<range_of_t_c<char_t> string_t>
800 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::trim_right(
801  const string_t& sStr) noexcept
802 {
803  return trim_right(sStr.cbegin(), sStr.cend());
804 }
805 
806 template<class char_t, class traits_t>
807 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::trim() noexcept
808 {
809  return _trim(
810  [](value_type ch)
811  {
812  return traits_t::is_space(ch);
813  });
814 }
815 
816 template<class char_t, class traits_t>
817 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::trim(
818  value_type chSymbol) noexcept
819 {
820  return _trim(
821  [chSymbol](value_type ch)
822  {
823  return ch == chSymbol;
824  });
825 }
826 
827 template<class char_t, class traits_t>
828 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::trim(
829  const_pointer pszStr) noexcept
830 {
831  if (pszStr)
832  {
833  return _trim(
834  [pszStr](value_type ch)
835  {
836  for (size_type j = 0; pszStr[j] != QX_CHAR_PREFIX(value_type, '\0'); ++j)
837  {
838  if (pszStr[j] == ch)
839  return true;
840  }
841 
842  return false;
843  });
844  }
845  else
846  {
847  return 0;
848  }
849 }
850 
851 template<class char_t, class traits_t>
852 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::trim(
853  const_pointer pszStr,
854  size_type nStrSize) noexcept
855 {
856  if (pszStr)
857  {
858  return _trim(
859  [pszStr, nStrSize](value_type ch)
860  {
861  for (size_type j = 0; j < nStrSize; ++j)
862  if (pszStr[j] == ch)
863  return true;
864 
865  return false;
866  });
867  }
868  else
869  {
870  return 0;
871  }
872 }
873 
874 template<class char_t, class traits_t>
875 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::trim(
876  const basic_string& sStr) noexcept
877 {
878  return trim(sStr.data(), sStr.size());
879 }
880 
881 template<class char_t, class traits_t>
882 template<class fwd_it_t>
883 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::trim(
884  fwd_it_t itBegin,
885  fwd_it_t itEnd) noexcept
886 {
887  return _trim(
888  [itBegin, itEnd](auto ch)
889  {
890  for (auto it = itBegin; it != itEnd; ++it)
891  if (*it == ch)
892  return true;
893 
894  return false;
895  });
896 }
897 
898 template<class char_t, class traits_t>
899 template<range_of_t_c<char_t> string_t>
900 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::trim(
901  const string_t& sStr) noexcept
902 {
903  return trim(sStr.cbegin(), sStr.cend());
904 }
905 
906 template<class char_t, class traits_t>
907 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::remove(
908  value_type chSymbol,
909  size_type nBegin,
910  size_type nEnd) noexcept
911 {
912  size_type nPos = find(chSymbol, nBegin, nEnd);
913 
914  if (nPos != npos)
915  erase(nPos, 1);
916 
917  return nPos;
918 }
919 
920 template<class char_t, class traits_t>
921 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::remove(
922  const_pointer pszStr,
923  size_type nBegin,
924  size_type nEnd,
925  size_type nStrSize) noexcept
926 {
927  if (pszStr)
928  {
929  const size_type nLocalStrSize = nStrSize != npos ? nStrSize : traits_t::length(pszStr);
930  const size_type nPos = find(pszStr, nBegin, nLocalStrSize, nEnd);
931 
932  if (nPos != npos)
933  erase(nPos, nLocalStrSize);
934 
935  return nPos;
936  }
937  else
938  {
939  return npos;
940  }
941 }
942 
943 template<class char_t, class traits_t>
944 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::remove(
945  const basic_string& sStr,
946  size_type nBegin,
947  size_type nEnd) noexcept
948 {
949  const size_type nPos = find(sStr, nBegin, nEnd);
950 
951  if (nPos != npos)
952  erase(nPos, sStr.size());
953 
954  return nPos;
955 }
956 
957 template<class char_t, class traits_t>
958 template<class fwd_it_t>
959 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::remove(
960  fwd_it_t itBegin,
961  fwd_it_t itEnd,
962  size_type nBegin,
963  size_type nEnd) noexcept
964 {
965  const size_type nPos = find(itBegin, itEnd, nBegin, nEnd);
966 
967  if (nPos != npos)
968  erase(nPos, static_cast<size_type>(std::distance(itBegin, itEnd)));
969 
970  return nPos;
971 }
972 
973 template<class char_t, class traits_t>
974 template<range_of_t_c<char_t> string_t>
975 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::remove(
976  const string_t& sStr,
977  size_type nBegin,
978  size_type nEnd) noexcept
979 {
980  return remove(sStr.cbegin(), sStr.cend(), nBegin, nEnd);
981 }
982 
983 template<class char_t, class traits_t>
984 inline bool basic_string<char_t, traits_t>::remove_prefix(value_type chSymbol) noexcept
985 {
986  return remove(chSymbol, static_cast<size_type>(0), static_cast<size_type>(1)) != npos;
987 }
988 
989 template<class char_t, class traits_t>
990 inline bool basic_string<char_t, traits_t>::remove_prefix(const_pointer pszStr, size_type nStrSize) noexcept
991 {
992  return remove(pszStr, static_cast<size_type>(0), nStrSize, nStrSize) != npos;
993 }
994 
995 template<class char_t, class traits_t>
997 {
998  return remove(sStr, static_cast<size_type>(0), sStr.size()) != npos;
999 }
1000 
1001 template<class char_t, class traits_t>
1002 template<class fwd_it_t>
1003 inline bool basic_string<char_t, traits_t>::remove_prefix(fwd_it_t itBegin, fwd_it_t itEnd) noexcept
1004 {
1005  return remove(itBegin, itEnd, static_cast<size_type>(0), static_cast<size_type>(std::distance(itBegin, itEnd)))
1006  != npos;
1007 }
1008 
1009 template<class char_t, class traits_t>
1010 template<range_of_t_c<char_t> string_t>
1011 inline bool basic_string<char_t, traits_t>::remove_prefix(const string_t& sStr) noexcept
1012 {
1013  return remove_prefix(sStr.cbegin(), sStr.cend());
1015 
1016 template<class char_t, class traits_t>
1017 inline bool basic_string<char_t, traits_t>::remove_suffix(value_type chSymbol) noexcept
1018 {
1019  const size_type nSize = size();
1020  return remove(chSymbol, nSize - 1, nSize) != npos;
1021 }
1022 
1023 template<class char_t, class traits_t>
1024 inline bool basic_string<char_t, traits_t>::remove_suffix(const_pointer pszStr, size_type nStrSize) noexcept
1025 {
1026  if (pszStr)
1027  {
1028  const size_type nSize = size();
1029  const size_type nLocalStrSize = nStrSize != npos ? nStrSize : traits_t::length(pszStr);
1030 
1031  return remove(pszStr, nSize - nLocalStrSize, nSize, nLocalStrSize) != npos;
1032  }
1033  else
1034  {
1035  return false;
1036  }
1037 }
1038 
1039 template<class char_t, class traits_t>
1041 {
1042  return remove_suffix(sStr.data(), sStr.size());
1044 
1045 template<class char_t, class traits_t>
1046 template<class fwd_it_t>
1047 inline bool basic_string<char_t, traits_t>::remove_suffix(fwd_it_t itBegin, fwd_it_t itEnd) noexcept
1048 {
1049  const size_type nSize = size();
1050  return remove(itBegin, itEnd, nSize - static_cast<size_type>(std::distance(itBegin, itEnd)), nSize) != npos;
1051 }
1052 
1053 template<class char_t, class traits_t>
1054 template<range_of_t_c<char_t> string_t>
1055 inline bool basic_string<char_t, traits_t>::remove_suffix(const string_t& sStr) noexcept
1056 {
1057  return remove_suffix(sStr.cbegin(), sStr.cend());
1058 }
1059 
1060 template<class char_t, class traits_t>
1061 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::remove_all(
1062  value_type chSymbol,
1063  size_type nBegin,
1064  size_type nEnd) noexcept
1065 {
1066  size_type nOccurrences = 0;
1067  size_type nLastOccurrencePos = nBegin;
1068 
1069  do
1070  {
1071  ++nOccurrences;
1072  nLastOccurrencePos = remove(chSymbol, nLastOccurrencePos, nEnd);
1073  } while (nLastOccurrencePos != npos);
1074 
1075  return nOccurrences - 1;
1076 }
1078 template<class char_t, class traits_t>
1079 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::remove_all(
1080  const_pointer pszStr,
1081  size_type nBegin,
1082  size_type nEnd,
1083  size_type nStrSize) noexcept
1084 {
1085  if (pszStr)
1086  {
1087  size_type nOccurrences = 0;
1088  size_type nLastOccurrencePos = nBegin;
1089 
1090  do
1091  {
1092  ++nOccurrences;
1093  nLastOccurrencePos = remove(pszStr, nLastOccurrencePos, nEnd, nStrSize);
1094  } while (nLastOccurrencePos != npos);
1095 
1096  return nOccurrences - 1;
1097  }
1098  else
1099  {
1100  return 0;
1101  }
1102 }
1103 
1104 template<class char_t, class traits_t>
1105 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::remove_all(
1106  const basic_string& sStr,
1107  size_type nBegin,
1108  size_type nEnd) noexcept
1109 {
1110  return remove_all(sStr.data(), nBegin, nEnd, sStr.size());
1111 }
1113 template<class char_t, class traits_t>
1114 template<class fwd_it_t>
1115 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::remove_all(
1116  fwd_it_t itFirst,
1117  fwd_it_t itLast,
1118  size_type nBegin,
1119  size_type nEnd) noexcept
1120 {
1121  size_type nOccurrences = 0;
1122  size_type nLastOccurrencePos = nBegin;
1123 
1124  do
1125  {
1126  ++nOccurrences;
1127  nLastOccurrencePos = remove(itFirst, itLast, nLastOccurrencePos, nEnd);
1128  } while (nLastOccurrencePos != npos);
1129 
1130  return nOccurrences - 1;
1131 }
1132 
1133 template<class char_t, class traits_t>
1134 template<range_of_t_c<char_t> string_t>
1135 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::remove_all(
1136  const string_t& sStr,
1137  size_type nBegin,
1138  size_type nEnd) noexcept
1139 {
1140  return remove_all(sStr.cbegin(), sStr.cend(), nBegin, nEnd);
1141 }
1143 template<class char_t, class traits_t>
1144 template<class find_string_t, class replace_string_t>
1145 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::replace(
1146  const find_string_t& sFind,
1147  const replace_string_t& sReplace,
1148  size_type nBegin,
1149  size_type nEnd) noexcept
1151  auto get_size = []<class T>(const T& val) -> size_type
1152  {
1153  if constexpr (std::is_same_v<T, value_type>)
1154  {
1155  return 1;
1156  }
1157  else if constexpr (std::is_same_v<T, const_pointer>)
1158  {
1159  return traits_t::length(val);
1160  }
1161  else if constexpr (range_of_t_c<T, char_t>)
1162  {
1163  return val.size();
1164  }
1165  else
1166  {
1167  QX_STATIC_ASSERT_NO_INSTANTIATION("Unexpected type");
1168  return 0;
1169  }
1170  };
1171 
1172  auto get_data = []<class T>(const T& val) -> const_pointer
1173  {
1174  if constexpr (std::is_same_v<T, value_type>)
1175  {
1176  return &val;
1177  }
1178  else if constexpr (std::is_same_v<T, const_pointer>)
1179  {
1180  return val;
1181  }
1182  else if constexpr (range_of_t_c<T, char_t>)
1183  {
1184  return val.data();
1185  }
1186  else
1187  {
1188  QX_STATIC_ASSERT_NO_INSTANTIATION("Unexpected type");
1189  return nullptr;
1190  }
1191  };
1192 
1193  if (size_type nPos = find(sFind, nBegin, nEnd); nPos != npos)
1194  {
1195  const size_type nStartSize = size();
1196  const size_type nFindSize = get_size(sFind);
1197  const size_type nReplaceSize = get_size(sReplace);
1198  const size_type nNewSize = nStartSize - nFindSize + nReplaceSize;
1199 
1200  _resize(nNewSize, string_resize_type::reserve);
1201 
1202  std::memmove(
1203  data() + nPos + nReplaceSize,
1204  data() + nPos + nFindSize,
1205  (nStartSize - nPos - nFindSize) * sizeof(value_type));
1206 
1207  std::memcpy(data() + nPos, get_data(sReplace), nReplaceSize * sizeof(value_type));
1208 
1209  _resize(nNewSize);
1210 
1211  return nPos + nReplaceSize;
1212  }
1213  else
1214  {
1215  return npos;
1216  }
1217 }
1218 
1219 template<class char_t, class traits_t>
1220 template<class find_string_t, class replace_string_t>
1221 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::replace_all(
1222  const find_string_t& sFind,
1223  const replace_string_t& sReplace,
1224  size_type nBegin,
1225  size_type nEnd) noexcept
1226 {
1227  size_type nOccurrences = 0;
1228  size_type nPos = nBegin;
1229 
1230  do
1231  {
1232  nPos = replace(sFind, sReplace, nPos, nEnd);
1233  if (nPos != npos)
1234  ++nOccurrences;
1235  } while (nPos != npos);
1236 
1237  return nOccurrences;
1238 }
1239 
1240 template<class char_t, class traits_t>
1241 inline int basic_string<char_t, traits_t>::compare(value_type chSymbol) const noexcept
1242 {
1243  return compare(&chSymbol, 1);
1244 }
1245 
1246 template<class char_t, class traits_t>
1247 inline int basic_string<char_t, traits_t>::compare(const_pointer pszStr) const noexcept
1248 {
1249  return traits_t::compare(data(), pszStr);
1251 
1252 template<class char_t, class traits_t>
1253 inline int basic_string<char_t, traits_t>::compare(const_pointer pStr, size_type nSymbols) const noexcept
1254 {
1255  return traits_t::compare_n(data(), pStr, nSymbols);
1256 }
1257 
1258 template<class char_t, class traits_t>
1259 inline int basic_string<char_t, traits_t>::compare(const basic_string& sStr) const noexcept
1260 {
1261  return compare(sStr.data(), sStr.size());
1262 }
1263 
1264 template<class char_t, class traits_t>
1265 template<class fwd_it_t>
1266 inline int basic_string<char_t, traits_t>::compare(fwd_it_t itBegin, fwd_it_t itEnd) const noexcept
1267 {
1268  return iter_strcmp(cbegin(), cend(), itBegin, itEnd);
1269 }
1270 
1271 template<class char_t, class traits_t>
1272 template<range_of_t_c<char_t> string_t>
1273 inline int basic_string<char_t, traits_t>::compare(const string_t& sStr) const noexcept
1274 {
1275  return compare(sStr.cbegin(), sStr.cend());
1276 }
1278 template<class char_t, class traits_t>
1279 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find(
1280  value_type chSymbol,
1281  size_type nBegin,
1282  size_type nEnd) const noexcept
1283 {
1284  return _find(
1285  nBegin,
1286  nEnd,
1287  [chSymbol](const_pointer pCurrentChar)
1288  {
1289  return *pCurrentChar == chSymbol;
1290  });
1291 }
1292 
1293 template<class char_t, class traits_t>
1294 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find(
1295  const_pointer pszWhat,
1296  size_type nBegin,
1297  size_type nWhatSize,
1298  size_type nEnd) const noexcept
1299 {
1300  if (pszWhat)
1301  {
1302  const size_type nLocalWhatSize = nWhatSize != npos ? nWhatSize : traits_t::length(pszWhat);
1303 
1304  return _find(
1305  nBegin,
1306  nEnd,
1307  [pszWhat, nLocalWhatSize](const_pointer pCurrentChar)
1308  {
1309  return !traits_t::compare_n(pszWhat, pCurrentChar, nLocalWhatSize);
1310  });
1311  }
1312  else
1313  {
1314  return npos;
1315  }
1316 }
1317 
1318 template<class char_t, class traits_t>
1319 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find(
1320  const basic_string& sWhat,
1321  size_type nBegin,
1322  size_type nEnd) const noexcept
1323 {
1324  return find(sWhat.data(), nBegin, sWhat.size(), nEnd);
1325 }
1326 
1327 template<class char_t, class traits_t>
1328 template<class fwd_it_t>
1329 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find(
1330  fwd_it_t itWhatBegin,
1331  fwd_it_t itWhatEnd,
1332  size_type nBegin,
1333  size_type nEnd) const noexcept
1334 {
1335  return _find(
1336  nBegin,
1337  nEnd,
1338  [this, itWhatBegin, itWhatEnd, nEnd](const_pointer pCurrentChar)
1339  {
1340  return !iter_strcmp(
1341  const_iterator(this, static_cast<size_type>(pCurrentChar - data())),
1342  const_iterator(this, static_cast<size_type>(nEnd)),
1343  itWhatBegin,
1344  itWhatEnd);
1345  });
1346 }
1347 
1348 template<class char_t, class traits_t>
1349 template<range_of_t_c<char_t> string_t>
1350 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find(
1351  string_t sWhat,
1352  size_type nBegin,
1353  size_type nEnd) const noexcept
1354 {
1355  return find(sWhat.cbegin(), sWhat.cend(), nBegin, nEnd);
1357 
1358 template<class char_t, class traits_t>
1359 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::rfind(
1360  value_type chSymbol,
1361  size_type nBegin,
1362  size_type nEnd) const noexcept
1364  return _rfind(
1365  nBegin,
1366  nEnd,
1367  [chSymbol](const_pointer pCurrentChar)
1368  {
1369  return *pCurrentChar == chSymbol;
1370  });
1371 }
1372 
1373 template<class char_t, class traits_t>
1374 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::rfind(
1375  const_pointer pszWhat,
1376  size_type nBegin,
1377  size_type nWhatSize,
1378  size_type nEnd) const noexcept
1379 {
1380  if (pszWhat)
1381  {
1382  const size_type nLocalWhatSize = nWhatSize != npos ? nWhatSize : traits_t::length(pszWhat);
1383 
1384  return _rfind(
1385  nBegin,
1386  nEnd,
1387  [pszWhat, nLocalWhatSize](const_pointer pCurrentChar)
1388  {
1389  return !traits_t::compare_n(pszWhat, pCurrentChar, nLocalWhatSize);
1390  });
1391  }
1392  else
1393  {
1394  return npos;
1395  }
1396 }
1398 template<class char_t, class traits_t>
1399 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::rfind(
1400  const basic_string& sWhat,
1401  size_type nBegin,
1402  size_type nEnd) const noexcept
1403 {
1404  return rfind(sWhat.data(), nBegin, sWhat.size(), nEnd);
1405 }
1406 
1407 template<class char_t, class traits_t>
1408 template<class fwd_it_t>
1409 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::rfind(
1410  fwd_it_t itWhatBegin,
1411  fwd_it_t itWhatEnd,
1412  size_type nBegin,
1413  size_type nEnd) const noexcept
1415  return _rfind(
1416  nBegin,
1417  nEnd,
1418  [this, itWhatBegin, itWhatEnd, nEnd](const_pointer pCurrentChar)
1419  {
1420  return !iter_strcmp(
1421  const_iterator(this, static_cast<size_type>(pCurrentChar - data())),
1422  const_iterator(this, static_cast<size_type>(nEnd)),
1423  itWhatBegin,
1424  itWhatEnd);
1425  });
1426 }
1427 
1428 template<class char_t, class traits_t>
1429 template<range_of_t_c<char_t> string_t>
1430 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::rfind(
1431  string_t sWhat,
1432  size_type nBegin,
1433  size_type nEnd) const noexcept
1434 {
1435  return rfind(sWhat.cbegin(), sWhat.cend(), nBegin, nEnd);
1436 }
1437 
1438 template<class char_t, class traits_t>
1439 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find_first_of(
1440  value_type chSymbol,
1441  size_type nBegin) const noexcept
1442 {
1443  return find(chSymbol, nBegin);
1444 }
1446 template<class char_t, class traits_t>
1447 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find_first_of(
1448  const_pointer pszWhat,
1449  size_type nBegin,
1450  size_type nWhatSize) const noexcept
1451 {
1452  if (pszWhat)
1453  {
1454  return _find_first_of(
1455  pszWhat,
1456  pszWhat + nWhatSize,
1457  nBegin,
1458  [](const_pointer pChar)
1459  {
1460  return pChar + 1;
1461  });
1462  }
1463  else
1464  {
1465  return npos;
1466  }
1467 }
1468 
1469 template<class char_t, class traits_t>
1470 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find_first_of(
1471  const_pointer pszWhat,
1472  size_type nBegin) const noexcept
1473 {
1474  if (pszWhat)
1475  {
1476  return _find_first_of(
1477  pszWhat,
1478  static_cast<const_pointer>(nullptr),
1479  nBegin,
1480  [](const_pointer pChar)
1481  {
1482  return *(pChar + 1) != QX_CHAR_PREFIX(value_type, '\0') ? pChar + 1 : nullptr;
1483  });
1484  }
1485  else
1486  {
1487  return npos;
1488  }
1490 
1491 template<class char_t, class traits_t>
1492 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find_first_of(
1493  const basic_string& sWhat,
1494  size_type nBegin) const noexcept
1495 {
1496  return find_first_of(sWhat.data(), nBegin, sWhat.size());
1497 }
1498 
1499 template<class char_t, class traits_t>
1500 template<class fwd_it_t>
1501 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find_first_of(
1502  fwd_it_t itWhatBegin,
1503  fwd_it_t itWhatEnd,
1504  size_type nBegin) const noexcept
1505 {
1506  return _find_first_of(
1507  itWhatBegin,
1508  itWhatEnd,
1509  nBegin,
1510  [](fwd_it_t itChar)
1511  {
1512  return ++itChar;
1513  });
1514 }
1515 
1516 template<class char_t, class traits_t>
1517 template<range_of_t_c<char_t> string_t>
1518 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find_first_of(
1519  string_t sWhat,
1520  size_type nBegin) const noexcept
1521 {
1522  return find_first_of(sWhat.cbegin(), sWhat.cend(), nBegin);
1523 }
1524 
1525 template<class char_t, class traits_t>
1526 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find_last_of(
1527  value_type chSymbol,
1528  size_type nEnd) const noexcept
1529 {
1530  return rfind(chSymbol, npos, nEnd);
1531 }
1532 
1533 template<class char_t, class traits_t>
1534 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find_last_of(
1535  const_pointer pszWhat,
1536  size_type nEnd,
1537  size_type nWhatSize) const noexcept
1538 {
1539  if (pszWhat)
1540  {
1541  return _find_last_of(
1542  pszWhat,
1543  pszWhat + nWhatSize,
1544  nEnd,
1545  [](const_pointer pChar)
1546  {
1547  return pChar + 1;
1548  });
1549  }
1550  else
1551  {
1552  return npos;
1553  }
1554 }
1555 
1556 template<class char_t, class traits_t>
1557 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find_last_of(
1558  const_pointer pszWhat,
1559  size_type nEnd) const noexcept
1560 {
1561  if (pszWhat)
1562  {
1563  return _find_last_of(
1564  pszWhat,
1565  static_cast<const_pointer>(nullptr),
1566  nEnd,
1567  [](const_pointer pChar)
1568  {
1569  return *(pChar + 1) != QX_CHAR_PREFIX(value_type, '\0') ? pChar + 1 : nullptr;
1570  });
1571  }
1572  else
1573  {
1574  return npos;
1575  }
1576 }
1577 
1578 template<class char_t, class traits_t>
1579 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find_last_of(
1580  const basic_string& sWhat,
1581  size_type nEnd) const noexcept
1582 {
1583  return find_last_of(sWhat.data(), nEnd, sWhat.size());
1584 }
1585 
1586 template<class char_t, class traits_t>
1587 template<class fwd_it_t>
1588 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find_last_of(
1589  fwd_it_t itWhatBegin,
1590  fwd_it_t itWhatEnd,
1591  size_type nEnd) const noexcept
1592 {
1593  return _find_last_of(
1594  itWhatBegin,
1595  itWhatEnd,
1596  nEnd,
1597  [](fwd_it_t itChar)
1598  {
1599  return ++itChar;
1600  });
1601 }
1602 
1603 template<class char_t, class traits_t>
1604 template<range_of_t_c<char_t> string_t>
1605 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find_last_of(
1606  string_t sWhat,
1607  size_type nEnd) const noexcept
1608 {
1609  return find_last_of(sWhat.cbegin(), sWhat.cend(), nEnd);
1610 }
1611 
1612 template<class char_t, class traits_t>
1613 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find_first_not_of(
1614  value_type chSymbol,
1615  size_type nBegin) const noexcept
1616 {
1617  return _find(
1618  nBegin,
1619  npos,
1620  [chSymbol](const_pointer pCurrentChar)
1621  {
1622  return *pCurrentChar != chSymbol;
1623  });
1624 }
1625 
1626 template<class char_t, class traits_t>
1627 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find_first_not_of(
1628  const_pointer pszWhat,
1629  size_type nBegin,
1630  size_type nWhatSize) const noexcept
1631 {
1632  if (pszWhat)
1633  {
1634  return _find_first_not_of(
1635  pszWhat,
1636  pszWhat + nWhatSize,
1637  nBegin,
1638  [](const_pointer pChar)
1639  {
1640  return pChar + 1;
1641  });
1642  }
1643  else
1644  {
1645  return npos;
1646  }
1647 }
1648 
1649 template<class char_t, class traits_t>
1650 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find_first_not_of(
1651  const_pointer pszWhat,
1652  size_type nBegin) const noexcept
1653 {
1654  if (pszWhat)
1655  {
1656  return _find_first_not_of(
1657  pszWhat,
1658  static_cast<const_pointer>(nullptr),
1659  nBegin,
1660  [](const_pointer pChar)
1661  {
1662  return *(pChar + 1) != QX_CHAR_PREFIX(value_type, '\0') ? pChar + 1 : nullptr;
1663  });
1664  }
1665  else
1666  {
1667  return npos;
1668  }
1669 }
1670 
1671 template<class char_t, class traits_t>
1672 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find_first_not_of(
1673  const basic_string& sWhat,
1674  size_type nBegin) const noexcept
1675 {
1676  return find_first_not_of(sWhat.data(), nBegin, sWhat.size());
1677 }
1678 
1679 template<class char_t, class traits_t>
1680 template<class fwd_it_t>
1681 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find_first_not_of(
1682  fwd_it_t itWhatBegin,
1683  fwd_it_t itWhatEnd,
1684  size_type nBegin) const noexcept
1685 {
1686  return _find_first_not_of(
1687  itWhatBegin,
1688  itWhatEnd,
1689  nBegin,
1690  [](fwd_it_t itChar)
1691  {
1692  return ++itChar;
1693  });
1694 }
1695 
1696 template<class char_t, class traits_t>
1697 template<range_of_t_c<char_t> string_t>
1698 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find_first_not_of(
1699  string_t sWhat,
1700  size_type nBegin) const noexcept
1701 {
1702  return find_first_not_of(sWhat.cbegin(), sWhat.cend(), nBegin);
1703 }
1704 
1705 template<class char_t, class traits_t>
1706 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find_last_not_of(
1707  value_type chSymbol,
1708  size_type nEnd) const noexcept
1709 {
1710  return _rfind(
1711  npos,
1712  nEnd,
1713  [chSymbol](const_pointer pCurrentChar)
1714  {
1715  return *pCurrentChar != chSymbol;
1716  });
1717 }
1718 
1719 template<class char_t, class traits_t>
1720 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find_last_not_of(
1721  const_pointer pszWhat,
1722  size_type nEnd,
1723  size_type nWhatSize) const noexcept
1724 {
1725  if (pszWhat)
1726  {
1727  return _find_last_not_of(
1728  pszWhat,
1729  pszWhat + nWhatSize,
1730  nEnd,
1731  [](const_pointer pChar)
1732  {
1733  return pChar + 1;
1734  });
1735  }
1736  else
1737  {
1738  return npos;
1739  }
1740 }
1741 
1742 template<class char_t, class traits_t>
1743 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find_last_not_of(
1744  const_pointer pszWhat,
1745  size_type nEnd) const noexcept
1746 {
1747  if (pszWhat)
1748  {
1749  return _find_last_not_of(
1750  pszWhat,
1751  static_cast<const_pointer>(nullptr),
1752  nEnd,
1753  [](const_pointer pChar)
1754  {
1755  return *(pChar + 1) != QX_CHAR_PREFIX(value_type, '\0') ? pChar + 1 : nullptr;
1756  });
1757  }
1758  else
1759  {
1760  return npos;
1761  }
1762 }
1763 
1764 template<class char_t, class traits_t>
1765 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find_last_not_of(
1766  const basic_string& sWhat,
1767  size_type nEnd) const noexcept
1768 {
1769  return find_last_not_of(sWhat.data(), nEnd, sWhat.size());
1770 }
1771 
1772 template<class char_t, class traits_t>
1773 template<class fwd_it_t>
1774 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find_last_not_of(
1775  fwd_it_t itWhatBegin,
1776  fwd_it_t itWhatEnd,
1777  size_type nEnd) const noexcept
1778 {
1779  return _find_last_not_of(
1780  itWhatBegin,
1781  itWhatEnd,
1782  nEnd,
1783  [](fwd_it_t itChar)
1784  {
1785  return ++itChar;
1786  });
1787 }
1788 
1789 template<class char_t, class traits_t>
1790 template<range_of_t_c<char_t> string_t>
1791 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::find_last_not_of(
1792  string_t sWhat,
1793  size_type nEnd) const noexcept
1794 {
1795  return find_last_not_of(sWhat.cbegin(), sWhat.cend(), nEnd);
1796 }
1797 
1798 template<class char_t, class traits_t>
1799 inline typename basic_string<char_t, traits_t>::views basic_string<char_t, traits_t>::split(
1800  const value_type chSeparator) const noexcept
1801 {
1802  views tokens;
1803 
1804  size_type nStart = 0;
1805  size_type nEnd = 0;
1806  while ((nEnd = find(chSeparator, nStart)) != npos)
1807  {
1808  tokens.emplace_back(substr(nStart, nEnd - nStart));
1809  nStart = nEnd;
1810  while (traits_t::compare_n(data() + nStart, &chSeparator, 1) == 0)
1811  ++nStart;
1812  }
1813 
1814  if (nStart != size())
1815  tokens.emplace_back(substr(nStart));
1816 
1817  return std::move(tokens);
1818 }
1819 
1820 template<class char_t, class traits_t>
1821 inline typename basic_string<char_t, traits_t>::views basic_string<char_t, traits_t>::split(
1822  const_pointer pszSeparator,
1823  size_type nSepLen) const noexcept
1824 {
1825  views tokens;
1826 
1827  if (!pszSeparator)
1828  return tokens;
1829 
1830  if (nSepLen == npos)
1831  nSepLen = traits_t::length(pszSeparator);
1832 
1833  size_type nStart = 0;
1834  size_type nEnd = 0;
1835  while ((nEnd = find(pszSeparator, nStart, nSepLen, npos)) != npos)
1836  {
1837  tokens.emplace_back(substr(nStart, nEnd - nStart));
1838  nStart = nEnd;
1839  while (traits_t::compare_n(data() + nStart, pszSeparator, nSepLen) == 0)
1840  nStart += nSepLen;
1841  }
1842 
1843  if (nStart != size())
1844  tokens.emplace_back(substr(nStart));
1845 
1846  return tokens;
1847 }
1848 
1849 template<class char_t, class traits_t>
1850 inline typename basic_string<char_t, traits_t>::views basic_string<char_t, traits_t>::split(
1851  const basic_string& sSeparator) const noexcept
1852 {
1853  return split(sSeparator.data(), sSeparator.size());
1854 }
1855 
1856 template<class char_t, class traits_t>
1857 template<class fwd_it_t>
1858 inline typename basic_string<char_t, traits_t>::views basic_string<char_t, traits_t>::split(
1859  fwd_it_t itSepFirst,
1860  fwd_it_t itSepLast) const noexcept
1861 {
1862  views tokens;
1863 
1864  const size_type nSepLen = static_cast<size_type>(std::distance(itSepFirst, itSepLast));
1865 
1866  size_type nStart = 0;
1867  size_type nEnd = 0;
1868  while ((nEnd = find(itSepFirst, itSepLast, nStart)) != npos)
1869  {
1870  tokens.emplace_back(substr(nStart, nEnd - nStart));
1871  nStart = nEnd + nSepLen;
1872  }
1873  tokens.emplace_back(substr(nStart));
1874 
1875  return tokens;
1876 }
1877 
1878 template<class char_t, class traits_t>
1879 template<range_of_t_c<char_t> string_t>
1880 inline typename basic_string<char_t, traits_t>::views basic_string<char_t, traits_t>::split(
1881  const string_t& sSeparator) const noexcept
1882 {
1883  return split(sSeparator.cbegin(), sSeparator.cend());
1884 }
1885 
1886 template<class char_t, class traits_t>
1887 inline bool basic_string<char_t, traits_t>::starts_with(value_type chSymbol) const noexcept
1888 {
1889  if (!empty())
1890  return at(0) == chSymbol;
1891  else
1892  return false;
1893 }
1894 
1895 template<class char_t, class traits_t>
1896 inline bool basic_string<char_t, traits_t>::starts_with(const_pointer pszStr, size_type nStrSize) const noexcept
1897 {
1898  if (pszStr)
1899  {
1900  if (size_type nThisSize = size(); nThisSize > 0)
1901  {
1902  if (nStrSize == npos)
1903  nStrSize = traits_t::length(pszStr);
1904 
1905  if (nStrSize <= nThisSize)
1906  return traits_t::compare_n(data(), pszStr, nStrSize) == 0;
1907  }
1908  }
1909 
1910  return false;
1911 }
1912 
1913 template<class char_t, class traits_t>
1914 inline bool basic_string<char_t, traits_t>::starts_with(const basic_string& sStr) const noexcept
1915 {
1916  return starts_with(sStr.data(), sStr.size());
1917 }
1918 
1919 template<class char_t, class traits_t>
1920 template<class fwd_it_t>
1921 inline bool basic_string<char_t, traits_t>::starts_with(fwd_it_t itBegin, fwd_it_t itEnd) const noexcept
1922 {
1923  auto nStrSize = std::distance(itBegin, itEnd);
1924  return iter_strcmp(cbegin(), cbegin() + static_cast<size_type>(nStrSize), itBegin, itEnd) == 0;
1925 }
1926 
1927 template<class char_t, class traits_t>
1928 template<range_of_t_c<char_t> string_t>
1929 inline bool basic_string<char_t, traits_t>::starts_with(const string_t& sStr) const noexcept
1930 {
1931  return starts_with(sStr.cbegin(), sStr.cend());
1932 }
1933 
1934 template<class char_t, class traits_t>
1935 inline bool basic_string<char_t, traits_t>::ends_with(value_type chSymbol) const noexcept
1936 {
1937  const size_type nSize = size();
1938  if (nSize > 0)
1939  return at(nSize - 1) == chSymbol;
1940  else
1941  return false;
1942 }
1943 
1944 template<class char_t, class traits_t>
1945 inline bool basic_string<char_t, traits_t>::ends_with(const_pointer pszStr, size_type nStrSize) const noexcept
1946 {
1947  if (pszStr)
1948  {
1949  if (size_type nThisSize = size(); nThisSize > 0)
1950  {
1951  if (nStrSize == npos)
1952  nStrSize = traits_t::length(pszStr);
1953 
1954  if (nStrSize <= nThisSize)
1955  return traits_t::compare_n(data() + nThisSize - nStrSize, pszStr, nStrSize) == 0;
1956  }
1957  }
1958 
1959  return false;
1960 }
1961 
1962 template<class char_t, class traits_t>
1963 inline bool basic_string<char_t, traits_t>::ends_with(const basic_string& sStr) const noexcept
1964 {
1965  return ends_with(sStr.data(), sStr.size());
1966 }
1967 
1968 template<class char_t, class traits_t>
1969 template<class fwd_it_t>
1970 inline bool basic_string<char_t, traits_t>::ends_with(fwd_it_t itBegin, fwd_it_t itEnd) const noexcept
1971 {
1972  return iter_strcmp(cend() - static_cast<size_type>(std::distance(itBegin, itEnd)), cend(), itBegin, itEnd) == 0;
1973 }
1974 
1975 template<class char_t, class traits_t>
1976 template<range_of_t_c<char_t> string_t>
1977 inline bool basic_string<char_t, traits_t>::ends_with(const string_t& sStr) const noexcept
1978 {
1979  return ends_with(sStr.cbegin(), sStr.cend());
1980 }
1981 
1982 template<class char_t, class traits_t>
1983 inline bool basic_string<char_t, traits_t>::contains(value_type chSymbol) const noexcept
1984 {
1985  return find(chSymbol) != npos;
1986 }
1987 
1988 template<class char_t, class traits_t>
1989 inline bool basic_string<char_t, traits_t>::contains(const_pointer pszStr, size_type nStrSize) const noexcept
1990 {
1991  return find(pszStr, 0, nStrSize) != npos;
1992 }
1993 
1994 template<class char_t, class traits_t>
1995 inline bool basic_string<char_t, traits_t>::contains(const basic_string& sStr) const noexcept
1996 {
1997  return find(sStr) != npos;
1998 }
1999 
2000 template<class char_t, class traits_t>
2001 template<class fwd_it_t>
2002 inline bool basic_string<char_t, traits_t>::contains(fwd_it_t itBegin, fwd_it_t itEnd) const noexcept
2003 {
2004  return find(itBegin, itEnd) != npos;
2005 }
2006 
2007 template<class char_t, class traits_t>
2008 template<range_of_t_c<char_t> string_t>
2009 inline bool basic_string<char_t, traits_t>::contains(const string_t& sStr) const noexcept
2010 {
2011  return find(sStr) != npos;
2012 }
2013 
2014 template<class char_t, class traits_t>
2015 inline basic_string<char_t, traits_t>& basic_string<char_t, traits_t>::operator=(const_pointer pszSource) noexcept
2016 {
2017  assign(pszSource);
2018  return *this;
2019 }
2020 
2021 template<class char_t, class traits_t>
2022 inline basic_string<char_t, traits_t>& basic_string<char_t, traits_t>::operator=(basic_string&& sStr) noexcept
2023 {
2024  assign(std::move(sStr));
2025  return *this;
2026 }
2027 
2028 template<class char_t, class traits_t>
2029 inline basic_string<char_t, traits_t>& basic_string<char_t, traits_t>::operator=(const basic_string& sStr) noexcept
2030 {
2031  assign(sStr);
2032  return *this;
2033 }
2034 
2035 template<class char_t, class traits_t>
2036 template<range_of_t_c<char_t> string_t>
2037 inline basic_string<char_t, traits_t>& basic_string<char_t, traits_t>::operator=(const string_t& sStr) noexcept
2038 {
2039  assign(sStr);
2040  return *this;
2041 }
2042 
2043 template<class char_t, class traits_t>
2044 inline basic_string<char_t, traits_t>& basic_string<char_t, traits_t>::operator+=(value_type chSymbol) noexcept
2045 {
2046  append(&chSymbol, 1);
2047  return *this;
2048 }
2049 
2050 template<class char_t, class traits_t>
2051 inline basic_string<char_t, traits_t>& basic_string<char_t, traits_t>::operator+=(const_pointer pszSource) noexcept
2052 {
2053  if (pszSource)
2054  append(pszSource, traits_t::length(pszSource));
2055 
2056  return *this;
2057 }
2058 
2059 template<class char_t, class traits_t>
2060 inline basic_string<char_t, traits_t>& basic_string<char_t, traits_t>::operator+=(const basic_string& sStr) noexcept
2061 {
2062  append(sStr.data(), sStr.size());
2063  return *this;
2064 }
2065 
2066 template<class char_t, class traits_t>
2067 template<range_of_t_c<char_t> string_t>
2068 inline basic_string<char_t, traits_t>& basic_string<char_t, traits_t>::operator+=(const string_t& sStr) noexcept
2069 {
2070  append(sStr.cbegin(), sStr.cend());
2071  return *this;
2072 }
2073 
2074 template<class char_t, class traits_t>
2075 inline bool basic_string<char_t, traits_t>::operator==(value_type chSymbol) const noexcept
2076 {
2077  return size() == 1 && at(0) == chSymbol;
2078 }
2079 
2080 template<class char_t, class traits_t>
2081 inline bool basic_string<char_t, traits_t>::operator==(const_pointer pszSource) const noexcept
2082 {
2083  return compare(pszSource) == 0;
2084 }
2085 
2086 template<class char_t, class traits_t>
2087 inline bool basic_string<char_t, traits_t>::operator==(const basic_string& sStr) const noexcept
2088 {
2089  return size() == sStr.size() && compare(sStr.data(), sStr.size()) == 0;
2090 }
2091 
2092 template<class char_t, class traits_t>
2093 template<range_of_t_c<char_t> string_t>
2094 inline bool basic_string<char_t, traits_t>::operator==(const string_t& sStr) const noexcept
2095 {
2096  return iter_strcmp(cbegin(), cend(), sStr.cbegin(), sStr.cend()) == 0;
2097 }
2098 
2099 template<class char_t, class traits_t>
2100 inline bool basic_string<char_t, traits_t>::operator!=(value_type chSymbol) const noexcept
2101 {
2102  return !operator==(chSymbol);
2103 }
2104 
2105 template<class char_t, class traits_t>
2106 inline bool basic_string<char_t, traits_t>::operator!=(const_pointer pszSource) const noexcept
2107 {
2108  return !operator==(pszSource);
2109 }
2110 
2111 template<class char_t, class traits_t>
2112 inline bool basic_string<char_t, traits_t>::operator!=(const basic_string& sStr) const noexcept
2113 {
2114  return !operator==(sStr);
2115 }
2116 
2117 template<class char_t, class traits_t>
2118 template<range_of_t_c<char_t> string_t>
2119 inline bool basic_string<char_t, traits_t>::operator!=(const string_t& sStr) const noexcept
2120 {
2121  return !operator==(sStr);
2122 }
2123 
2124 template<class char_t, class traits_t>
2125 inline bool basic_string<char_t, traits_t>::operator<(value_type chSymbol) const noexcept
2126 {
2127  return compare(&chSymbol, 1) < 0;
2128 }
2129 
2130 template<class char_t, class traits_t>
2131 inline bool basic_string<char_t, traits_t>::operator<(const_pointer pszSource) const noexcept
2132 {
2133  return compare(pszSource) < 0;
2134 }
2135 
2136 template<class char_t, class traits_t>
2137 inline bool basic_string<char_t, traits_t>::operator<(const basic_string& sStr) const noexcept
2138 {
2139  return compare(sStr.data(), sStr.size()) < 0;
2140 }
2141 
2142 template<class char_t, class traits_t>
2143 template<range_of_t_c<char_t> string_t>
2144 inline bool basic_string<char_t, traits_t>::operator<(const string_t& sStr) const noexcept
2145 {
2146  return iter_strcmp(cbegin(), cend(), sStr.cbegin(), sStr.cend()) < 0;
2147 }
2148 
2149 template<class char_t, class traits_t>
2150 inline bool basic_string<char_t, traits_t>::operator<=(value_type chSymbol) const noexcept
2151 {
2152  return compare(&chSymbol, 1) <= 0;
2153 }
2154 
2155 template<class char_t, class traits_t>
2156 inline bool basic_string<char_t, traits_t>::operator<=(const_pointer pszSource) const noexcept
2157 {
2158  return compare(pszSource) <= 0;
2159 }
2160 
2161 template<class char_t, class traits_t>
2162 inline bool basic_string<char_t, traits_t>::operator<=(const basic_string& sStr) const noexcept
2163 {
2164  return compare(sStr.data(), sStr.size()) <= 0;
2165 }
2166 
2167 template<class char_t, class traits_t>
2168 template<range_of_t_c<char_t> string_t>
2169 inline bool basic_string<char_t, traits_t>::operator<=(const string_t& sStr) const noexcept
2170 {
2171  return iter_strcmp(cbegin(), cend(), sStr.cbegin(), sStr.cend()) <= 0;
2172 }
2173 
2174 template<class char_t, class traits_t>
2175 inline bool basic_string<char_t, traits_t>::operator>(value_type chSymbol) const noexcept
2176 {
2177  return compare(&chSymbol, 1) > 0;
2178 }
2179 
2180 template<class char_t, class traits_t>
2181 inline bool basic_string<char_t, traits_t>::operator>(const_pointer pszSource) const noexcept
2182 {
2183  return compare(pszSource) > 0;
2184 }
2185 
2186 template<class char_t, class traits_t>
2187 inline bool basic_string<char_t, traits_t>::operator>(const basic_string& sStr) const noexcept
2188 {
2189  return compare(sStr.data(), sStr.size()) > 0;
2190 }
2191 
2192 template<class char_t, class traits_t>
2193 template<range_of_t_c<char_t> string_t>
2194 inline bool basic_string<char_t, traits_t>::operator>(const string_t& sStr) const noexcept
2195 {
2196  return iter_strcmp(cbegin(), cend(), sStr.cbegin(), sStr.cend()) > 0;
2197 }
2198 
2199 template<class char_t, class traits_t>
2200 inline bool basic_string<char_t, traits_t>::operator>=(value_type chSymbol) const noexcept
2201 {
2202  return compare(&chSymbol, 1) >= 0;
2203 }
2204 
2205 template<class char_t, class traits_t>
2206 inline bool basic_string<char_t, traits_t>::operator>=(const_pointer pszSource) const noexcept
2207 {
2208  return compare(pszSource) >= 0;
2209 }
2210 
2211 template<class char_t, class traits_t>
2212 inline bool basic_string<char_t, traits_t>::operator>=(const basic_string& sStr) const noexcept
2213 {
2214  return compare(sStr.data(), sStr.size()) >= 0;
2215 }
2216 
2217 template<class char_t, class traits_t>
2218 template<range_of_t_c<char_t> string_t>
2219 inline bool basic_string<char_t, traits_t>::operator>=(const string_t& sStr) const noexcept
2220 {
2221  return iter_strcmp(cbegin(), cend(), sStr.cbegin(), sStr.cend()) >= 0;
2222 }
2223 
2224 template<class char_t, class traits_t>
2225 inline typename basic_string<char_t, traits_t>::reference basic_string<char_t, traits_t>::operator[](
2226  size_type nSymbol) noexcept
2227 {
2228  return at(nSymbol);
2229 }
2230 
2231 template<class char_t, class traits_t>
2232 inline typename basic_string<char_t, traits_t>::const_reference basic_string<char_t, traits_t>::operator[](
2233  size_type nSymbol) const noexcept
2234 {
2235  return at(nSymbol);
2236 }
2237 
2238 template<class char_t, class traits_t>
2239 inline basic_string<char_t, traits_t>::operator std::basic_string_view<
2240  typename basic_string<char_t, traits_t>::value_type>() const noexcept
2241 {
2242  return string_view(data(), size());
2243 }
2244 
2245 template<class char_t, class traits_t>
2246 inline basic_string<char_t, traits_t>::operator bool() const noexcept
2247 {
2248  return !empty();
2249 }
2250 
2251 template<class char_t, class traits_t>
2252 inline bool basic_string<char_t, traits_t>::_resize(size_type nSymbols, string_resize_type eType) noexcept
2253 {
2254  const bool bRet =
2255  m_Data.resize(nSymbols, eType == string_resize_type::shrink_to_fit ? 0 : traits_t::align(), eType);
2256 
2257  if (bRet && eType == string_resize_type::common)
2258  at(nSymbols) = QX_CHAR_PREFIX(typename traits_t::value_type, '\0');
2259 
2260  return bRet;
2261 }
2262 
2263 template<class char_t, class traits_t>
2264 template<class searcher_t>
2265 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::_trim_left(
2266  const searcher_t& searcher) noexcept
2267 {
2268  size_type nSymbols = 0;
2269 
2270  for (size_type i = 0; i < size(); ++i)
2271  {
2272  if (searcher(at(i)))
2273  ++nSymbols;
2274  else
2275  break;
2276  }
2277 
2278  erase(0, nSymbols);
2279  return nSymbols;
2280 }
2281 
2282 template<class char_t, class traits_t>
2283 template<class searcher_t>
2284 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::_trim_right(
2285  const searcher_t& searcher) noexcept
2286 {
2287  size_type nSymbols = 0;
2288  size_type nSize = size();
2289 
2290  for (size_type i = nSize - 1; i != std::numeric_limits<size_type>::max(); --i)
2291  {
2292  if (searcher(at(i)))
2293  ++nSymbols;
2294  else
2295  break;
2296  }
2297 
2298  erase(nSize - nSymbols, nSymbols);
2299  return nSymbols;
2300 }
2301 
2302 template<class char_t, class traits_t>
2303 template<class searcher_t>
2304 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::_trim(
2305  const searcher_t& searcher) noexcept
2306 {
2307  const size_type nSize = size();
2308  size_type nStartPos = 0;
2309  size_type nEndPos = nSize;
2310 
2311  while (nStartPos < nSize && searcher(at(nStartPos)))
2312  ++nStartPos;
2313 
2314  while (nEndPos > nStartPos && searcher(at(nEndPos - 1)))
2315  --nEndPos;
2316 
2317  size_type nNewSize = nEndPos - nStartPos;
2318 
2319  std::memmove(data(), data() + nStartPos, nNewSize * sizeof(value_type));
2320 
2321  _resize(nNewSize);
2322  return nSize - nNewSize;
2323 }
2324 
2325 template<class char_t, class traits_t>
2326 template<class comparator_t>
2327 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::_find(
2328  size_type nBegin,
2329  size_type nEnd,
2330  const comparator_t& comparator) const noexcept
2331 {
2332  if (nEnd == npos)
2333  nEnd = size();
2334 
2335  const_pointer pData = data();
2336  const_pointer pCurrentChar = pData + nBegin;
2337  const_pointer pEnd = pData + nEnd;
2338 
2339  while (pCurrentChar < pEnd)
2340  {
2341  if (comparator(pCurrentChar))
2342  return static_cast<size_type>(pCurrentChar - pData);
2343  else
2344  ++pCurrentChar;
2345  }
2346 
2347  return npos;
2348 }
2349 
2350 template<class char_t, class traits_t>
2351 template<class comparator_t>
2352 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::_rfind(
2353  size_type nBegin,
2354  size_type nEnd,
2355  const comparator_t& comparator) const noexcept
2356 {
2357  if (nBegin == npos)
2358  nBegin = size() - 1;
2359 
2360  const_pointer pData = data();
2361  const_pointer pCurrentChar = pData + nBegin;
2362  const_pointer pEnd = pData + nEnd;
2363 
2364  while (pCurrentChar >= pEnd)
2365  {
2366  if (comparator(pCurrentChar))
2367  return static_cast<size_type>(pCurrentChar - pData);
2368  else
2369  --pCurrentChar;
2370  }
2371 
2372  return npos;
2373 }
2374 
2375 template<class char_t, class traits_t>
2376 template<class incrementer_t, class fwd_it_t>
2377 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::_find_first_of(
2378  fwd_it_t itBegin,
2379  fwd_it_t itEnd,
2380  size_type nBegin,
2381  const incrementer_t& incrementer) const noexcept
2382 {
2383  for (size_type i = nBegin; i < size(); ++i)
2384  for (fwd_it_t it = itBegin; it != itEnd; it = incrementer(it))
2385  if (*it == at(i))
2386  return i;
2387 
2388  return npos;
2389 }
2390 
2391 template<class char_t, class traits_t>
2392 template<class incrementer_t, class fwd_it_t>
2393 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::_find_last_of(
2394  fwd_it_t itBegin,
2395  fwd_it_t itEnd,
2396  size_type nEnd,
2397  const incrementer_t& incrementer) const noexcept
2398 {
2399  for (size_type i = size() - 1; i != nEnd - 1; --i)
2400  for (fwd_it_t it = itBegin; it != itEnd; it = incrementer(it))
2401  if (*it == at(i))
2402  return i;
2403 
2404  return npos;
2405 }
2406 
2407 template<class char_t, class traits_t>
2408 template<class incrementer_t, class fwd_it_t>
2409 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::_find_first_not_of(
2410  fwd_it_t itBegin,
2411  fwd_it_t itEnd,
2412  size_type nBegin,
2413  const incrementer_t& incrementer) const noexcept
2414 {
2415  for (size_type i = nBegin; i < size(); ++i)
2416  {
2417  bool bFoundOneOf = false;
2418  for (fwd_it_t it = itBegin; !bFoundOneOf && it != itEnd; it = incrementer(it))
2419  bFoundOneOf |= *it == at(i);
2420 
2421  if (!bFoundOneOf)
2422  return i;
2423  }
2424 
2425  return npos;
2426 }
2427 
2428 template<class char_t, class traits_t>
2429 template<class incrementer_t, class fwd_it_t>
2430 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::_find_last_not_of(
2431  fwd_it_t itBegin,
2432  fwd_it_t itEnd,
2433  size_type nEnd,
2434  const incrementer_t& incrementer) const noexcept
2435 {
2436  for (size_type i = size() - 1; i != nEnd - 1; --i)
2437  {
2438  bool bFoundOneOf = false;
2439  for (fwd_it_t it = itBegin; !bFoundOneOf && it != itEnd; it = incrementer(it))
2440  bFoundOneOf |= *it == at(i);
2441 
2442  if (!bFoundOneOf)
2443  return i;
2444  }
2445 
2446  return npos;
2447 }
2448 
2449 template<class char_t, class traits_t>
2450 basic_string<char_t, traits_t> operator+(
2451  const basic_string<char_t, traits_t>& lhs,
2452  const basic_string<char_t, traits_t>& rhs) noexcept
2453 {
2454  basic_string<char_t, traits_t> str(lhs);
2455  str += rhs;
2456  return str;
2457 }
2458 
2459 template<class char_t, class traits_t>
2460 basic_string<char_t, traits_t> operator+(
2461  basic_string<char_t, traits_t>&& lhs,
2462  const basic_string<char_t, traits_t>& rhs) noexcept
2463 {
2464  basic_string<char_t, traits_t> str(std::move(lhs));
2465  str += rhs;
2466  return str;
2467 }
2468 
2469 template<class char_t, class traits_t>
2470 basic_string<char_t, traits_t> operator+(
2471  const basic_string<char_t, traits_t>& lhs,
2472  typename traits_t::const_pointer rhs) noexcept
2473 {
2474  basic_string<char_t, traits_t> str(lhs);
2475  str += rhs;
2476  return str;
2477 }
2478 
2479 template<class char_t, class traits_t>
2480 basic_string<char_t, traits_t> operator+(
2481  basic_string<char_t, traits_t>&& lhs,
2482  typename traits_t::const_pointer rhs) noexcept
2483 {
2484  basic_string<char_t, traits_t> str(std::move(lhs));
2485  str += rhs;
2486  return str;
2487 }
2488 
2489 template<class char_t, class traits_t>
2490 basic_string<char_t, traits_t> operator+(
2491  typename traits_t::const_pointer lhs,
2492  const basic_string<char_t, traits_t>& rhs) noexcept
2493 {
2494  basic_string<char_t, traits_t> str(lhs);
2495  str += rhs;
2496  return str;
2497 }
2498 
2499 template<class char_t, class traits_t>
2500 basic_string<char_t, traits_t> operator+(
2501  const basic_string<char_t, traits_t>& lhs,
2502  typename traits_t::value_type rhs) noexcept
2503 {
2504  basic_string<char_t, traits_t> str(lhs);
2505  str += rhs;
2506  return str;
2507 }
2508 
2509 template<class char_t, class traits_t>
2510 basic_string<char_t, traits_t> operator+(
2511  basic_string<char_t, traits_t>&& lhs,
2512  typename traits_t::value_type rhs) noexcept
2513 {
2514  basic_string<char_t, traits_t> str(std::move(lhs));
2515  str += rhs;
2516  return str;
2517 }
2518 
2519 template<class char_t, class traits_t>
2520 basic_string<char_t, traits_t> operator+(
2521  typename traits_t::value_type lhs,
2522  const basic_string<char_t, traits_t>& rhs) noexcept
2523 {
2524  basic_string<char_t, traits_t> str(&lhs, 1);
2525  str += rhs;
2526  return str;
2527 }
2528 
2529 template<class char_t, class traits_t, range_of_t_c<char_t> string_t>
2530 basic_string<char_t, traits_t> operator+(const basic_string<char_t, traits_t>& lhs, const string_t& rhs) noexcept
2531 {
2532  basic_string<char_t, traits_t> str(lhs);
2533  str += rhs;
2534  return str;
2535 }
2536 
2537 template<class char_t, class traits_t, range_of_t_c<char_t> string_t>
2538 basic_string<char_t, traits_t> operator+(basic_string<char_t, traits_t>&& lhs, const string_t& rhs) noexcept
2539 {
2540  basic_string<char_t, traits_t> str(std::move(lhs));
2541  str += rhs;
2542  return str;
2543 }
2544 
2545 template<class char_t, class traits_t, range_of_t_c<char_t> string_t>
2546 basic_string<char_t, traits_t> operator+(const string_t& lhs, const basic_string<char_t, traits_t>& rhs) noexcept
2547 {
2548  basic_string<char_t, traits_t> str(lhs);
2549  str += rhs;
2550  return str;
2551 }
2552 
2553 /**
2554  @brief Get string size (excluding null terminator)
2555  @retval - string size (excluding null terminator)
2556 **/
2557 template<class char_t, class traits_t>
2558 inline typename basic_string<char_t, traits_t>::size_type basic_string<char_t, traits_t>::size() const noexcept
2559 {
2560  return m_Data.size();
2561 }
2562 
2563 /**
2564  @brief Get c-string
2565  @retval - c-string
2566 **/
2567 template<class char_t, class traits_t>
2568 inline typename basic_string<char_t, traits_t>::pointer basic_string<char_t, traits_t>::data() noexcept
2569 {
2570  return m_Data.data();
2571 }
2572 
2573 /**
2574  @brief Get char at the ind position
2575  @param nIndex - char index
2576  @retval - char value
2577 **/
2578 template<class char_t, class traits_t>
2579 inline typename basic_string<char_t, traits_t>::reference basic_string<char_t, traits_t>::at(size_type nIndex) noexcept
2580 {
2581  return data()[nIndex];
2582 }
2583 
2584 /**
2585  @brief Clear string
2586 **/
2587 template<class char_t, class traits_t>
2588 inline void basic_string<char_t, traits_t>::clear() noexcept
2589 {
2590  assign(QX_STR_PREFIX(typename traits_t::value_type, ""));
2591 }
2592 
2593 } // namespace qx
2594 
2595 
2596 
2597 // -------------------------------------------------------- hash -------------------------------------------------------
2598 
2599 namespace std
2600 {
2601 
2602 template<class char_t, class traits_t>
2603 struct hash<qx::basic_string<char_t, traits_t>>
2604 {
2605  size_t operator()(const qx::basic_string<char_t, traits_t>& str) const noexcept
2606  {
2607  return traits_t::hash_function(str.data(), traits_t::hash_seed(), str.size());
2608  }
2609 };
2610 
2611 // -------------------------------------------------------- swap -------------------------------------------------------
2612 
2613 template<class char_t, class traits_t>
2615 {
2616  lhs.swap(rhs);
2617 }
2618 
2619 // ----------------------------------------------------- formatter -----------------------------------------------------
2620 
2621 template<class char_t, class traits_t>
2622 struct formatter<qx::basic_string<char_t, traits_t>, char_t> : qx::basic_formatter
2623 {
2624  template<class format_context_type>
2625  constexpr auto format(const qx::basic_string<char_t, traits_t>& value, format_context_type& ctx)
2626  {
2627  return format_to(ctx.out(), QX_STR_PREFIX(char_t, "{}"), value.c_str());
2628  }
2629 };
2630 
2631 } // namespace std
2632 
2633 
2634 // ------------------------------------------- istream / ostream overloading -------------------------------------------
2635 
2636 template<class char_t, class traits_t>
2637 qx::details::ostream<char_t>& operator<<(
2638  qx::details::ostream<char_t>& os,
2640 {
2641  os << str.data();
2642  return os;
2643 }
2644 
2645 template<class char_t, class traits_t>
2646 qx::details::istream<char_t>& operator>>(qx::details::istream<char_t>& is, qx::basic_string<char_t, traits_t>& str)
2647 {
2648  typename qx::details::istream<traits_t>::iostate ret_bit = qx::details::istream<traits_t>::goodbit;
2649 
2650  auto try_push_back = [&str, &is, &ret_bit](char_t ch)
2651  {
2652  typename traits_t::size_type nCurrentSize = str.size();
2653  if (str._resize(nCurrentSize + 1))
2654  {
2655  str[nCurrentSize] = ch;
2656  return true;
2657  }
2658  else
2659  {
2660  is.unget();
2661  ret_bit |= qx::details::istream<traits_t>::failbit;
2662  return false;
2663  }
2664  };
2665 
2666  str.clear();
2667  char_t ch;
2668 
2669  // skip all space symbols and add first symbol is any
2670  while (is.get(ch))
2671  {
2672  if (!traits_t::is_space(ch))
2673  {
2674  try_push_back(ch);
2675  break;
2676  }
2677  }
2678 
2679  // add other symbols until space symbol
2680  while (is.get(ch))
2681  {
2682  if (!traits_t::is_space(ch))
2683  {
2684  if (!try_push_back(ch))
2685  break;
2686  }
2687  else
2688  {
2689  is.unget();
2690  break;
2691  }
2692  }
2693 
2694  is.setstate(ret_bit);
2695  return is;
2696 }
String class.
Definition: string.h:64
void push_back(value_type chSymbol) noexcept
Insert char in the end of the string.
Definition: string.inl:541
void erase(iterator itFirst, iterator itLast) noexcept
Erase substring.
Definition: string.inl:553
size_type find(value_type chSymbol, size_type nBegin=0, size_type nEnd=npos) const noexcept
Find substring.
Definition: string.inl:1279
bool remove_prefix(value_type chSymbol) noexcept
Remove string prefix if matches.
Definition: string.inl:984
requires format_acceptable_args< char_t, args_t... > void append_format(format_string_type< args_t... > sFormat, const args_t &... args)
Append the formatted string to the current one.
Definition: string.inl:143
size_type remove_all(value_type chSymbol, size_type nBegin=0, size_type nEnd=npos) noexcept
Remove all occurrences of a substring in a string.
Definition: string.inl:1061
std::optional< to_t > to(const_pointer pszFormat=nullptr) const noexcept
Convert string to specified type.
Definition: string.inl:268
int compare(value_type chSymbol) const noexcept
Performs a binary comparison of the characters.
Definition: string.inl:1241
requires format_acceptable_args< char_t, args_t... > void append_vformat(string_view svFormat, const args_t &... args)
Append the formatted string to the current one.
Definition: string.inl:172
size_type find_last_of(value_type chSymbol, size_type nEnd=0) const noexcept
Find last position of character.
Definition: string.inl:1526
requires static format_acceptable_args< char_t, args_t... > basic_string static_vformat(string_view svFormat, const args_t &... args)
Create a string by formatting it with the format string and the args.
void push_front(value_type chSymbol) noexcept
Insert char in the beginning of the string.
Definition: string.inl:547
size_type capacity() const noexcept
Get allocated memory size (including null terminator)
Definition: string.inl:254
value_type pop_back() noexcept
Erase last char and return it.
Definition: string.inl:591
size_type copy(pointer pDest, size_type nCount, size_type nPos=0) const noexcept
Copies a substring [nPos, nPos + nCount) to character string pointed to by pDest.
Definition: string.inl:315
views split(const value_type chSeparator) const noexcept
Split string by separator.
Definition: string.inl:1799
size_type rfind(value_type chSymbol, size_type nBegin=npos, size_type nEnd=0) const noexcept
Find substring (reverse direction)
Definition: string.inl:1359
size_type length() const noexcept
Get string length.
Definition: string.inl:242
void free() noexcept
Clear string and free allocated memory.
Definition: string.inl:202
void from(const from_t &data)
Construct string from custom type.
Definition: string.inl:333
size_type trim_right() noexcept
Trim the string to the right (whitespace characters)
Definition: string.inl:707
void assign(size_type nSymbols, value_type chSymbol) noexcept
Assign by filling.
Definition: string.inl:64
size_type replace_all(const find_string_t &sFind, const replace_string_t &sReplace, size_type nBegin=0, size_type nEnd=npos) noexcept
Replace all occurrences of sFind with sReplace.
Definition: string.inl:1221
bool ends_with(value_type chSymbol) const noexcept
Check if current string ends with char.
Definition: string.inl:1935
void swap(basic_string &sOther) noexcept
Swap this str and other.
Definition: string.inl:179
value_type pop_front() noexcept
Erase first char and return it.
Definition: string.inl:599
void append(value_type chSymbol) noexcept
Append char.
Definition: string.inl:372
bool contains(value_type chSymbol) const noexcept
Check if string contains char.
Definition: string.inl:1983
size_type trim() noexcept
Trim the string to the both sides (whitespace characters)
Definition: string.inl:807
void to_upper() noexcept
Convert string to uppercase.
Definition: string.inl:223
void shrink_to_fit() noexcept
Fit allocated size to string's actual size.
Definition: string.inl:195
static basic_string static_from(const from_t &data)
Construct string from custom type and get it.
requires static format_acceptable_args< char_t, args_t... > basic_string static_format(format_string_type< args_t... > sFormat, const args_t &... args)
Create a string by formatting it with the format string and the args.
size_type replace(const find_string_t &sFind, const replace_string_t &sReplace, size_type nBegin=0, size_type nEnd=npos) noexcept
Replace first occurrence of sFind with sReplace.
Definition: string.inl:1145
size_type find_last_not_of(value_type chSymbol, size_type nEnd=0) const noexcept
Finds the last character not equal to chSymbol.
Definition: string.inl:1706
size_type trim_left() noexcept
Trim the string to the left (whitespace characters)
Definition: string.inl:607
size_type reserve(size_type nCapacity) noexcept
Reserve memory for the string.
Definition: string.inl:185
string_view substr(size_type nPos, size_type nSymbols=npos) const noexcept
Get substring.
Definition: string.inl:208
void to_lower() noexcept
Convert string to lowercase.
Definition: string.inl:216
requires format_acceptable_args< char_t, args_t... > void format(format_string_type< args_t... > sFormat, const args_t &... args)
Clear the string and format it with the format string and the args.
Definition: string.inl:125
size_type insert(size_type nPos, value_type chSymbol) noexcept
Insert substring.
Definition: string.inl:412
bool remove_suffix(value_type chSymbol) noexcept
Remove string suffix if matches.
Definition: string.inl:1017
const_pointer c_str() const noexcept
Get pointer to string zero terminated.
Definition: string.inl:248
size_type remove(value_type chSymbol, size_type nBegin=0, size_type nEnd=npos) noexcept
Remove the first occurrence of a substring in a string.
Definition: string.inl:907
static constexpr size_type max_size() noexcept
Get the theoretical maximum of string size.
Definition: string.inl:260
bool starts_with(value_type chSymbol) const noexcept
Check if current string starts with char.
Definition: string.inl:1887
value_type front() const noexcept
Get first char of the string.
Definition: string.inl:230
value_type back() const noexcept
Get last char of the string.
Definition: string.inl:236
size_type find_first_of(value_type chSymbol, size_type nBegin=0) const noexcept
Find first position of character.
Definition: string.inl:1439
requires format_acceptable_args< char_t, args_t... > void vformat(string_view svFormat, const args_t &... args)
Clear the string and format it with the format string and the args.
Definition: string.inl:151
size_type find_first_not_of(value_type chSymbol, size_type nBegin=0) const noexcept
Finds the first character not equal to chSymbol.
Definition: string.inl:1613
Const random access iterator type.
Definition: iterator.h:74
Non-const random access iterator type.
Definition: iterator.h:35
@ append
append all
#define QX_CHAR_PREFIX(value_t, ch)
Chose witch of prefixes add to char : L or none.
Definition: string_utils.h:257
constexpr int iter_strcmp(fwd_it_1_t itBegin1, fwd_it_1_t itEnd1, fwd_it_2_t itBegin2, fwd_it_2_t itEnd2) noexcept
Compares string 1 with string 2.
Definition: string_utils.h:150
#define QX_STR_PREFIX(value_t, str)
Chose witch of prefixes add to string : L or none.
Definition: string_utils.h:249