【发布时间】:2013-11-22 05:59:05
【问题描述】:
假设我想在 C++11 中创建一个包含 10 个指针的 std::vector,每个指针都指向一个默认构造的类 Foo 实例。这是一种方法:
std::vector<Foo*> foos;
for (int i = 0; i != 10; ++i) {
foos.push_back(new Foo());
}
有避免for循环的惯用方法吗?
【问题讨论】:
假设我想在 C++11 中创建一个包含 10 个指针的 std::vector,每个指针都指向一个默认构造的类 Foo 实例。这是一种方法:
std::vector<Foo*> foos;
for (int i = 0; i != 10; ++i) {
foos.push_back(new Foo());
}
有避免for循环的惯用方法吗?
【问题讨论】:
有避免
for循环的惯用方法吗?
不是真的。但是,您的代码不是惯用的 C++11,因为它使用 new 而没有 unique_ptr。不要那样做。事实上,从 C++14 开始,您应该考虑在普通代码中弃用 new 的常规用法(因为 C++14 引入了 make_unique)。
【讨论】:
如果您想避免显式的for 循环,那么是的,有一种方法。
std::generate_n(std::back_inserter(foos), 10, [] { return new Foo(); });
这看起来很惯用。
好吧,不管是否循环,这几乎是一种选择。但是不再推荐使用原始指针,因为在没有 RAII 的情况下很难避免泄漏它们。根据需要使用智能指针,例如 std::unique_ptr 或 std::shared_ptr。
std::vector<std::unique_ptr<Foo>> foos;
std::generate_n(std::back_inserter(foos), 10, [] { return std::unique_ptr<Foo>(new Foo()); });
在 C++14 中,您可以使用std::make_unique。所以你可以完全放弃new。
希望对您有所帮助。
【讨论】:
new 的情况下初始化智能指针?我错过了什么吗?
new 的情况下做到这一点。
make_unique 相对容易:template<typename T, typename... Args> std::unique_ptr<T> my_make_unique(Args&& args) { return {new T(std::forward<Args>(args)...)}; }。据我所知,C++1y make_unique 在 std::unique_ptr<T[]> 支持下做了一些奇特的事情。
没有人提到标准算法std::generate_n。所以你可以写
const size_t N = 10;
std::vector<foo *> foos;
foos.reserve( N );
std::generate_n( std::back_inserter( foos ), N, [] { return new foo(); };
当然实际上是同一个循环。:)
【讨论】: