我正在尝试为make_unique
创建和使用std::unique_ptr
,与std::make_shared
described here存在std::shared_ptr
的方式相同。 Herb Sutter mentions可能实现的make_unique
看起来像这样:
template<typename T, typename ...Args>
std::unique_ptr<T> make_unique( Args&& ...args )
{
return std::unique_ptr<T>( new T( std::forward<Args>(args)... ) );
}
它似乎对我不起作用。我正在使用以下示例程序:
// testproject.cpp : Defines the entry point for the console application.
#include "stdafx.h"
#include <iostream>
#include <memory>
#include <utility>
struct A {
A(int&& n) { std::cout << "rvalue overload, n=" << n << "\n"; }
A(int& n) { std::cout << "lvalue overload, n=" << n << "\n"; }
};
template<typename T, typename ...Args>
std::unique_ptr<T> make_unique( Args&& ...args ) {
return std::unique_ptr<T>( new T( std::forward<Args>(args)... ) );
}
int main() {
std::unique_ptr<A> p1 = make_unique<A>(2); // rvalue
int i = 1;
std::unique_ptr<A> p2 = make_unique<A>(i); // lvalue
}
编译器(我正在使用VS2010)给出了以下输出:
1>d:\projects\testproject\testproject\testproject.cpp(15): error C2143: syntax error : missing ',' before '...'
1>d:\projects\testproject\testproject\testproject.cpp(16): error C2065: 'Args' : undeclared identifier
1>d:\projects\testproject\testproject\testproject.cpp(16): error C2988: unrecognizable template declaration/definition
1>d:\projects\testproject\testproject\testproject.cpp(16): error C2059: syntax error : '...'
1>d:\projects\testproject\testproject\testproject.cpp(22): error C2143: syntax error : missing ';' before '{'
1>d:\projects\testproject\testproject\testproject.cpp(22): error C2447: '{' : missing function header (old-style formal list?)
此外,如果您将make_unique
实施替换为以下
template<class T, class U>
std::unique_ptr<T> make_unique(U&& u) {
return std::unique_ptr<T>(new T(std::forward<U>(u)));
}
(取自this示例),它编译并正常工作。
谁能告诉我这是什么问题?在我看来,VS2010在模板声明中遇到了...
的问题,我不知道该怎么办。
答案 0 :(得分:14)
在Visual C ++ 11的发布版本中没有可用的变量模板。但是,您可以使用大量复制/粘贴代码模拟不同数量参数的参数扩展,或者使用Microsoft中使用的相同编译器技巧。自己实现“伪变量”。来自Herb Sutter博客的评论:http://herbsutter.com/gotw/_102/#comment-6428
#include <memory> // brings in TEMPLATE macros.
#define _MAKE_UNIQUE(TEMPLATE_LIST, PADDING_LIST, LIST, COMMA, X1, X2, X3, X4) \
\
template<class T COMMA LIST(_CLASS_TYPE)> \
inline std::unique_ptr<T> make_unique(LIST(_TYPE_REFREF_ARG)) \
{ \
return std::unique_ptr<T>(new T(LIST(_FORWARD_ARG))); \
}
_VARIADIC_EXPAND_0X(_MAKE_UNIQUE, , , , )
#undef _MAKE_UNIQUE
答案 1 :(得分:5)
根据MSDN,Visual C ++ 2010或2012不支持可变参数模板。