6

我在头文件的命名空间中有一个类。该类需要模板类型,而我只想使用某些类型。下面显示一个例子。

文件 a.hpp

// a.hpp
namespace a_ns {
template<class T>    
class a {
    // stuff
};
typedef a<double> a_double;
} // end of namespace
// stuff

文件 b.hpp

// b.hpp
#include <a.hpp>
namespace b_ns {
    typedef a_ns::a_double b;
}

文件 main.cpp

// main.cpp
#include "b.hpp"
int main() {
    b_ns::b my_b; // <<<--- I LIKE this!
    a_ns::a<float> my_a_which_is_not_allowed; // <<<--- I DO NOT LIKE THIS THOUGH! D:
}

因此,正如您从这个久违的示例中看到的那样,最终目标是不允许最终用户将 a 声明class afloat类型名,并且只能使用具有特定类型的预定义类,如typedef a<double> a_double;.

我认为上面的这个例子会允许这样做,但是我错了,因为我可以a<float>像上面那样创建一个,因为我包含b.hpp,而它又包含a.hpp! 所以你看到了问题!(希望?)

如果可能的话,可能有一个简单的解决方案。

4

2 回答 2

6

如果您只想能够使用类型别名而不是a直接使用,则可以将其放入用户应该知道不使用的实现命名空间中:

namespace a_ns {

namespace detail {
    template<class T>        
    class a {
        // stuff
    };
}

typedef detail::a<double> a_double;
} // end of namespace

现在任何东西都可以使用a_double,但要a直接使用,你的detail命名空间必须被挖掘,这通常被认为是一件坏事。如果用户决定要这样做,他们已经放弃了避免麻烦,您不应该采取额外的措施来阻止他们伤害自己。

于 2013-08-23T18:15:19.527 回答
0

这是使用 static_assert 的方法

#include <type_traits>
template <typename T>
class X
{
  T i;
  static_assert(!std::is_same<float,T>::value,"Don't use floating point");
};



int main()
{
    X<int> a;
    //X<float> b; fails at compile time
    return 0;
}

只要变量不是 const 或 volatile,这将起作用

于 2013-08-23T18:05:57.007 回答