向std::exception添加error_info

4

我正在将boost::exception集成到现有代码中。现有的一些代码使用了BOOST_THROW_EXCEPTION,但某些代码可能仍会抛出标准的std::exception

我想在中间捕获点添加error_info。根据文档,如果异常是boost::exception,我可以这样做:

try {
    do_something()
}
catch( boost::exception & e ) {
    e << boost::errinfo_file_name(file_name);
    throw;
}

但这只会将信息添加到boost异常中。我希望也将其添加到std::exception中。最干净的方法是什么?以下是一种方法,但它会导致一些代码重复:

try {
    do_something()
}
catch( boost::exception & e ) {
    e << boost::errinfo_file_name(file_name);
    throw;
}
catch( std::exception & e ) {
    throw enable_error_info(e) << boost::errinfo_file_name(file_name);
}

有没有一种方法相当于“将当前异常作为boost异常给我,或者如果它不是boost异常,则创建一个boost异常”?

编辑boost::enable_error_info()有点像,但返回的是原始异常的副本,这会切掉我捕获的异常的boost::exception部分。举个例子:

int main()
{
    try {
        try {
            BOOST_THROW_EXCEPTION( std::runtime_error( "foo" ) );
        }
        catch( std::exception & e ) {
            std::cerr << e.what() << std::endl; // "foo" 
            if( const char* const* function = boost::get_error_info<boost::throw_function>(e) ) std::cerr << *function << std::endl; // "int main()"
            throw boost::enable_error_info(e) << boost::errinfo_file_name("bar");
        }
    }
    catch( std::exception & e ) {
        std::cerr << e.what() << std::endl; // "std::exception" 
        if( const char* const* function = boost::get_error_info<boost::throw_function>(e) ) std::cerr << *function << std::endl; // NOTHING
    }

    return 0;
}

编辑: 我尝试使用boost::current_exception(), 但它也会导致异常数据被截断。因为多重继承导致的切片,任何复制异常的尝试都会丢失一些数据。这也是为什么文档中说您应该始终使用throw而不是throw e来重新抛出异常的原因。所以,除非必要,我真的不想产生任何复制。

理想情况下,我希望编写以下代码,其中current_exception_as_boost_exception()如果当前异常已经是boost::exception,则返回对其的引用,否则将其传递给boost::enable_error_info的结果。

try {
    do_something()
}
catch( std::exception & e ) {
    throw current_exception_as_boost_exception() << boost::errinfo_file_name(file_name);
}
< p >那么< code >boost::enable_current_exception< /code >是用来做什么的呢?它的目的非常不清楚,而且在任何教程中都没有使用。< /p >

4
您似乎说enable_error_info()能够工作,但它是“过度”的。既然异常通常不应该发生,因此性能并不真正成问题,那么过度的部分有什么关系呢? - John Zwinck
我关心的不是性能问题,而是在复制过程中可能会丢失一些信息的可能性。 boost :: copy_exception()(我假设这就是enable_error_info()使用的东西)可能会在过程中丢失类型信息(请参阅http://stackoverflow.com/questions/9973499/why-do-i-lose-type-information-when-using-boostcopy-exception)。 因此,除非我有一些保证,即enable_error_info()不会删除任何内容,否则我宁愿避免复制。 - user1924406
1个回答

1

这里有一个解决方案可以满足我的需求。但感觉好像在重新发明轮子。难道没有内置的方法来实现同样的事情吗?

struct rethrow
{
    rethrow()
    {
        try{ throw; }
        // Already a boost::exception
        catch( boost::exception& ) {} 
        // Something else. Make it a boost::exception
        catch( ... ) { ptr = boost::current_exception(); } 
    }

    template<class T> 
    rethrow const& operator<<( const T& t ) const
    {
        try
        {
            re();
        }
        catch( boost::exception& e )
        {
            e << t;
        }
        return *this;
    }

    ~rethrow()
    {
        re();
    }

private:
    void re() const
    {
        if( !ptr ) throw;
        else boost::rethrow_exception( ptr );
    }

    boost::exception_ptr ptr; 
};


int main()
{
    try {
        try {
            throw std::runtime_error( "foo" ); // or BOOST_THROW_EXCEPTION( std::runtime_error( "foo" ) );
        }
        catch( std::exception & e ) {
            rethrow() << boost::errinfo_file_name("bar");
        }
    }
    catch( std::exception & e ) {
        std::cerr << __LINE__ << ": caught " << e.what() << std::endl; // "caught foo"
        if( const char* const* function = boost::get_error_info<boost::throw_function>(e) ) std::cerr << __LINE__ << ": throw from " << *function << std::endl; // "throw from int main()" (when using BOOST_THROW_EXCEPTION)
        if( std::string const* fileName = boost::get_error_info<boost::errinfo_file_name>(e) ) std::cerr << __LINE__ << ": trying to open " << *fileName << std::endl; // "trying to open bar"
    }

    return 0;
}

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接