Namensräume
Varianten
Aktionen

std::forward_like

Von cppreference.com
< cpp‎ | utility
 
 
Dienstprogramm-Bibliotheken
Sprachunterstützung
Typunterstützung (Basistypen, RTTI)
Bibliotheks-Feature-Test-Makros (C++20)
Programm-Dienstprogramme
Variadische Funktionen
Coroutine-Unterstützung (C++20)
Vertragsunterstützung (C++26)
Drei-Wege-Vergleich
(C++20)
(C++20)(C++20)(C++20)  
(C++20)(C++20)(C++20)

Allgemeine Hilfsmittel
Relationale Operatoren (in C++20 veraltet)
Ganzzahl-Vergleichsfunktionen
(C++20)(C++20)(C++20)  
(C++20)
Swap und Typ-Operationen
(C++14)
(C++11)
(C++11)
forward_like
(C++23)
(C++11)
(C++17)
Gemeinsame Vokabulartypen
(C++11)
(C++17)
(C++17)
(C++17)
(C++11)
(C++17)
(C++23)



 
Definiert in der Header-Datei <utility>
template< class T, class U >
constexpr auto&& forward_like( U&& x ) noexcept;
(seit C++23)

Gibt eine Referenz auf x zurück, die ähnliche Eigenschaften wie T&& hat.

Der Rückgabetyp wird wie folgt bestimmt:

  1. Wenn std::remove_reference_t<T> ein const-qualifizierter Typ ist, dann ist der referenzierte Typ des Rückgabetyps const std::remove_reference_t<U>. Andernfalls ist der referenzierte Typ std::remove_reference_t<U>.
  2. Wenn T&& ein Lvalue-Referenztyp ist, dann ist der Rückgabetyp ebenfalls ein Lvalue-Referenztyp. Andernfalls ist der Rückgabetyp ein Rvalue-Referenztyp.

Wenn T kein referenzierbarer Typ ist, ist das Programm ill-formed.

Inhalt

[bearbeiten] Parameter

x - Ein Wert, der ähnlich wie Typ T weitergeleitet werden soll

[bearbeiten] Rückgabewert

Eine Referenz auf x vom oben bestimmten Typ.

[bearbeiten] Anmerkungen

Ähnlich wie std::forward, std::move und std::as_const ist std::forward_like ein Typ-Cast, der nur die Wertkategorie eines Ausdrucks beeinflusst oder potenziell eine const-Qualifizierung hinzufügt.

Wenn m ein tatsächliches Mitglied ist und somit o.m ein gültiger Ausdruck ist, wird dies in C++20-Code üblicherweise als std::forward<decltype(o)>(o).m geschrieben.

Dies führt zu drei möglichen Modellen, genannt merge, tuple und language.

  • merge: Vereinige die const-Qualifizierer und übernehme die Wertkategorie des Owner.
  • tuple: Was std::get<0>(Owner) tut, unter der Annahme, dass Owner ein std::tuple<Member> ist.
  • language: Was std::forward<decltype(Owner)>(o).m tut.

Das Hauptszenario, für das std::forward_like gedacht ist, ist die Anpassung von "weit entfernten" Objekten. Weder das tuple- noch das language-Szenario tun das Richtige für diesen Hauptanwendungsfall, daher wird das merge-Modell für std::forward_like verwendet.

Feature-Test-Makro Wert Std Feature
__cpp_lib_forward_like 202207L (C++23) std::forward_like

[bearbeiten] Mögliche Implementierung

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);
    }
}

[bearbeiten] Beispiel

#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';
}

Ausgabe

mutable lvalue
mutable lvalue
mutable lvalue
 
const lvalue
const lvalue
const lvalue
 
mutable rvalue
mutable rvalue
mutable rvalue
 
const rvalue
const rvalue
const rvalue

[bearbeiten] Siehe auch

(C++11)
konvertiert das Argument in ein xvalue
(Funktionsvorlage) [edit]
(C++11)
leitet ein Funktionsargument weiter und verwendet das Typ-Template-Argument, um seine Wertkategorie zu erhalten
(Funktionsvorlage) [edit]
(C++17)
erhält eine Referenz auf const zu seinem Argument
(Funktionsvorlage) [edit]