如何dllexport从std :: runtime_error派生的类?

Dev*_*lar 14 c++ windows dll exception dllexport

我已经设置了一个库,提供了一个从标准异常派生的异常类:

#include <stdexcept>
#include <string>

class BaseException : public std::runtime_error
{
    public:
        BaseException( std::string const & msg );
};
Run Code Online (Sandbox Code Playgroud)

到现在为止还挺好.在Unix上编译和处理得很好.现在我准备将其编译成Windows DLL:

#ifdef WIN32
#define MY_EXPORT __declspec(dllexport)
#else
#define MY_EXPORT
#endif

#include <stdexcept>
#include <string>

class MY_EXPORT BaseException : public std::runtime_error
{
    public:
        BaseException( std::string const & msg );
};
Run Code Online (Sandbox Code Playgroud)

但是,这给了我警告C4275 : non – DLL-interface class 'std::runtime_error' used as base for DLL-interface class 'BaseException'.

不幸的是,我对微软风格的文档有些过敏:过于冗长,而且不是很重要.它让我完全不知道对我解决问题的实际期望是什么.

你们有人可以开导我吗?我可以删除基类,但然后捕获std::runtime_errorstd::exception不会捕获我的自定义异常类,我非常希望这是可能的.所以...?

Nia*_*all 12

在这种情况下,您可以选择几种方法.

  1. 出口它.
  2. 忽略它.
  3. 在线吧.

重要的是要记住,从dll导出类的"正确"方法是导出整个类,包括基类和成员.出于这个原因,有几种技术,例如CodeProject上的这种技术,它使用"接口"和适当的工厂来创建类(和匹配的销毁).

在这种情况下,这对你来说并不是很有用,尝试导出std::runtime_error可能是更多的努力,并可能在以后引入更大的问题.

这里Microsoft Connect站点(webarchive)中获取,这些错误的家族本质上是噪音;

我建议首先避免这种情况 - 将STL类型放在DLL的界面中会强制您按照STL的规则进行播放(具体来说,您不能混用VC的不同主要版本,并且您的IDL设置必须匹配).但是,有一种解决方法.C4251基本上是噪音,可以沉默......

Stephan T. Lavavej(Micrsoft的C++库的维护者之一).

只要编译器选项在整个项目中保持一致,只需将此警告静音即可.

最后一个选项是定义BaseException内联类,而不是导出它.

根据我的经验,内联选项几乎总是最容易的异常类.


VS2015的C++运行时中的更改导致导出的更改std::exception(它不从运行时导出).

现在,内联选项似乎是最合适的(您的里程可能会有所不同).

class Exception : exception {
public:
    char const* what() const override;
};

inline char const* Exception::what() const {
    /*...*/
};
Run Code Online (Sandbox Code Playgroud)