gpt4 book ai didi

c++ - 使用可能抛出 C++ 的函数初始化 const 对象

转载 作者:太空宇宙 更新时间:2023-11-04 13:44:03 26 4
gpt4 key购买 nike

我正在编写一个函数,它调用一个可能抛出异常的函数来初始化一个 const 成员。问题是:如果我将初始化放在 try block 中,它会在超出范围时被删除。如果我将初始化放在 try block 之外,我就无法捕捉到异常。解决方法是:

double functionA()
{
double a;
try
{
a=functionThatMayThrow();
}
catch(std::except&p)
{
std::throw_with_nested(std::runtime_error(__func__);
}
const double b=a;
//More code that uses b and computes x
return x;
}

我想知道是否有更好的方法来做到这一点。像这样的东西:

double functionA()
{
const double b=functionThatMayThrow();
//catch the exception somehow and rethrow with nested information

//Unchanged
//More code that uses b and computes x
return x;
}

也许我只是对问题采取了完全错误的方法。你能给我一些建议吗?任何 boost 或 c++11 相关的解决方案也不错。

最佳答案

要向任何传递的异常添加一些跟踪信息,您1可以这样做:

auto func_A()
-> double
{
try
{
// The complete original function body.

double const a = 2*some_func();
return a/2;
}
catch( ... )
{
throw_with_nested( runtime_error( __func__ ) );
}
}

但是在任何地方重复这一点可能有点乏味且容易出错,因此您可能希望集中逻辑,而不是将函数表达为例如

auto func_A()
-> double
{
return with_trace( __func__, [&]() -> double
{
// The complete original function body.

double const a = 2*some_func();
return a/2;
} );
}

现在这需要一些支持机制,包括向下调用和向上调用,例如

#include <exception>    // std::throw_with_nested
#include <iostream>
#include <stdexcept> // std::runtime_error, std::exception
#include <stdlib.h> // EXIT_FAILURE, EXIT_SUCCESS
#include <string> // std::string
#include <vector> // std::vector

namespace cppx {
using std::runtime_error;
using std::throw_with_nested;

template< class Func >
auto with_trace( char const* const func_name, Func const body )
-> decltype( body() )
{
try
{
return body();
}
catch( ... )
{
throw_with_nested( runtime_error( func_name ) );
}
}

namespace x_io
{
using std::endl;
using std::exception;
using std::cerr; // For client code.
using std::ostream;
using std::rethrow_if_nested;

namespace impl {
void print_on( ostream& stream, exception const& x, bool const is_nested )
{
stream << (is_nested? ">" : "!") << x.what() << endl;
try
{
rethrow_if_nested( x );
}
catch( exception const& nested_x )
{
print_on( stream, nested_x, true );
}
catch( ... )
{
stream << "<unknown exception type>" << endl;
}
}
} // namespace impl

auto operator<<( ostream& stream, exception const& x )
-> ostream&
{
impl::print_on( stream, x, false );
return stream;
}
} // namespace x_io
} // namespace cppx

namespace my {
using std::runtime_error;
using std::string;
using std::vector;
using cppx::with_trace;

bool fail_is_good;

auto some_func()
-> double
{
if( fail_is_good ) { throw runtime_error( "some_func" ); }
return 3.14;
}

auto func_A()
-> double
{
return with_trace( __func__, [&]() -> double
{
double const a = 2*some_func();
return a/2;
} );
}

void cpp_main( vector< string > const& args )
{
using std::cout; using std::endl;
my::fail_is_good = (args.size() > 1);
cout << func_A() << endl;
}
} // namespace my

auto main( int argc, char* argv[] )
-> int
{
using std::exception; using std::string; using std::vector;
try
{
my::cpp_main( vector<string>( argv, argv + argc ) );
return EXIT_SUCCESS;
}
catch( exception const& x )
{
using namespace cppx::x_io;
cerr << x << endl;
}
return EXIT_FAILURE;
}

1) 不幸的是,Visual C++ 12.0 (2013),即使在其 2013 年 11 月的 CTP 化身中,也不支持 std::throw_with_nested。只需实现即可轻松添加此类支持。另外,我记得它不支持 __func__,但该宏也很容易定义。

关于c++ - 使用可能抛出 C++ 的函数初始化 const 对象,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26641736/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com