函数模板全特化是完全替代主模板的定制版本,必须依附于已有模板;重载则是独立普通函数,优先级最高且无模板依赖限制。
特化(template)不是“加一个新函数”,而是为某组具体类型**完全替代**原本的模板生成逻辑;重载则是普通函数重载机制,和模板无关。编译器在函数匹配时,先找非模板函数(含特化),再考虑模板实例化——但特化本身仍属于模板家族,而重载函数是完全独立的实体。
template void f(T); 是主模板
template void f(int); 是全特化:它不参与模板推导,只对 int 生效,且优先级高于主模板void f(int); 是重载:它根本不是模板,是普通函数,优先级最高(比任何模板都高)你不能凭空写一个 template void g,除非之前已声明过 template。而重载没有这个限制:void g(double); 可以单独存在,哪怕从未定义过模板。
T 换成 int* 或 std::vector
C++ 禁止对函数模板做偏特化,比如 template 这种写法是**非法的**。你可能会以为这是“针对 std::vector 的偏特化”,其实它是另一个独立的函数模板(主模板),和 template 构成重载关系。
templatevoid h(T); // 主模板 template void h(std::vector ); // ❌ 不是偏特化,是重载模板(新模板) void h(const char*); // ✅ 普通重载
template struct X<:vector>> { ... }; ,这是合法偏特化std::enable_if)或概念(C++20 requires)来模拟error: function template partial specialization is not allowed
当多个候选同时存在(普通函数、特化、重载模板、主模板),编译器按严格顺序筛选:普通函数 ≈ 特化 > 重载模板 > 主模板。但 cv 限定(const、&)会让实际行为变得微妙——比如 void f(const int&) 和 template 对 int x; 调用 f(x),前者胜出;但若只有 template void f 特化,则它不匹配 const int&,反而可能退回到主模板。
f(int) ≠ f(const int&)
std::is_same_v 或概念约束,而非特化,避免优先级陷阱