4

I'm reviewing the boost units library and I am puzzled why the boost::units::unit class has an extra template parameter. Here is the example:

http://www.boost.org/doc/libs/1_57_0/boost/units/unit.hpp

template<class Dim,class System, class Enable>
class unit
{
    public:
        typedef unit<Dim, System>   unit_type;
        typedef unit<Dim,System>    this_type;
        typedef Dim                 dimension_type; 
        typedef System              system_type;

        unit() { }
        unit(const this_type&) { }
        //~unit() { }  

        this_type& operator=(const this_type&) { return *this; }

        // sun will ignore errors resulting from templates
        // instantiated in the return type of a function.
        // Make sure that we get an error anyway by putting.
        // the check in the destructor.
        #ifdef __SUNPRO_CC
        ~unit() {
            BOOST_MPL_ASSERT((detail::check_system<System, Dim>));
            BOOST_MPL_ASSERT((is_dimension_list<Dim>));
        }
        #else
    private:
        BOOST_MPL_ASSERT((detail::check_system<System, Dim>));
        BOOST_MPL_ASSERT((is_dimension_list<Dim>));
        #endif
};

The class is used to add dimensions to dimension system.

typedef unit<pressure_dimension,si::system>      pressure;

What purpose would "Enable" serve in this case?

4

1 回答 1

5

该类模板"units_fwd.hpp".

template<class Dim,class System, class Enable=void> class unit;

在那里我们看到第三个参数默认为void. 我什至想template<class Dim,class System, class=void> class unit;在实现中让它保持无名,但可能存在编译器兼容性或代码标准,这意味着他们给它命名Enable

这种“额外”类型允许对前两种类型进行 SFINAE 测试。如果您创建一个专业化,其中第 3 种类型仅对 的某些子集有效,Dim并且System额外的参数(默认为void)允许您这样做。

您甚至可以将基本实现留空(或者;具有{};不同的效果)并且只进行专门化,使测试失败的参数被视为无效选项。

这是 SFINAE 的一个玩具示例:

template<class T, class=void> struct is_int : std::false_type {};
template<class T> struct is_int< T,
  std::enable_if_t< std::is_same<T, int>{} >
> : std::true_type {};

基础特化继承自false_type. 但是,如果该类型T在下一个特化中通过了我的测试,那么它是首选的,并且结果继承自true_type.

在这种情况下,我们最好只做is_int< int, void >:std::true_type{},但在更复杂的情况下(比如is_integral),我们可以做一个几乎任意的编译时检查,并使用它来启用或禁用特化true_type

于 2015-02-26T20:35:21.133 回答