Frage:
Wie man C 11-Redewendungen zum Schreiben basierend auf Alexandrescu verwendet Konzept Einfacher ScopeGuard?
Antwort:
Das Folgende ist eine ScopeGuard-Implementierung basierend auf dem C 11-Idiom:
<code class="cpp">namespace RAII { template< typename Lambda > class ScopeGuard { mutable bool committed; Lambda rollbackLambda; public: ScopeGuard( const Lambda& _l) : committed(false) , rollbackLambda(_l) {} template< typename AdquireLambda > ScopeGuard( const AdquireLambda& _al , const Lambda& _l) : committed(false) , rollbackLambda(_l) { _al(); } ~ScopeGuard() { if (!committed) rollbackLambda(); } inline void commit() const { committed = true; } }; template< typename aLambda , typename rLambda> const ScopeGuard< rLambda >& makeScopeGuard( const aLambda& _a , const rLambda& _r) { return ScopeGuard< rLambda >( _a , _r ); } template<typename rLambda> const ScopeGuard< rLambda >& makeScopeGuard(const rLambda& _r) { return ScopeGuard< rLambda >(_r ); } }</code>
Verwendung:
<code class="cpp">void SomeFuncThatShouldBehaveAtomicallyInCaseOfExceptions() { std::vector<int> myVec; std::vector<int> someOtherVec; myVec.push_back(5); //first constructor, adquire happens elsewhere const auto& a = RAII::makeScopeGuard( [&]() { myVec.pop_back(); } ); //sintactically neater, since everything happens in a single line const auto& b = RAII::makeScopeGuard( [&]() { someOtherVec.push_back(42); } , [&]() { someOtherVec.pop_back(); } ); b.commit(); a.commit(); }</code>
Vorteile:
Potenzielle Probleme:
Das obige ist der detaillierte Inhalt vonIm Folgenden finden Sie mehrere englische Titel, die zum Inhalt des Artikels passen: Prägnanter Titel (kurz und direkt): * Wie erstellt man einen einfachen ScopeGuard mit C 11-Redewendungen? * Implementierung eines einfachen ScopeGuard in C 11: Ein praktischer Ansatz * C 11 ScopeGuard: Ein prägnantes und praktisches I. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!