以下のようなパラメーターラッピングヘルパーコードを書き込もうとしています
#include <type_traits>
#include <string>
struct test{};
namespace ns {
struct test{};
}
template<typename T>
struct arg_wrapper;
template<>
struct arg_wrapper<test>
{
arg_wrapper(test&) {}
};
template<>
struct arg_wrapper<ns::test>
{
arg_wrapper(ns::test&) {}
};
template<>
struct arg_wrapper<std::string>
{
arg_wrapper(const std::string& value) {}
};
template<typename... Args>
void callee(const Args&... args)
{
}
template<typename... Args>
void wrapper_variadic(Args&&... args)
{
callee(arg_wrapper<typename std::decay<Args>::type>(args)...);
}
template<typename Args>
void wrapper_fixed(Args&& args)
{
callee(arg_wrapper<typename std::decay<Args>::type>(args));
}
int main()
{
std::string a("test");
wrapper_variadic(a); // works well
wrapper_variadic(std::string("test")); // compile error
wrapper_variadic(test()); // works well
wrapper_variadic(ns::test()); // compile error
wrapper_fixed(std::string("test")); // works well
wrapper_fixed(test()); // works well
wrapper_fixed(ns::test()); // works well
}
このコードは、名前空間のない型の固定パラメーター関数または左辺値参照で機能しますが、私のコンパイラは、一部の名前空間で右辺値参照パラメーターを使用しようとする試みを拒否します。vc11 ctp バージョンを使用していますが、エラー メッセージがややわかりにくいです (以下を参照)。
1>d:\work\toy\vs2012test\vs2012test\source.cpp(39): error C2065: 'basic_string<char,std::char_traits<char>,std::allocator<char> >' : undeclared identifier
1> d:\work\toy\vs2012test\vs2012test\source.cpp(52) : see reference to function template instantiation 'void wrapper_variadic<std::basic_string<char,std::char_traits<char>,std::allocator<char>>>(std::basic_string<char,std::char_traits<char>,std::allocator<char>> &&)' being compiled
1>d:\work\toy\vs2012test\vs2012test\source.cpp(39): error C2923: 'std::decay' : 'basic_string<char,std::char_traits<char>,std::allocator<char> >' is not a valid template type argument for parameter '_Ty'
1>d:\work\toy\vs2012test\vs2012test\source.cpp(39): error C2955: 'std::decay' : use of class template requires template argument list
1> c:\program files (x86)\microsoft visual studio 11.0\vc\include\type_traits(1620) : see declaration of 'std::decay'
1>d:\work\toy\vs2012test\vs2012test\source.cpp(39): error C2440: '<function-style-cast>' : cannot convert from 'std::basic_string<char,std::char_traits<char>,std::allocator<char>>' to 'arg_wrapper<_If<std::is_array<remove_reference<_Ty>::type>::value,remove_extent<remove_reference<_Ty>::type>::type*,_If<std::is_function<remove_reference<_Ty>::type>::value,add_pointer<remove_reference<_Ty>::type>::type,remove_cv<remove_reference<_Ty>::type>::type>::type>::type>'
1> Source or target has incomplete type
1>d:\work\toy\vs2012test\vs2012test\source.cpp(39): error C2440: '<function-style-cast>' : cannot convert from 'ns::test' to 'arg_wrapper<test>'
1> No constructor could take the source type, or constructor overload resolution was ambiguous
1> d:\work\toy\vs2012test\vs2012test\source.cpp(54) : see reference to function template instantiation 'void wrapper_variadic<ns::test>(ns::test &&)' being compiled
コンパイラのバグのようですが、自信がありません。これはバグですか?そうでない場合、問題を解決するにはどうすればよいですか?