1

我在 Windows Vista 上使用 Visual Studio 2008,并将函数转换为调试 DLL。

编译器错误:

我收到 boost::operator 模板的可访问性错误:

error C2248: 'Field::Integer::Integer' : cannot access protected member declared in class 'Field::Integer'  
c:\program files\boost\boost_1_52_0\boost\operators.hpp(257) : while compiling class template member function 'Field::Integer boost::operator +(Field::Integer,const Field::Integer &)'
1>        c:\program files\boost\boost_1_52_0\boost\operators.hpp(836) : see reference to class template instantiation 'boost::addable1<T,B>' being compiled
1>        with
1>        [
1>            T=Field::Integer,
1>            B=boost::detail::empty_base<Field::Integer>
1>        ]
1>        see reference to class template instantiation 'boost::addable<T>' being compiled
1>        with
1>        [
1>            T=Field::Integer
1>        ]
1>        see reference to class template instantiation Field::Numeric<Value_Type,Descendant_Class>' being compiled
1>        with
1>        [
1>            Value_Type=int,
1>            Descendant_Class=Field::Integer
1>        ]

代码(简化为基本语句):

#ifndef FIELD_INTEGER_HPP
#define FIELD_INTEGER_HPP

#ifdef FIELD_EXPORTS
#define FIELD_API __declspec(dllexport)
#else
#define FIELD_API __declspec(dllimport)
#endif

#include "boost/operators.hpp"

namespace Field
{

template <class Value_Type, class FIELD_API Descendant_Class>
class FIELD_API Numeric
    : public boost::addable<Descendant_Class>,
      public boost::subtractable<Descendant_Class>,
      public boost::multipliable<Descendant_Class>,
      public boost::dividable<Descendant_Class>
{
  public:
                                Numeric(const Value_Type&   new_value = 0);
                                Numeric(const Numeric& fn);
    virtual                     ~Numeric();

    Descendant_Class            operator+=(const Descendant_Class& dc);
    Descendant_Class            operator-=(const Descendant_Class& dc);
    Descendant_Class            operator*=(const Descendant_Class& dc);
    Descendant_Class            operator/=(const Descendant_Class& dc);

    void                        clear_field(void);
    bool                        supports_value_as_string(void) const;
};


class FIELD_API Integer
    : public Field::Numeric<int, Field::Integer>
{
  public:
        //! Destructor
    virtual                     ~Integer();
  protected:
    //! Constructor
                                Integer(const int               new_value);

    //! Copy constructor
                                Integer(const Integer& fui);
};

} // End namespace Field

#endif  // FIELD_INTEGER_HPP

我的目标是将上述代码制作成可导出的调试 DLL 或发布 DLL。
代码在静态库设置中构建时没有错误。

问题:

在上面的代码中,要使其成为 Debug 或 Release DLL(Visual Studio 2008、Windows Vista、32 位)需要进行哪些修改?

我搜索了 web 和 StackOverflow,我只得到了使用模板的结果,没有将类作为模板参数和 DLL 传递。

4

1 回答 1

1

发生错误是因为Numeric继承自boost::addable(以及另外 3 个相关的类)。operator+这将生成签名的非成员函数

Field::Integer boost::operator +(Field::Integer,const Field::Integer &)

它采用左参数按值的原因是为了优化右值引用和复制省略。这需要访问boost::operator+的复制构造函数Integerprotected因此会出现错误。我不明白为什么编译为静态库对你有用,而 DLL 则不行。

对于这样的访问问题,推荐的方法是制作复制构造函数public。不想要Integer作为叶类对我来说似乎是一个设计错误,因为如果这真的是你想要的,为什么你仍然希望能够将它用于加法和其他算术运算?

另一种方法是给予友谊boost::operator+(Integer, Integer const&)。不推荐使用友谊路线,因为它依赖于boost::addable. 通常你会给类授予友谊boost::addable,但它的实现将使用非成员朋友函数operator+而不是成员函数operator+。你不应该让你自己的类Integer依赖于这样的实现细节。

于 2013-05-28T14:56:41.567 回答