0

0

C++函数模板与类模板结合使用实例

P粉602998670

P粉602998670

发布时间:2025-09-02 09:33:01

|

1023人浏览过

|

来源于php中文网

原创

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

c++函数模板与类模板结合使用实例

C++函数模板和类模板结合使用,能极大提升代码的灵活性和复用性。简单来说,就是用模板类来存储数据,然后用模板函数来操作这些数据,类型可以根据需要自动推导。

解决方案

下面通过一个例子来说明如何结合使用 C++ 函数模板和类模板。假设我们需要一个通用的数组类,可以存储各种类型的数据,并且需要一个函数来查找数组中的最大值。

#include 
#include 
#include 

// 类模板:通用数组类
template 
class GenericArray {
private:
    std::vector data;
public:
    GenericArray(int size) : data(size) {}

    T& operator[](int index) {
        return data[index];
    }

    int getSize() const {
        return data.size();
    }
};

// 函数模板:查找数组最大值
template 
T findMax(const GenericArray& 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 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 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
    类型的数组作为参数。
  • 遍历数组,找到最大值。
  • 返回最大值。

main
函数:

  • 演示了如何创建
    int
    double
    类型的
    GenericArray
    实例。
  • 演示了如何使用
    findMax
    函数模板查找不同类型数组的最大值。

如何在类模板的成员函数中使用函数模板?

在类模板的成员函数中使用函数模板,其实就是把函数模板的定义放在类模板的内部。这样做可以进一步提升代码的模块化程度,让类模板的功能更加强大。

#include 
#include 
#include 

template 
class GenericArray {
private:
    std::vector data;
public:
    GenericArray(int size) : data(size) {}

    T& operator[](int index) {
        return data[index];
    }

    int getSize() const {
        return data.size();
    }

    // 在类模板内部定义函数模板
    template 
    U findMax() const {
        if (data.empty()) {
            throw std::runtime_error("Array is empty");
        }

        U maxVal = static_cast(data[0]); // 显式转换,确保类型匹配
        for (size_t i = 1; i < data.size(); ++i) {
            if (static_cast(data[i]) > maxVal) { // 显式转换
                maxVal = static_cast(data[i]);
            }
        }
        return maxVal;
    }
};

int main() {
    GenericArray intArray(5);
    intArray[0] = 10;
    intArray[1] = 5;
    intArray[2] = 20;
    intArray[3] = 15;
    intArray[4] = 8;

    // 显式指定模板参数
    double maxInt = intArray.findMax();
    std::cout << "Max int value: " << maxInt << std::endl; // 输出 double 类型

    GenericArray doubleArray(3);
    doubleArray[0] = 3.14;
    doubleArray[1] = 2.71;
    doubleArray[2] = 1.618;

    // 显式指定模板参数
    int maxDouble = doubleArray.findMax();
    std::cout << "Max double value: " << maxDouble << std::endl; // 输出 int 类型

    return 0;
}

在这个例子中,

findMax
函数模板被定义为
GenericArray
类模板的成员函数。 关键在于,你可以在调用
findMax
的时候,显式指定返回值的类型。

模板函数与模板类结合使用时,如何处理类型转换问题?

类型转换是使用模板时经常遇到的问题。如果模板函数和模板类处理的数据类型不一致,就需要进行类型转换。

成新网络商城购物系统
成新网络商城购物系统

使用模板与程序分离的方式构建,依靠专门设计的数据库操作类实现数据库存取,具有专有错误处理模块,通过 Email 实时报告数据库错误,除具有满足购物需要的全部功能外,成新商城购物系统还对购物系统体系做了丰富的扩展,全新设计的搜索功能,自定义成新商城购物系统代码功能代码已经全面优化,杜绝SQL注入漏洞前台测试用户名:admin密码:admin888后台管理员名:admin密码:admin888

下载

例如,在上面的例子中,

findMax
函数模板返回值的类型
U
可以和数组元素的类型
T
不同。 为了保证比较的正确性,需要使用
static_cast
将数组元素转换为
U
类型。

template 
class GenericArray {
    // ...
    template 
    U findMax() const {
        // ...
        U maxVal = static_cast(data[0]);
        // ...
        if (static_cast(data[i]) > maxVal) {
            // ...
        }
        // ...
    }
};

static_cast
是一种编译时类型转换,它比 C 风格的类型转换更加安全。 但是,使用
static_cast
也需要小心,因为它可能会导致数据丢失或精度损失。 例如,将
double
类型转换为
int
类型会丢失小数部分。

如何在模板函数中访问模板类的私有成员?

在模板函数中直接访问模板类的私有成员是不允许的,因为私有成员只能在类的内部访问。 但是,可以通过以下几种方式来间接访问私有成员:

  1. 友元函数: 可以将模板函数声明为模板类的友元函数。这样,模板函数就可以访问模板类的所有成员,包括私有成员。

    template 
    class GenericArray {
    private:
        std::vector data;
    
        // 声明模板函数为友元函数
        template 
        friend U findMax(const GenericArray& arr);
    
    public:
        GenericArray(int size) : data(size) {}
    
        T& operator[](int index) {
            return data[index];
        }
    
        int getSize() const {
            return data.size();
        }
    };
    
    template 
    T findMax(const GenericArray& 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;
    }
  2. 提供公共接口: 可以在模板类中提供公共的成员函数,用于访问或修改私有成员。 这样,模板函数就可以通过调用这些公共接口来间接访问私有成员。 这是更推荐的做法,因为它更符合面向对象的设计原则。

    template 
    class GenericArray {
    private:
        std::vector data;
    
    public:
        GenericArray(int size) : data(size) {}
    
        T& operator[](int index) {
            return data[index];
        }
    
        int getSize() const {
            return data.size();
        }
    
        // 提供公共接口访问私有成员
        const std::vector& getData() const {
            return data;
        }
    };
    
    template 
    T findMax(const GenericArray& arr) {
        // 通过公共接口访问 data
        const std::vector& 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)进行自定义操作?

函数对象(Functor)是一个行为类似函数的对象。 它可以是一个类的实例,该类重载了

operator()
运算符。 使用函数对象可以实现更加灵活和可定制的操作。

#include 
#include 
#include 

// 函数对象:自定义比较函数
template 
class GreaterThan {
private:
    T threshold;
public:
    GreaterThan(T threshold) : threshold(threshold) {}

    bool operator()(const T& value) const {
        return value > threshold;
    }
};

template 
class GenericArray {
private:
    std::vector data;
public:
    GenericArray(int size) : data(size) {}

    T& operator[](int index) {
        return data[index];
    }

    int getSize() const {
        return data.size();
    }

    // 使用函数对象进行过滤
    std::vector filter(const GreaterThan& predicate) const {
        std::vector result;
        for (const T& value : data) {
            if (predicate(value)) {
                result.push_back(value);
            }
        }
        return result;
    }
};

int main() {
    GenericArray intArray(5);
    intArray[0] = 10;
    intArray[1] = 5;
    intArray[2] = 20;
    intArray[3] = 15;
    intArray[4] = 8;

    // 创建一个函数对象
    GreaterThan gt10(10);

    // 使用函数对象过滤数组
    std::vector 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++函数模板与类模板结合使用,可以编写出高度灵活和可复用的代码。理解类型转换、友元函数、函数对象等概念,可以更好地利用模板的优势。

相关专题

更多
数据类型有哪几种
数据类型有哪几种

数据类型有整型、浮点型、字符型、字符串型、布尔型、数组、结构体和枚举等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

303

2023.10.31

php数据类型
php数据类型

本专题整合了php数据类型相关内容,阅读专题下面的文章了解更多详细内容。

222

2025.10.31

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1465

2023.10.24

Go语言中的运算符有哪些
Go语言中的运算符有哪些

Go语言中的运算符有:1、加法运算符;2、减法运算符;3、乘法运算符;4、除法运算符;5、取余运算符;6、比较运算符;7、位运算符;8、按位与运算符;9、按位或运算符;10、按位异或运算符等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

228

2024.02.23

php三元运算符用法
php三元运算符用法

本专题整合了php三元运算符相关教程,阅读专题下面的文章了解更多详细内容。

85

2025.10.17

go语言 面向对象
go语言 面向对象

本专题整合了go语言面向对象相关内容,阅读专题下面的文章了解更多详细内容。

56

2025.09.05

java面向对象
java面向对象

本专题整合了java面向对象相关内容,阅读专题下面的文章了解更多详细内容。

49

2025.11.27

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

318

2023.08.02

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

43

2026.01.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Bootstrap 5教程
Bootstrap 5教程

共46课时 | 2.9万人学习

ECMAScript6 / ES6---十天技能课堂
ECMAScript6 / ES6---十天技能课堂

共25课时 | 1.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号