std:: forward_like
|
Défini dans l'en-tête
<utility>
|
||
|
template
<
class
T,
class
U
>
constexpr auto && forward_like ( U && x ) noexcept ; |
(depuis C++23) | |
Retourne une référence à
x
qui possède des propriétés similaires à
T&&
.
Le type de retour est déterminé comme suit :
- Si std:: remove_reference_t < T > est un type qualifié const, alors le type référencé du type de retour est const std:: remove_reference_t < U > . Sinon, le type référencé est std:: remove_reference_t < U > .
-
Si
T&&est un type de référence lvalue, alors le type de retour est également un type de référence lvalue. Sinon, le type de retour est un type de référence rvalue.
Si
T
n'est pas un
type référençable
, le programme est mal formé.
Table des matières |
Paramètres
| x | - |
une valeur doit être transmise comme le type
T
|
Valeur de retour
Une référence à x du type déterminé comme ci-dessus.
Notes
Comme
std::forward
,
std::move
, et
std::as_const
,
std::forward_like
est un transtypage qui n'influence que la
catégorie de valeur
d'une expression, ou ajoute potentiellement une qualification const.
Lorsque
m
est un membre réel et donc
o.
m
une expression valide, cela s'écrit généralement
std::
forward
<
decltype
(
o
)
>
(
o
)
.
m
dans le code C++20.
Cela conduit à trois modèles possibles, appelés merge , tuple , et language .
-
merge
: fusionner les
const
qualifiers, et adopter la catégorie de valeur du
Owner. -
tuple
: ce que fait
std
::
get
<
0
>
(
Owner
)
, en supposant que
Ownerest un std:: tuple < Member > . - language : ce que fait std:: forward < decltype ( Owner ) > ( o ) . m .
Le scénario principal que
std::forward_like
adresse est l'adaptation d'objets « distants ». Ni le
tuple
ni le
langage
ne traitent correctement ce cas d'usage principal, c'est pourquoi le modèle de
fusion
est utilisé pour
std::forward_like
.
| Macro de test de fonctionnalité | Valeur | Std | Fonctionnalité |
|---|---|---|---|
__cpp_lib_forward_like
|
202207L
|
(C++23) |
std::forward_like
|
Implémentation possible
template<class T, class U> constexpr auto&& forward_like(U&& x) noexcept { constexpr bool is_adding_const = std::is_const_v<std::remove_reference_t<T>>; if constexpr (std::is_lvalue_reference_v<T&&>) { if constexpr (is_adding_const) return std::as_const(x); else return static_cast<U&>(x); } else { if constexpr (is_adding_const) return std::move(std::as_const(x)); else return std::move(x); } } |
Exemple
#include <cstddef> #include <iostream> #include <memory> #include <optional> #include <type_traits> #include <utility> #include <vector> struct TypeTeller { void operator()(this auto&& self) { using SelfType = decltype(self); using UnrefSelfType = std::remove_reference_t<SelfType>; if constexpr (std::is_lvalue_reference_v<SelfType>) { if constexpr (std::is_const_v<UnrefSelfType>) std::cout << "const lvalue\n"; else std::cout << "lvalue mutable\n"; } else { if constexpr (std::is_const_v<UnrefSelfType>) std::cout << "const rvalue\n"; else std::cout << "valeur mutable rvalue\n"; } } }; struct FarStates { std::unique_ptr<TypeTeller> ptr; std::optional<TypeTeller> opt; std::vector<TypeTeller> container; auto&& from_opt(this auto&& self) { return std::forward_like<decltype(self)>(self.opt.value()); // Il est acceptable d'utiliser std::forward<decltype(self)>(self).opt.value(), // car std::optional fournit des accesseurs appropriés. } auto&& operator[](this auto&& self, std::size_t i) { return std::forward_like<decltype(self)>(self.container.at(i)); // Il n'est pas très judicieux d'utiliser std::forward(self)[i], car // les conteneurs ne fournissent pas d'accès par indice avec rvalue, bien qu'ils pourraient. } auto&& from_ptr(this auto&& self) { if (!self.ptr) throw std::bad_optional_access{}; return std::forward_like<decltype(self)>(*self.ptr); // Il n'est pas bon d'utiliser *std::forward<decltype(self)>(self).ptr, car // std::unique_ptr<TypeTeller> se déréférence toujours en une lvalue non constante. } }; int main() { FarStates my_state { .ptr{std::make_unique<TypeTeller>()}, .opt{std::in_place, TypeTeller{}}, .container{std::vector<TypeTeller>(1)}, }; my_state.from_ptr()(); my_state.from_opt()(); my_state[0](); std::cout << '\n'; std::as_const(my_state).from_ptr()(); std::as_const(my_state).from_opt()(); std::as_const(my_state)[0](); std::cout << '\n'; std::move(my_state).from_ptr()(); std::move(my_state).from_opt()(); std::move(my_state)[0](); std::cout << '\n'; std::move(std::as_const(my_state)).from_ptr()(); std::move(std::as_const(my_state)).from_opt()(); std::move(std::as_const(my_state))[0](); std::cout << '\n'; }
Sortie :
lvalue mutable lvalue mutable lvalue mutable lvalue const lvalue const lvalue const rvalue mutable rvalue mutable rvalue mutable rvalue const rvalue const rvalue const
Voir aussi
|
(C++11)
|
convertit l'argument en une xvalue
(modèle de fonction) |
|
(C++11)
|
transmet un argument de fonction et utilise l'argument de type template pour préserver sa catégorie de valeur
(modèle de fonction) |
|
(C++17)
|
obtient une référence
const
vers son argument
(modèle de fonction) |