通过类模板定义通用数据结构,结合函数模板实现灵活操作,支持类型自动推导与转换,利用友元函数或公共接口访问私有成员,并可通过函数对象实现自定义逻辑,提升代码复用性与扩展性。

C++函数模板和类模板结合使用,能极大提升代码的灵活性和复用性。简单来说,就是用模板类来存储数据,然后用模板函数来操作这些数据,类型可以根据需要自动推导。
解决方案
下面通过一个例子来说明如何结合使用 C++ 函数模板和类模板。假设我们需要一个通用的数组类,可以存储各种类型的数据,并且需要一个函数来查找数组中的最大值。
#include <iostream>
#include <vector>
#include <algorithm>
// 类模板:通用数组类
template <typename T>
class GenericArray {
private:
    std::vector<T> data;
public:
    GenericArray(int size) : data(size) {}
    T& operator[](int index) {
        return data[index];
    }
    int getSize() const {
        return data.size();
    }
};
// 函数模板:查找数组最大值
template <typename T>
T findMax(const GenericArray<T>& arr) {
    if (arr.getSize() == 0) {
        throw std::runtime_error("Array is empty");
    }
    T maxVal = arr[0];
    for (int i = 1; i < arr.getSize(); ++i) {
        if (arr[i] > maxVal) {
            maxVal = arr[i];
        }
    }
    return maxVal;
}
int main() {
    // 创建一个存储 int 类型的数组
    GenericArray<int> intArray(5);
    intArray[0] = 10;
    intArray[1] = 5;
    intArray[2] = 20;
    intArray[3] = 15;
    intArray[4] = 8;
    // 使用函数模板查找 int 数组的最大值
    int maxInt = findMax(intArray);
    std::cout << "Max int value: " << maxInt << std::endl;
    // 创建一个存储 double 类型的数组
    GenericArray<double> doubleArray(3);
    doubleArray[0] = 3.14;
    doubleArray[1] = 2.71;
    doubleArray[2] = 1.618;
    // 使用函数模板查找 double 数组的最大值
    double maxDouble = findMax(doubleArray);
    std::cout << "Max double value: " << maxDouble << std::endl;
    return 0;
}类模板 GenericArray
立即学习“C++免费学习笔记(深入)”;
T
std::vector
operator[]
getSize()
函数模板 findMax
GenericArray<T>
main
int
double
GenericArray
findMax
在类模板的成员函数中使用函数模板,其实就是把函数模板的定义放在类模板的内部。这样做可以进一步提升代码的模块化程度,让类模板的功能更加强大。
#include <iostream>
#include <vector>
#include <algorithm>
template <typename T>
class GenericArray {
private:
    std::vector<T> data;
public:
    GenericArray(int size) : data(size) {}
    T& operator[](int index) {
        return data[index];
    }
    int getSize() const {
        return data.size();
    }
    // 在类模板内部定义函数模板
    template <typename U>
    U findMax() const {
        if (data.empty()) {
            throw std::runtime_error("Array is empty");
        }
        U maxVal = static_cast<U>(data[0]); // 显式转换,确保类型匹配
        for (size_t i = 1; i < data.size(); ++i) {
            if (static_cast<U>(data[i]) > maxVal) { // 显式转换
                maxVal = static_cast<U>(data[i]);
            }
        }
        return maxVal;
    }
};
int main() {
    GenericArray<int> intArray(5);
    intArray[0] = 10;
    intArray[1] = 5;
    intArray[2] = 20;
    intArray[3] = 15;
    intArray[4] = 8;
    // 显式指定模板参数
    double maxInt = intArray.findMax<double>();
    std::cout << "Max int value: " << maxInt << std::endl; // 输出 double 类型
    GenericArray<double> doubleArray(3);
    doubleArray[0] = 3.14;
    doubleArray[1] = 2.71;
    doubleArray[2] = 1.618;
    // 显式指定模板参数
    int maxDouble = doubleArray.findMax<int>();
    std::cout << "Max double value: " << maxDouble << std::endl; // 输出 int 类型
    return 0;
}在这个例子中,
findMax
GenericArray
findMax
类型转换是使用模板时经常遇到的问题。如果模板函数和模板类处理的数据类型不一致,就需要进行类型转换。
例如,在上面的例子中,
findMax
U
T
static_cast
U
template <typename T>
class GenericArray {
    // ...
    template <typename U>
    U findMax() const {
        // ...
        U maxVal = static_cast<U>(data[0]);
        // ...
        if (static_cast<U>(data[i]) > maxVal) {
            // ...
        }
        // ...
    }
};static_cast
static_cast
double
int
在模板函数中直接访问模板类的私有成员是不允许的,因为私有成员只能在类的内部访问。 但是,可以通过以下几种方式来间接访问私有成员:
友元函数: 可以将模板函数声明为模板类的友元函数。这样,模板函数就可以访问模板类的所有成员,包括私有成员。
template <typename T>
class GenericArray {
private:
    std::vector<T> data;
    // 声明模板函数为友元函数
    template <typename U>
    friend U findMax(const GenericArray<T>& arr);
public:
    GenericArray(int size) : data(size) {}
    T& operator[](int index) {
        return data[index];
    }
    int getSize() const {
        return data.size();
    }
};
template <typename T>
T findMax(const GenericArray<T>& arr) {
    // 现在可以访问 arr.data
    if (arr.data.empty()) {
        throw std::runtime_error("Array is empty");
    }
    T maxVal = arr.data[0];
    for (size_t i = 1; i < arr.data.size(); ++i) {
        if (arr.data[i] > maxVal) {
            maxVal = arr.data[i];
        }
    }
    return maxVal;
}提供公共接口: 可以在模板类中提供公共的成员函数,用于访问或修改私有成员。 这样,模板函数就可以通过调用这些公共接口来间接访问私有成员。 这是更推荐的做法,因为它更符合面向对象的设计原则。
template <typename T>
class GenericArray {
private:
    std::vector<T> data;
public:
    GenericArray(int size) : data(size) {}
    T& operator[](int index) {
        return data[index];
    }
    int getSize() const {
        return data.size();
    }
    // 提供公共接口访问私有成员
    const std::vector<T>& getData() const {
        return data;
    }
};
template <typename T>
T findMax(const GenericArray<T>& arr) {
    // 通过公共接口访问 data
    const std::vector<T>& data = arr.getData();
    if (data.empty()) {
        throw std::runtime_error("Array is empty");
    }
    T maxVal = data[0];
    for (size_t i = 1; i < data.size(); ++i) {
        if (data[i] > maxVal) {
            maxVal = data[i];
        }
    }
    return maxVal;
}函数对象(Functor)是一个行为类似函数的对象。 它可以是一个类的实例,该类重载了
operator()
#include <iostream>
#include <vector>
#include <algorithm>
// 函数对象:自定义比较函数
template <typename T>
class GreaterThan {
private:
    T threshold;
public:
    GreaterThan(T threshold) : threshold(threshold) {}
    bool operator()(const T& value) const {
        return value > threshold;
    }
};
template <typename T>
class GenericArray {
private:
    std::vector<T> data;
public:
    GenericArray(int size) : data(size) {}
    T& operator[](int index) {
        return data[index];
    }
    int getSize() const {
        return data.size();
    }
    // 使用函数对象进行过滤
    std::vector<T> filter(const GreaterThan<T>& predicate) const {
        std::vector<T> result;
        for (const T& value : data) {
            if (predicate(value)) {
                result.push_back(value);
            }
        }
        return result;
    }
};
int main() {
    GenericArray<int> intArray(5);
    intArray[0] = 10;
    intArray[1] = 5;
    intArray[2] = 20;
    intArray[3] = 15;
    intArray[4] = 8;
    // 创建一个函数对象
    GreaterThan<int> gt10(10);
    // 使用函数对象过滤数组
    std::vector<int> filteredArray = intArray.filter(gt10);
    std::cout << "Filtered array: ";
    for (int value : filteredArray) {
        std::cout << value << " ";
    }
    std::cout << std::endl;
    return 0;
}在这个例子中,
GreaterThan
GenericArray
filter
GreaterThan
总的来说,C++函数模板与类模板结合使用,可以编写出高度灵活和可复用的代码。理解类型转换、友元函数、函数对象等概念,可以更好地利用模板的优势。
以上就是C++函数模板与类模板结合使用实例的详细内容,更多请关注php中文网其它相关文章!
 
                        
                        每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
 
                 
                                
                                 收藏
收藏
                                                                            ![smarty模板引擎[函数篇]](https://img.php.cn/upload/course/000/000/068/62554919bac5d558.png) 
                                
                                 收藏
收藏
                                                                            ![Smarty模板引擎[基础篇]](https://img.php.cn/upload/course/000/000/068/62567ac03df54222.png) 
                                
                                 收藏
收藏
                                                                            Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号