c++ - 如何在 C++ 中正确实现我自己的异常处理程序

标签 c++ exception

我正在从 C# 转向 C++,所以如果问题是基本问题或有一些误解,请原谅我...

我想构建自己的异常以用于我的 try/catch block 。我需要报告自定义异常代码、自定义异常消息和自定义异常源来源 - 我可能拥有所有或部分这些参数。

所以我构建了那个类:

CommonException.hpp

namespace exceptionhelper
{

    class CommonException : public std::exception {

    public:
        CommonException();
        CommonException(std::string message);
        CommonException(std::string source, std::string message);
        CommonException(int code, std::string source, std::string message);
        virtual ~CommonException();
        const char *what() const throw();


    private:
        int exceptionCode;
        std::string exceptionSource;
        std::string exceptionMessage;
    };

}

和实现:

CommonException.cpp

namespace exceptionhelper {
        CommonException::CommonException() {
            exceptionCode = 0;
            exceptionMessage = "No message.";
            exceptionSource = "No source.";
        }
        CommonException::CommonException(std::string message) {
            exceptionCode = 0;
            exceptionMessage = message;
            exceptionSource = "No source.";
        }
        CommonException::CommonException(std::string source, std::string message) {
            exceptionCode = 0;
            exceptionMessage = message;
            exceptionSource = source;
        }
        CommonException::CommonException(int code, std::string source, std::string message) {
            exceptionCode = code;
            exceptionMessage = message;
            exceptionSource = source;
        }
        CommonException::~CommonException() {
        }
        const char *CommonException::what() const throw()
        {
            std::stringstream s;
            s << "Code    : " << exceptionCode << std::endl;
            s << "Source  : " << exceptionSource << std::endl;
            s << "Message : " << exceptionMessage << std::endl;
            return s.str().c_str();
        }
    }

最后是我的实现:

main ()
{
   try {
        ... code ...

        throw new CommonException(10, "here", "test exception");

   }
   catch (const std::exception &exc)
   {
            std::cerr << "Exception detected:" << std::endl;
            std::cerr << exc.what();
        }
        catch (...)
        {
            std::cerr << "Unknown exception called." << std::endl;
            throw;
        }
}

出于某种原因,我得到了这个结果:

Unknown exception called.
terminate called after throwing an instance of 'linuxcommon::exceptionhelper::CommonException*'
Aborted (core dumped)

问题:

a) 为什么我没有捕捉到我的自定义异常? b) 我很确定有更好的方法来进行这种异常处理,但我还不能弄清楚......

感谢您的帮助...

最佳答案

代码的一些注释。

  1. 您可能希望从 std::runtime_error 派生您的异常类(而不是 std::exception ),因为 std::runtime_error已经提供了带有错误消息字符串的构造函数。 当然你可以在派生类中添加你自己的异常的数据成员。

  2. 您不需要为您的异常类定义一个空主体的虚拟析构函数,因为您没有要执行的显式清理代码。 std::exception有一个虚拟析构函数,这对您的派生异常类来说工作得很好。

  3. 您可以使用更惯用的 C++ 语法在构造函数中初始化您的异常数据成员,例如而不是:

CommonException::CommonException() {
    exceptionCode = 0;
    exceptionMessage = "No message.";
    exceptionSource = "No source.";
}

你可以使用:

CommonException::CommonException()
  : exceptionCode(0), 
    exceptionMessage("No message."), 
    exceptionSource("No source.")
{ }
  1. 如果你传递字符串参数,你仍然可以按值传递,但你可能想要std::move()从值来初始化数据成员,例如而不是:
CommonException::CommonException(std::string source, std::string message) {
    exceptionCode = 0;
    exceptionMessage = message;
    exceptionSource = source;
}

你可以这样做:

CommonException::CommonException(std::string source, std::string message) 
    : exceptionCode(0),
      exceptionMessage(std::move(message)),
      exceptionSource(std::move(source))
{
}
  1. 考虑制作单字符串参数构造函数 explicit ,以防止从字符串到异常的虚假隐式转换:

    explicit CommonException(std::string message);
    
  2. 在当前形式中,您的 what()方法实现可以抛出,因为在 << 上插入操作( std::stringstream )可以抛出:

const char *CommonException::what() const throw()
{
    std::stringstream s;
    s << "Code    : " + exceptionCode << std::endl;
    s << "Source  : " + exceptionSource << std::endl;
    s << "Message : " + exceptionMessage << std::endl;
    return s.str().c_str();
}

因此,删除 throw()规范,使其简单:

const char* CommonException::what() const

(作为旁注,将方法标记为非抛出的现代 C++11 方法是使用 noexcept)。

您可能还想简单地使用 '\n'而不是 std::endl避免过早悲观。

此外,您将在此行中返回一个临时字符串:

 return s.str().c_str();

const char*返回给调用者的指针只会指向调用站点的一些垃圾:这会引入错误。

要解决这个问题,您可能需要考虑添加一个 std::string数据成员,在异常对象构造期间格式化该数据成员内的整个错误消息字符串(即在您的构造函数中 - 您也可以构建一个私有(private)辅助方法来执行此操作,以避免在每个构造函数中重复代码),然后返回 m_str.c_str()来自 what()方法。

如果您从 std::runtime_error 派生异常类,您可以在构建时构建整个错误消息字符串,并将其传递给 std::runtime_error的构造函数。在这种情况下,std::runtime_error::what()会做正确的事,你不需要覆盖 what()在你的异常类中。

例如

// Derive from std::runtime_error instead of std::exception
class CommonException : public std::runtime_error

...

CommonException::CommonException( /* your exception data */ )
    : std::runtime_error(BuildErrorMessage( /* your exception data */ )
{ }

// Private helper method
std::string CommonException::BuildErrorMessage( /* your exception data */ )
{
    // Build your exception message string here,
    // e.g. you can use std::ostringstream here,
    // and just call its str() method
    // to return the whole error message string.
    ...
}

在异常的“客户端”端,您有:

  ... code ...

    throw new CommonException(10, "here", "test exception");

}
catch (const std::exception &exc)

相反,考虑按值 抛出,而不是在堆上动态分配异常对象,例如简单地做:

throw CommonException( /* your exception data*/ );

关于c++ - 如何在 C++ 中正确实现我自己的异常处理程序,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29906737/

相关文章:

java - Android Studio - 在现有的旧项目中启用 native C++ 调试 (card.io Android Source)

c++ - 字符型变量的地址

c++ - 非数据类型模板参数,生成更多特化?

asp.net - 在共享服务器上发布时获取 "System.Security.SecurityException: That assembly does not allow partially trusted callers"

java - 如何修复java中的 "java.net.SocketException"错误

c++ - 在未处理的异常上重新启动应用程序

java - H2:无法从 jar 内的 sql 文件初始化数据库模式

c++ - 没有背景的透明 SDL_Surface

c++ - 如何以静态方式使用另一个 constexpr 数组初始化一个数组

javascript - 尝试将事件从其组件之一传播到自定义 HTML 元素时抛出