正在学习C++11的新特性,用非类型模板写了一个函数包装器,我是这样写的:
#include <iostream>
#include <cstdlib>
#include <string>
#include <functional>
void hello() {
std::cout << "Hello, world!\n";
return;
}
template< std::function<void()> i>
void wrapper() {
i();
}
int main() {
std::function<void()> f = hello;
wrapper<f>();
return 0;
}
在VS2013上编译错误,提示是
“std::function”: 非类型模板参数“i”的类型非法
但是当我将wrapper的定义改成
template<void i()>
void wrapper() {
i();
}
将调用改成wrapper<hello>();
之后编译运行就一切正常了。请问这是什么原因?
另外请问std::function除了能包装匿名函数外,还有什么情况下与函数对象或者普通函数指针表现不同呢?谢谢。
因為非型別模板實參必須是常數表達式。因為模板中的東西都必須是可在編譯期確定的。
http://en.cppreference.com/w/...
wrapper<f>()
中 f是一個變量,它的值是運行期才能決定的。所以模板沒辦法編譯通過,總不能等到運行時再Instantiate出一個wrapper<f>
函數,讓其中的i指向f吧?For pointers to functions, the valid arguments are pointers to functions with linkage (or constant expressions that evaluate to null pointer values).,所以
<void i()>
可以編譯通過:你的寫法,
template < std::function<void()> i >
,這裡的i
明顯是一個變量,而不是類型,如果要聲明類型應該寫成template <typename Func>
。不過如果聲明為類型,wrapper
當然就沒辦法工作了,因為i()
就相當於實例化一個空的std::function
對象,並沒有做任何事情,最終當然就不能得到你想要的效果。一般來說,你應該這樣實現
wrapper
才正常。std::function
最大的功能是表達匿名函數,特別是[]
裡面捕捉了當前上下文變數的匿名函數,結合著std::shared_ptr
一起用,會有一種動態語言的錯覺。