std::is_invocable用于编译期判断可调用对象能否以给定参数类型调用,支持泛型编程中的条件分支;例如结合if constexpr可选择性执行函数调用,避免运行时错误。相关类型包括std::is_invocable_r_v(检查返回值类型)和std::is_nothrow_invocable_v(检查是否noexcept),广泛应用于模板元编程、通用回调和DSL设计中。
std::is_invocable 是 C++17 引入的一个类型特性(type trait),用于在编译期判断某个可调用对象(如函数、函数指针、lambda、函数对象等)是否可以用给定的参数类型进行调用。它属于头文件 中的一部分,是实现 SFINAE(替换失败不是错误)和模板元编程的重要工具。
在泛型编程中,我们经常需要根据传入的对象是否能以特定方式调用,来选择不同的实现路径。std::is_invocable 可以帮助我们在编译时完成这种判断。
例如,你想写一个通用包装器,只对能接受 int 参数的可调用对象执行操作:
#include#include template void call_if_callable_with_int(F f) { if constexpr (std::is_invocable_v ) { f(42); } else { std::cout << "Cannot call with int\n"; } } void func(int x) { std::cout << "func called: " << x << '\n'; } void func_str(const std::string&) { } int main() { call_if_callable_with_int(func); // OK call_if_callable_with_int(func_str); // 不会调用,输出提示 }
除了 std::is_invocable,C++ 还提供了几个相关的类型特性,用途略有不同:
比如你希望一个函数能被 double 调用,并且返回 int:
std::is_invocable_r_v
这类类型特性常用于高级模板设计中,例如:
if constexpr 实现编译期分支,避免运行时开销。基本上就这些。std::is_invocable 让你在模板里“试探”一个调用是否成立,而不必真的去调用它,既安全又高效。