【问题标题】:using sfinae to select different method implementations in a clase template使用 sfinae 在类模板中选择不同的方法实现
【发布时间】:2013-09-09 14:03:28
【问题描述】:

我有一个用于 GTK GUI 的简单包装类,如下所示:

template <class T>
class LabeledEntry
{
    string name;
    T var;

    Gtk::HBox hbox;
    Gtk::Label label;
    Gtk::Entry entry;

    public:
    LabeledEntry( string _name, T _var, Gtk::VBox* vbox):
        name( _name ),
        var(_var)
    {
        label.set_text( name.c_str() );
        ostringstream os;
        os << var ;

        entry.set_text( os.str().c_str() );
        hbox.add( label );
        hbox.add( entry );

        vbox->add( hbox);
    }

    T Get()
    {
        string valString( entry.get_text());
        istringstream is( valString);
        is >> noskipws >> var;

        return var;
    }
};

如果T 的类型是string,现在我需要一个针对T Get() 的特殊实现,因为跳过字符串的空格不起作用。所以这里我在方法中需要getline

我找到了很多 std::is__xxx 模板来检查很多属性,比如 is_integral 等等。但我需要直接与给定类型进行比较。有机会吗?

在类中编写这两个实现的语法如何?比如:

class ...
{ 
    std::enable_if( true, XXX_IS_STRING)::type Get()
    {
    }

    std::enable_if ( false, XXX_IS_SRING)::type Get()
    {
    }
};

对不起,我在成员参数列表中使用没有模板参数的 SFINAE 有点困惑。

【问题讨论】:

    标签: c++ templates c++11 sfinae


    【解决方案1】:

    您应该将Get 函数模板化并像这样使用std::enable_if

    #include <type_traits>
    #include <iostream>
    #include <string>
    
    template <class T>
    class LabeledEntry
    {
        // ...
    public:
        template <class U = T>
        typename std::enable_if<std::is_same<U, std::string>::value, U>::type
            Get()
        {
            return {"string"};
        }
    
        template <class U = T>
        typename std::enable_if<!std::is_same<U, std::string>::value, U>::type
            Get()
        {
            return {42};
        }
    };
    
    int main()
    {
        LabeledEntry<std::string> sle;
        std::cout << sle.Get() << std::endl;
    
        LabeledEntry<int> ile;
        std::cout << ile.Get() << std::endl;
        return 0;
    }
    

    【讨论】:

      【解决方案2】:

      如果只有成员函数需要基于类模板类型的特殊实现,那么简单地为成员函数定义一个特化可能比使用 SFINAE 更容易。

      #include <iostream>
      #include <string>
      
      template <class T>
      class LabeledEntry
      {
        // ...
      public:
        T Get()
        {
          return 42;
        }
      };
      
      template <>
      std::string LabeledEntry<std::string>::Get()
      {
        return "string";
      }
      
      int main()
      {
        std::cout << LabeledEntry<std::string>().Get() << "\n"
                  << LabeledEntry<int>().Get() << std::endl;
      }
      

      这会导致以下输出:

      string
      42
      

      【讨论】:

      • 你是对的!有时解决方案比预期的要简单得多 :-) 太好了!
      【解决方案3】:

      但我需要直接与给定类型进行比较。有机会吗?

      使用std::is_same&lt;T, std::string&gt;测试T是否为std::string

      您可以定义一个别名模板来简化:

        template<typename T>
          using is_string = std::is_same<T, std::string>;
      

      对不起,我在成员参数列表中使用没有模板参数的 SFINAE 有点困惑。

      你不能对非模板做 SFINAE,所以如果成员函数不是模板函数你不能使用 SFINAE。

      您的选择包括:

      • 使成员函数成为模板(参见soon's answer

      • 转发到另一个模板函数:

      
          T Get() const
          { return Get_impl( entry.get_text() ); }
      
        private:
          template<typename P>
            using enable_if = typename std::enable_if<P::value>::type;
      
          template<typename U, typename Requires = enable_if<is_string<U>>>
            U Get_impl( const U& u ) const
            { ... }
      
          template<typename U, typename Requires = enable_if<!is_string<U>>>
            U Get_impl( const U& u ) const
            { ... }
      
      • 或定义LabeledEntry&lt;std::string&gt;LabeledEntry 用来进行转换的其他类的模板特化。

      【讨论】:

        【解决方案4】:

        一种方法是在您委派工作的类之外使用函子。这是一个简单的例子,它没有做任何有用的事情,但你应该得到图片:

        template <typename U>
        struct get_functor
        {
            std::string operator()(U const & val) const
            {
                std::stringstream ss;
                ss << val;
                return ss.str();
            }
        };
        
        template <>
        struct get_functor<std::string>
        {
            std::string operator()(std::string const &) const
            {
                return "something else";
            }
        };
        
        template <typename T>
        struct demo
        {
            explicit demo(T val) : val_(val) {}
        
            decltype(get_functor<T>()(std::declval<T>())) get() const
            {
                return get_functor<T>()(val_);
            }
        
            private:
                T const val_;
        };
        

        或者您可以使用enable_if,但您必须创建一个虚拟模板typename U = T 以启用 SFINAE:

        enum class enable_if_helper {};
        
        template <bool C> 
        using enable_if = typename std::enable_if<C, enable_if_helper>::type;
        
        template <bool C>
        using disable_if = typename std::enable_if<!C, enable_if_helper>::type;
        
        template <typename T>
        struct demo
        {
            explicit demo(T val) : val_(val) {}
        
            template <
                typename U = T, 
                enable_if<std::is_same<U, std::string>::value>...
            >
            std::string get() const
            {
                return "something else"; 
            }
        
            template <
                typename U = T,
                disable_if<std::is_same<U, std::string>::value>...
            >
            std::string get() const
            {
                std::stringstream ss;
                ss << val_;
                return ss.str();
            }
        
            private:
                T const val_;
        };
        

        【讨论】:

          猜你喜欢
          • 1970-01-01
          • 1970-01-01
          • 1970-01-01
          • 1970-01-01
          • 2019-01-26
          • 1970-01-01
          • 1970-01-01
          • 1970-01-01
          • 1970-01-01
          相关资源
          最近更新 更多