目录
模板特化(Specialization)与偏特化(Partial Specialization)
模板模板参数(Template Template Parameters)
摘要
C++中的模板(template)是一种非常强大的工具,用于泛型编程和代码复用。模板可以分为函数模板、类模板、别名模板、变量模板、模板模板参数等。这些模板能够处理任意类型的数据,提高代码的灵活性和可重用性。
函数模板
函数模板允许函数对任意类型的数据进行操作。
基本用法
#include <iostream>
template <typename T>
T max(T a, T b) {
return (a > b) ? a : b;
}
int main() {
std::cout << "Max of 3 and 7: " << max(3, 7) << std::endl;
std::cout << "Max of 3.5 and 2.1: " << max(3.5, 2.1) << std::endl;
std::cout << "Max of 'a' and 'z': " << max('a', 'z') << std::endl;
return 0;
}
重载与特化
函数模板可以与普通函数重载,并且可以进行模板特化。
#include <iostream>
template <typename T>
T max(T a, T b) {
return (a > b) ? a : b;
}
// 特化版本
template <>
const char* max(const char* a, const char* b) {
return (std::strcmp(a, b) > 0) ? a : b;
}
// 重载版本
int max(int a, int b) {
return (a > b) ? a : b;
}
int main() {
std::cout << "Max of 3 and 7: " << max(3, 7) << std::endl;
std::cout << "Max of \"hello\" and \"world\": " << max("hello", "world") << std::endl;
return 0;
}
// Out
Max of 3 and 7: 7
Max of "hello" and "world": world
类模板
类模板允许类处理任意类型的数据。
基本用法
#include <iostream>
template <typename T>
class Pair {
private:
T first, second;
public:
Pair(T a, T b) : first(a), second(b) {}
T getFirst() const { return first; }
T getSecond() const { return second; }
};
int main() {
Pair<int> intPair(1, 2);
Pair<double> doublePair(3.4, 7.8);
std::cout << "First of intPair: " << intPair.getFirst() << std::endl;
std::cout << "Second of doublePair: " << doublePair.getSecond() << std::endl;
return 0;
}
模板特化(Specialization)与偏特化(Partial Specialization)
类模板可以进行全特化(Full Specialization)和偏特化(Partial Specialization -- 偏特化仅在类模板中支持)。
#include <iostream>
// 主模板
template <typename T>
class Storage {
private:
T value;
public:
Storage(T val) : value(val) {}
void print() const { std::cout << "Value: " << value << std::endl; }
};
// 全特化版本
// 全特化是为模板的某个具体类型提供完全不同的实现。
template <>
class Storage<const char*> {
private:
const char* value;
public:
Storage(const char* val) : value(val) {}
void print() const { std::cout << "Value: " << value << std::endl; }
};
// 偏特化版本
// 偏特化是指为模板的一部分参数提供特定的实现,而不是所有参数。偏特化仅适用于类模板,而不适用于函数模板。
template <typename T>
class Storage<T*> {
private:
T* value;
public:
Storage(T* val) : value(val) {}
void print() const { std::cout << "Pointer Value: " << *value << std::endl; }
};
int main() {
Storage<int> intStorage(5);
Storage<const char*> stringStorage("Hello");
int value = 10;
Storage<int*> intPointerStorage(&value);
intStorage.print();
stringStorage.print();
intPointerStorage.print();
return 0;
}
// Out
Value: 5
Value: Hello
Pointer Value: 10
别名模板(Alias Templates)
别名模板用于创建复杂类型的简化别名。
基本用法
#include <iostream>
#include <vector>
// 定义一个别名模板
template <typename T>
using Vec = std::vector<T>;
int main() {
Vec<int> intVec = {1, 2, 3, 4, 5};
for (int i : intVec) {
std::cout << i << " ";
}
std::cout << std::endl;
return 0;
}
变量模板(Variable Templates)
变量模板允许定义模板变量。
基本用法
#include <iostream>
// 定义一个变量模板
template <typename T>
constexpr T pi = T(3.1415926535897932385);
int main() {
std::cout << "pi<int>: " << pi<int> << std::endl;
std::cout << "pi<float>: " << pi<float> << std::endl;
std::cout << "pi<double>: " << pi<double> << std::endl;
return 0;
}
// Out
pi<int>: 3
pi<float>: 3.14159
pi<double>: 3.14159
模板模板参数(Template Template Parameters)
模板模板参数允许模板参数本身是模板。
基本用法
#include <iostream>
#include <vector>
// 定义一个模板类
template <typename T, template <typename, typename> class Container>
class Stack {
private:
Container<T, std::allocator<T>> elements;
public:
void push(const T& elem) {
elements.push_back(elem);
}
void pop() {
if (elements.empty()) {
throw std::out_of_range("Stack<>::pop(): empty stack");
}
elements.pop_back();
}
T top() const {
if (elements.empty()) {
throw std::out_of_range("Stack<>::top(): empty stack");
}
return elements.back();
}
};
int main() {
Stack<int, std::vector> intStack;
intStack.push(7);
intStack.push(42);
std::cout << "Top of intStack: " << intStack.top() << std::endl;
intStack.pop();
std::cout << "Top of intStack after pop: " << intStack.top() << std::endl;
return 0;
}
// Out
Top of intStack: 42
Top of intStack after pop: 7
总结
1. 模板实例化
模板实例化发生在使用模板时,编译器会根据模板参数生成具体类型的代码。模板实例化可能会导致代码膨胀(code bloat),因为每个不同的模板参数都会生成一份独立的代码。
2. 模板元编程
模板元编程是利用模板在编译时进行计算和逻辑判断的技术,常用于实现复杂的数据结构和算法。
3. 编译错误
模板代码的编译错误信息通常较难理解,因为错误通常发生在模板实例化的过程中。
4. 类型推导
模板参数的类型推导有时会遇到困难,特别是在模板函数中。
5. 递归模板实例化
在某些高级用法中,模板可能会涉及递归实例化(如元编程)。这种情况下,需要确保递归在某个点终止,以避免编译时无限递归。
我们可以通过合理使用函数模板、类模板、别名模板、变量模板和模板模板参数,能在一定程度上提高代码的灵活性和可维护性。然而,模板的使用也带来了一些复杂情况和注意事项,如编译时间、代码膨胀、调试困难等。所以在实际项目中,根据项目需求,我们需要合理设计和选择模板,这样能够有效地避免问题的发生。