创建1个元素集的最佳\最简单\最快的方法是什么? (C ++)

时间:2016-06-01 08:57:35

标签: c++ c++11 boost stl set

Somethimes我需要1个用户类型元素的集合(或任何其他容器)并以这种方式创建它们:

boost::assign::list_of(typeVariable).convert_to_container<std::unordered_set<UserType> >()

有人知道更美丽的方式吗?

PS: 例如,我们有任何业务逻辑API,它搜索元素,并采用类型的集合(或其他容器)进行选择。不同的用户可以访问不同的类型。我们也可以选择任何一种类型进行过滤,在这种情况下,我们将从过滤选项中选择这一种类型。

所以我只想用一行编写代码的简单方法。我目前的版本是:

getElements(filter.type != UNDEFINED
            ? boost::assign::list_of(filter.type).convert_to_container<std::set<UserType> >()
            : std::set<UserType>(allowedTypes.begin(), allowedTypes.end()))

4 个答案:

答案 0 :(得分:4)

std::unordered_set<UsertType> set = {typeVariable}

或者包含一个生成为临时集的完整工作代码:

#include <iostream>
#include <unordered_set>
using namespace std;

int main() {
    /* First (unused in the rest of the code) set */
    std::unordered_set<int> set = {1};

    /* Set generated on the fly, if you don't even want to create
       a variable for it */
    cout << *std::unordered_set<int>({2}).begin() << endl;

    return 0;
}

答案 1 :(得分:4)

不幸的是,在C ++中,您不能简单地使用auto set={2.3};,因为这不会创建集合,而是std::initializer_list。但是,在

的帮助下
template<typename Tp>
inline std::unordered_set<Tp> make_set(Tp const&x)
{ return {x}; }

你可以使用

auto set = make_set(2.3);

无需明确指定set的类型(如另一个答案)。

您可以扩展make_set()以获取相同类型的任意数量的参数(我将此作为练习留给您),但是使用初始化列表添加重载更容易

template<typename Tp>
std::unordered_set<Tp> make_set(std::initializer_list<Tp> const&x)
{ return {x}; }

auto set = make_set({2.3,3.1});

创建一个包含两个元素的std::unordered_set<double>

编辑在我看来,没有纯std解决方案可以避免明确命名集合的类型(std::unordered_set<possibly_templated_user_type>)。因此,make_set()(或类似)是唯一的解决方案,可以说应该由std提供(类似于std::make_unique()std::make_shared())。

答案 2 :(得分:2)

利用指向对象的指针可以用作单元素迭代器:

auto x = f();
std::set<T> xs(&x, &x + 1);

答案 3 :(得分:1)

  

更新请注意,如果getElements函数已经采用明确键入的set<...> const&,您只需编写:

if (...)
    getElements({ filter.type});
else
    getElements({ allowedTypes.begin(), allowedTypes.end() });
     

使用C ++统一初始化和初始化列表语法

假设你有这样的API:

template <typename Container>
void getElements(Container const& xs);

// usage:    
getElements(std::vector<int> {1, 2});
getElements(std::list<std::string> {"1", "2"});

我认为您正在寻找一个额外的界面:

template <typename V>
void getElements(std::initializer_list<V> const& xs) { 
    getElements(std::set<V>(xs));
}

样本

<强> Live On Coliru

#include <set>
#include <vector>
#include <list>
#include <string>
#include <iostream>

template <typename Container>
void getElements(Container const& xs) { 
    std::cout << __PRETTY_FUNCTION__ << " ";
    for(auto& x : xs) std::cout << x << ";";
    std::cout << "\n";
}

template <typename V>
void getElements(std::initializer_list<V> const& xs) { 
    getElements(std::set<V>(xs));
}

int main() {
    getElements(std::vector<int> {1, 2});
    getElements(std::list<std::string> {"1", "2"});

    getElements({1});
    getElements({1, 2});
}

打印

void getElements(const Container&) [with Container = std::vector<int>] 1;2;
void getElements(const Container&) [with Container = std::__cxx11::list<std::__cxx11::basic_string<char> >] 1;2;
void getElements(const Container&) [with Container = std::set<int, std::less<int>, std::allocator<int> >] 1;
void getElements(const Container&) [with Container = std::set<int, std::less<int>, std::allocator<int> >] 1;2;