- algorithm[meta header]
- std[meta namespace]
- function[meta id-type]
- cpp20[meta cpp]
namespace std {
template<class InputIterator1, class InputIterator2, class Cmp>
constexpr auto
lexicographical_compare_three_way(
InputIterator1 first1,
InputIterator1 last1,
InputIterator2 first2,
InputIterator2 last2,
Cmp comp)
-> decltype(comp(*first1, *first2)); // (1)
template<class InputIterator1, class InputIterator2>
constexpr auto
lexicographical_compare_three_way(
InputIterator1 first1,
InputIterator1 last1,
InputIterator2 first2,
InputIterator2 last2); // (2)
}
2つのイテレータ範囲[first1, last1)
と[first2, last2)
を辞書式順序による三方比較によって比較する。
このアルゴリズムは、コンテナのoperator<=>()
の実装で使用される。
- (1) :
decltype(comp(*first1, *first2))
の型は比較カテゴリ型のうちの1つであること。
first1
-- 比較する1つ目のイテレータ範囲の先頭イテレータ。last1
-- 比較する1つ目のイテレータ範囲の終端イテレータ。first2
-- 比較する2つ目のイテレータ範囲の先頭イテレータ。last2
-- 比較する2つ目のイテレータ範囲の終端イテレータ。comp
-- 使用する三方比較をカスタマイズする関数オブジェクト。
まず、N
をmin(last1 - first1, last2 - first2)
、E(n)
をcomp(*(first1 + n), *(first2 + n))
で定義する。
-
(1) : 次のいずれか
E(i) != 0
がtrue
となる[0, N)
内の最小の整数i
について、E(i)
comp
の意味で異なる最初の要素についての三方比較の結果を返す
- そのような
i
が存在しない場合 :(last1 - first1) <=> (last2 - first2)
- 全ての要素が等しいならば、長さを比較する
-
(2) : 以下と等価、すなわち(1)に委譲
return lexicographical_compare_three_way(first1, last1, first2, last2, compare_three_way());
- compare_three_way[link /reference/compare/compare_three_way.md]
戻り値型となる比較カテゴリ型をCat
とすると、
イテレータ範囲[first1, last1)
が、辞書式比較でイテレータ範囲[first2, last2)
より大きい場合はCat::greater
を返し、小さい場合Cat::less
を返し、等しいのならばCat::equivalent
を返す。
「効果」節のN
について
高々N
回のcomp
による比較が行われる。
#include <iostream>
#include <compare>
#include <algorithm>
#include <cctype>
//大文字小文字を同値として扱って比較
auto weak_comp = [](char cl, char cr) -> std::weak_ordering {
char c1, c2;
if (std::isalpha(static_cast<unsigned char>(cl)) && std::isalpha(static_cast<unsigned char>(cr))) {
c1 = std::tolower(cl);
c2 = std::tolower(cr);
} else {
c1 = cl;
c2 = cr;
}
return c1 <=> c2;
};
int main() {
std::string str1 = "abcdefghijklmnopqrstuvwxyz";
std::string str2 = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
std::string str3 = "abcdefghijklm";
std::cout << std::boolalpha;
//カスタマイズした比較による同じ長さのイテレータ範囲の比較
{
auto comp = std::lexicographical_compare_three_way(str1.begin(), str1.end(), str2.begin(), str2.end(), weak_comp);
std::cout << (comp == 0) << std::endl;
}
//デフォルトの比較による異なる長さのイテレータ範囲の比較
{
auto comp = std::lexicographical_compare_three_way(str1.begin(), str1.end(), str3.begin(), str3.end());
std::cout << (comp > 0) << std::endl;
}
}
- weak_ordering[link /reference/compare/weak_ordering.md]
true
true
- C++20
- Clang: ??
- GCC: 10.1 [mark verified]
- Visual C++: ??