table of contents
std::forward_like(3) | C++ Standard Libary | std::forward_like(3) |
NAME¶
std::forward_like - std::forward_like
Synopsis¶
Defined in header <utility>
template< class T, class U > (since C++23)
[[nodiscard]] constexpr auto&& forward_like( U&& x )
noexcept;
Returns a reference to x which has similar properties to T&&.
The return type is determined as below:
1. If std::remove_reference_t<T> is a const-qualified type, then the
referenced
type of the return type is const std::remove_reference_t<U>. Otherwise,
the
referenced type is std::remove_reference_t<U>.
2. If T&& is an lvalue reference type, then the return type is also
an lvalue
reference type. Otherwise, the return type is an rvalue reference type.
If T is not a referenceable type, the program is ill-formed.
Parameters¶
x - a value needs to be forwarded like type T
Return value¶
A reference to x of the type determined as above.
Notes¶
Like std::forward, std::move, and std::as_const,
std::forward_like is a type cast
that only influences the value category of an expression, or potentially adds
const-qualification.
When m is an actual member and thus o.m a valid expression, this is usually
spelled
as std::forward<decltype(o)>(o).m in C++20 code.
This leads to three possible models, called merge, tuple, and language.
* merge: merge the const qualifiers, and adopt the value category of the
Owner.
* tuple: what std::get<0>(Owner) does, assuming Owner is a
std::tuple<Member>.
* language: what std::forward<decltype(Owner)>(o).m does.
The main scenario that std::forward_like caters to is adapting
“far” objects.
Neither the tuple nor the language scenarios do the right thing for that main
use-case, so the merge model is used for std::forward_like.
Feature-test macro Value Std Feature
__cpp_lib_forward_like 202207L (C++23) std::forward_like
Possible implementation¶
template<class T, class U>
[[nodiscard]] 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);
}
}
Example¶
// Run this code
#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 << "mutable lvalue\n";
}
else
{
if constexpr (std::is_const_v<UnrefSelfType>)
std::cout << "const rvalue\n";
else
std::cout << "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());
// It is OK to use std::forward<decltype(self)>(self).opt.value(),
// because std::optional provides suitable accessors.
}
auto&& operator[](this auto&& self, std::size_t i)
{
return std::forward_like<decltype(self)>(self.container.at(i));
// It is not so good to use std::forward<decltype(self)>(self)[i],
because
// containers do not provide rvalue subscript access, although they could.
}
auto&& from_ptr(this auto&& self)
{
if (!self.ptr)
throw std::bad_optional_access{};
return std::forward_like<decltype(self)>(*self.ptr);
// It is not good to use *std::forward<decltype(self)>(self).ptr,
because
// std::unique_ptr<TypeTeller> always dereferences to a non-const
lvalue.
}
};
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';
}
Output:¶
mutable lvalue
mutable lvalue
mutable lvalue
const lvalue
const lvalue
const lvalue
mutable rvalue
mutable rvalue
mutable rvalue
const rvalue
const rvalue
const rvalue
See also¶
move obtains an rvalue reference
(C++11) (function template)
forward forwards a function argument
(C++11) (function template)
as_const obtains a reference to const to its argument
(C++17) (function template)
2024.06.10 | http://cppreference.com |