Heim >Backend-Entwicklung >C++ >Wie implementiert man Move Capture in C-Lambdas?

Wie implementiert man Move Capture in C-Lambdas?

Linda Hamilton
Linda HamiltonOriginal
2024-12-26 13:35:10791Durchsuche

How to Implement Move Capture in C   Lambdas?

Move Capture in Lambdas

Frage:

Wie implementieren wir auch Move Capture? bekannt als R-Wert-Referenzen, in C 11-Lambdas? Zum Beispiel:

std::unique_ptr<int> myPointer(new int);

std::function<void(void)> example = [std::move(myPointer)] {
   *myPointer = 4;
};

Antwort:

Generalisierte Lambda-Erfassung in C 14

In C 14, generalisiertes Lambda Capture ermöglicht die Bewegungserfassung. Dieser Code ist jetzt gültig:

using namespace std;

auto u = make_unique<some_type>(some, parameters);  
go.run([u = move(u)] { do_something_with(u); }); 

Um Objekte von einem Lambda in eine andere Funktion zu verschieben, machen Sie das Lambda veränderbar:

go.run([u = move(u)] mutable { do_something_with(std::move(u)); });

Workaround für Move Capture in C 11

Eine Hilfsfunktion, make_rref, kann die Bewegungserfassung erleichtern. Seine Implementierung ist wie folgt:

#include <cassert>
#include <memory>
#include <utility>

template <typename T>
struct rref_impl {
    rref_impl() = delete;
    rref_impl(T&& x) : x{std::move(x)} {}
    rref_impl(rref_impl& other)
        : x{std::move(other.x)}, isCopied{true}
    {
        assert(other.isCopied == false);
    }
    rref_impl(rref_impl&& other)
        : x{std::move(other.x)}, isCopied{std::move(other.isCopied)}
    {
    }
    rref_impl& operator=(rref_impl other) = delete;
    T& operator&&() {
        return std::move(x);
    }

private:
    T x;
    bool isCopied = false;
};

template<typename T> rref_impl<T> make_rref(T&& x) {
    return rref_impl<T>{std::move(x)};
}

Ein Testfall für make_rref:

int main() {
    std::unique_ptr<int> p{new int(0)};
    auto rref = make_rref(std::move(p));
    auto lambda =
        [rref]() mutable -> std::unique_ptr<int> { return rref.move(); };
    assert(lambda());
    assert(!lambda());
}

Emulierung der generalisierten Lambda-Erfassung in C 11

Ein weiterer Eine Problemumgehung wird durch die Funktion „capture()“ bereitgestellt:

#include <cassert>
#include <memory>

int main() {
    std::unique_ptr<int> p{new int(0)};
    auto lambda = capture(std::move(p),
        [](std::unique_ptr<int>& p) { return std::move(p); });
    assert(lambda());
    assert(!lambda());
}

capture ist implementiert als folgt:

#include <utility>

template <typename T, typename F>
class capture_impl {
    T x;
    F f;
public:
    capture_impl(T&& x, F&& f)
        : x{std::forward<T>(x)}, f{std::forward<F>(f)} {}

    template <typename ...Ts> auto operator()(Ts&& ...args)
        -> decltype(f(x, std::forward<Ts>(args)...)) {
        return f(x, std::forward<Ts>(args)...);
    }

    template <typename ...Ts> auto operator()(Ts&& ...args) const
        -> decltype(f(x, std::forward<Ts>(args)...)) {
        return f(x, std::forward<Ts>(args)...);
    }
};

template <typename T, typename F>
capture_impl<T, F> capture(T&& x, F&& f) {
    return capture_impl<T, F>(
        std::forward<T>(x), std::forward<F>(f));
}

Diese Lösung verhindert das Kopieren des Lambda, wenn der erfasste Typ nicht kopierbar ist, wodurch Laufzeitfehler vermieden werden.

Das obige ist der detaillierte Inhalt vonWie implementiert man Move Capture in C-Lambdas?. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Stellungnahme:
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn