c++ - 在 C++0x 中模拟 finally block

标签 c++ lambda c++11 try-catch finally

灵感来自 the other topic ,我写了这段代码来模拟 finally block :

#include <cassert>
#include <iostream>

struct base { virtual ~base(){} };

template<typename TLambda>
struct exec : base 
{
   TLambda lambda;
   exec(TLambda l) : lambda(l){}
   ~exec() { lambda(); }
};

class lambda{
    base *pbase;
public:
    template<typename TLambda>
    lambda(TLambda l): pbase(new exec<TLambda>(l)){}
    ~lambda() { delete pbase; }
};

class A{
    int a;
public:
    void start(){
        int a=1;        
        lambda finally = [&]{a=2; std::cout<<"finally executed";}; 
        try{
            assert(a==1);
            //do stuff
        }
        catch(int){
            //do stuff
        }
    }
};
int main() {
    A a;
    a.start();
}

输出(ideone):

finally executed

@Johannes 似乎认为它不完全正确,commented that :

It can crash if the compiler doesn't elide the temporary in the copy initialization, because then it deletes twice with the same pointer value

我想知 Prop 体是怎么回事。帮助我理解问题:-)


编辑:

问题修复为:

class lambda{
    base *pbase;
public:
    template<typename TLambda>
    lambda(TLambda l): pbase(new exec<TLambda>(l)){}
    ~lambda() { delete pbase; }

    lambda(const lambda&)= delete;            //disable copy ctor
    lambda& operator=(const lambda&)= delete; //disable copy assignment
};

然后将其用作:

//direct initialization, no copy-initialization
lambda finally([&]{a=2;  std::cout << "finally executed" << std::endl; }); 

完整代码:http://www.ideone.com/hsX0X

最佳答案

在这个初始化中:

lambda finally = [&]{a=2; std::cout<<"finally executed";};

可以使用 lambda 的隐式定义的复制构造函数。这将只复制原始指针 pbase,然后将多次删除它。

例如

$ g++ -std=c++0x -Wall -Wextra -pedantic -fno-elide-constructors lambdafun.cc 
$ ./a.out 
a.out: lambdafun.cc:29: void A::start(): Assertion `a==1' failed.
finally executedAborted (core dumped)

实际上,您的断言触发掩盖了双重删除问题,但这证明了我强调的崩溃。

$ g++ -std=c++0x -Wall -Wextra -pedantic -fno-elide-constructors -DNDEBUG lambdafun.cc 
$ ./a.out 
Segmentation fault (core dumped)

关于c++ - 在 C++0x 中模拟 finally block ,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6167515/

相关文章:

java - 使用java 8除对象列表中的bigdecimal值?

C++: std::move with rvalue reference 不 move 内容

c++ - 使用 fstream 访问 linux 用户主目录

c++ - 这是 lcov 的错误还是对一个函数中的不同命中计数有意义?

c++ - AutoCAD 实体可以序列化吗?

c++ - 如何创建具有指定数量的元素(相同类型)的 boost::tuple?

java - 验证一个 Map 中的所有键/值对是否存在于另一个 Map 中

c# - 使用 C# 表达式树,我可以创建一个匿名到类型化对象映射吗?

c++11 - C++ - 可变参数模板部分特化

c++ - 隐式生成的赋值运算符应该是 & ref 限定的吗?