c++ - 为提供不同接口(interface)的容器包装容器

标签 c++ c++11 containers wrapper variadic-templates

我想为 STL 容器创建通用的“包装器”,例如:

template<template <typename, typename...> class Container = std::vector >
class ContainerWrapper{
  add();
  size();
  find();
  resize();
  sort(); 
  /**/
}

+迭代器。 我希望成员函数有不同的实现,具体取决于容器提供的方法。 C++ 模板系统足以创建这个吗?仅使用标准(无增强,不干扰预处理器)这是否可能?

我知道如何以困难的方式做到这一点 - 为每个 STL 容器编写模板专门化。但我希望它也能与其他容器一起使用,而且我也在寻找更通用的方法来做到这一点。

另外,这里有什么更好的?继承 Container 还是将 Container 作为组件?

最佳答案

不久前,我为我的一个项目开发了类似的东西。

我提取了一个完整的工作示例(原始代码更复杂)来展示如何使用方法addVal()(调用push_back()), push()insert()push_front()) 向包装容器添加值。

此代码适用于 std::vectorstd::setstd::multisetstd::unordered_setstd::unordered_multisetstd::dequestd::queue std::priority_queuestd::forward_liststd::stack

其他容器(例如 std::array)可能需要不同的 cntWrp 专门化。

我不想解释每一行代码,但是,如果您有疑问,我可以尝试回复(或者,如果您愿意,我可以给您我的 github 项目的链接)。

示例

#include <set>
#include <vector>
#include <iostream>
#include <stdexcept>
#include <type_traits>

class emptyClass
 { };

template <typename ... Ts>
struct funcType;

template <typename T, typename ... Ts>
struct funcType<T, Ts...>
 { using type
      = typename std::conditional<T::result,
         typename T::type, typename funcType<Ts...>::type>::type; };

template <>
struct funcType<>
 { using type = emptyClass; };


#define methodCheck_1(meth)                            \
                                                       \
   class helpMeth_1_##meth {};                         \
                                                       \
   template <typename T, typename A>                   \
   struct isWithMethod_1_##meth                        \
    {                                                  \
      template<typename U>                             \
      static decltype(U().meth(A())) func (U*);        \
                                                       \
      template<typename U>                             \
      static emptyClass func (...);                    \
                                                       \
      static const bool result                         \
         = ! std::is_same<emptyClass,                  \
                decltype(func<T>(nullptr))>::value;    \
                                                       \
      using  type = helpMeth_1_##meth;                 \
    }

methodCheck_1(insert);
methodCheck_1(push);
methodCheck_1(push_back);
methodCheck_1(push_front);

template <typename>
class cntWrp;

template <template <typename ...> class C, typename X, typename ... Xs>
class cntWrp< C<X, Xs...> >
 {
   private:

      using addModeType = typename funcType<
         isWithMethod_1_push_back<C<X, Xs...>, X>,
         isWithMethod_1_insert<C<X, Xs...>, X>,
         isWithMethod_1_push<C<X, Xs...>, X>,
         isWithMethod_1_push_front<C<X, Xs...>, X>>::type;

      static constexpr addModeType  addMode {};

      void addVal (X const & x, helpMeth_1_push_back const)
       { val.push_back(x); }

      void addVal (X const & x, helpMeth_1_push const)
       { val.push(x); }

      void addVal (X const & x, helpMeth_1_insert const)
       { val.insert(x); }

      void addVal (X const & x, helpMeth_1_push_front const)
       { val.push_front(x); }

      void addVal (X const & x, emptyClass const)
       { throw std::runtime_error("cntWr<>::addVal without mode"); }

   public:

      C<X, Xs...> val {};

      cntWrp ()
       { }

      cntWrp (C<X, Xs...> const & v0) : val { v0 }
       { }

      void addVal (X const & x)
       { addVal(x, addMode); }
 };

int main ()
 {
   cntWrp<std::set<int>>  csi;

   csi.addVal(2);
   csi.addVal(7);
   csi.addVal(5);

   std::cout << "set:" << std::endl;

   for ( auto const elem : csi.val )
      std::cout << elem << std::endl;

   cntWrp<std::vector<int>> cvi;

   cvi.addVal(2);
   cvi.addVal(7);
   cvi.addVal(5);

   std::cout << "vector:" << std::endl;

   for ( auto const elem : cvi.val )
      std::cout << elem << std::endl;
 }

关于c++ - 为提供不同接口(interface)的容器包装容器,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43103510/

相关文章:

c++ - C/C++ 从字符缓冲区读取以填充结构

c++ - 这里的 const 修饰符不是不必要的吗?

php - 警告 : session_start(): Cannot find save handler 'redis'

docker - 即使镜像位于公共(public)注册表中,在本地 VM 上运行 `docker stack deploy` 也会导致 "No such image"错误

c++ - 使用 C++98 标准编写的代码可以用新的编译器编译吗?

c++ - C++ 中右值引用的分配和不变性

c++ - 与 C++ unordered_map 的并行性

c++ - 使用 unique_ptr move 语义

docker build 访问主机文件夹

用于使用 MySql 的 C++ 库