C/C++ 函数重载

        函数多态是C++在C语言的基础新增的功能。默认参数能够使用不同数目的参数调用同一个函数,而函数多态(函数重载)让您能够使用多个同名的函数。术语“多态”指的是有多种形式,因此函数多态允许函数可以有多种形式。类似地,术语“函数重载”指的是可以有多个同名的函数,因此对名称进行了重载。这两个术语指的是同一回事,但我们通常使用函数重载。可以通过函数重载来设计一系列函数一它们完成相同的工作,但使用不同的参数列表。

        重载函数就像是有多种含义的动词。例如,Piggy 小姐可以在棒球场为家乡球队助威(root),也可以在地里种植(root)菌类作物。根据上下文可以知道在每一种情况下root的含义是什么。同样,C++使用上下文来确定要使用的重载函数版本。

        函数重载的关键是函数的参数列表一也称为函数特征标(function signature)。如果两个函数的参数数目和类型相同,同时参数的排列顺序也相同,则它们的特征标相同,而变量名是无关紧要的。C++允许定义名称相同的函数,条件是它们的特征标不同。如果参数数目和/或参数类型不同,则特征标也不同。例如,可以定文一组原型如下的print()函数

        void print(const char *str, int width);    // #1
        void print(double d, int width);        // #2
        void print(long l, int width);            // #3
        void print(int i; int width);            // #4
        void print(const char * str);            // #5

        使用 print()函数时,编译器将根据所采取的用法使用有相应特征标的原型:
        print("Pancakes", 15);        // 使用 #1
        print("Syrup");                // 使用 #5
        print(1999, 10);            // 使用 #2
        print(1999L,12)            // 使用 #4
        print(1999l,15);            // 使用 #3

        例如,print(“Pancakes”, 15)使用一个字符串和一个整数作为参数,这与原型匹配使用被重载的函数时,需要在函数调用中使用正确的参数类型。例如,对于下面的语句:

        unsigned int year = 3210;
        print(year,6);            // 调用时存在二义性

        print()调用与那个原型匹配呢?它不与任何原型匹配!没有匹配的原型并不会自动停止使用其中的某个函数,因为C++将尝试使用标准类型转换强制进行匹配。如果#2原型是print()唯一的原型,则函数调用print(year, 6)将把year转换为double类型。但在上面的代码中,有3个将数字作为第一个参数的原型,因此有3种转换year的方式。在这种情况下,C++将拒绝这种函数调用,并将其视为错误。


        一些看起来彼此不同的特征标是不能共存的。例如,请看下面的两个原型:

        double cube(double x);
        double cube(double& x);

        您可能认为可以在此处使用函数重载,因为它们的特征标看起来不同。然而,请从编译器的角度来考虑这个问题。假设有下面这样的代码:

        cout << cube(x);
        
        参数X与double x原型和double &x原型都匹配, 因此编译器无法确定究竞应使用哪个原型。为避免这种混乱,编译器在检查函数特征标时,将把类型引用和类型本身视为同一个特征标。

        匹配函数时,并不区分const和非cost变量。请看下面的原型:

        void dribble(char * bits);            // 重载函数
        void dribble(const char *cbits);    // 重载函数
        void dabble(char *bits):            // 非重载函数
        void drivel(const char *bitsl);        // 非重载函数


        下面列出了各种函数调用对应的原型:

        const char p1[20) = "How's the weather?";
        char p2[20] = "How'sbusiness?“;
        dribbre(pl);        //调用dribble(const char *);
        dribbIe(p2);        //调用dribble(Char *)

        dabble(p1);        //no match
        dabble(p2);        // dabble(char *)

        drive(p1);        // drivel(const char *);
        drivel(p2);        // drivel(const char *);
        dribble()函数有两个原型,一个用于const指针, 另一个用于常规指针,编译器将根据实参是否为const来决定使用哪个原型。dribble()函数只与带非const参数的调用匹配,而drivel()函数可以与带const或非const参数的调用匹配。drivel()和 dabble()之所以在行为上有这种差别, 主要是由于将非const 值赋给cost变量是合法的,但反之则是非法的。


        请记住,是特征标,而不是函数类型使得可以对函数进行重载。例如,下面的两个声明是互斥的:
        
        long gronk(int n, float m); 
        double gronk(int n, froat m);
        
        因此C++不允许以这种方式重载gronk()。返回类型可以不同,但特征标也必须不同:

        long gronk(int n,float m);
        double gronk(float n,float m);


重载引用参数
        类设计和STL经常使用引用参数,因此知道不同引用类型的重载很有用。请春下面三个原型:

        void sink(double & rl);
        void sank(const double & r2);
        void sunk(double && r3);

        左值引用参数rl与可修改的左值参数(如 double变量)匹配;const 左值引用参数r2与可修改的左值参数、const左值参数和右值参数(如两个double值的和)匹配;
最后,左值引用参数r3与左值匹配。注意到与r1或r3匹配的参数都与r2 匹配。这就带来了一个问题,如果重载使用这三种参数的函数,结果将如何?答案是将调用最匹配的版本:

        void staff(double & rs) ;
        voit staff(const double & rcs);
        void stove(double & r1);
        void stove(const double & r2);
        void stove(double && r3];

        这让您能够根据参数是左值、const 还是右值来定制函数的行为:

        double x= 55.5;
        const double y = 32.0;
        stove(x);     //调用 stove (double &)
        stove(y);    //调用 stove (const double &)
        stove(x+y); //调用 stove (double &)

        如果没有定义函数 stove double &&), stove(x+y)将调用函数stove(const double &)。

示例源码:

// Len2024_0101.cpp : 此文件包含 "main" 函数。程序执行将在此处开始并结束。
//#include <iostream>
using namespace std;unsigned long left(unsigned long num, unsigned ct);
char* left(const char * str, int n = 1);int main()
{char trip[] = "Hawaii!!";unsigned long n = 12345678;int i;char * temp;for (i = 1; i < 10; i++){cout << left(n, i) << endl;temp = left(trip, i);cout << temp << endl;delete[] temp;}return 0;
}
unsigned long left(unsigned long num, unsigned ct)
{unsigned digits = 1;unsigned long n = num;if (ct == 0 || num == 0){return 0;}while (n /= 10){digits++;}if (digits > ct){ct = digits - ct;while (ct--){num /= 10;}return num;}elsereturn num;}
char* left(const char * str, int n)
{if (n < 0){n = 0;}char* p = new char[n + 1];int i;for (i = 0; i < n && str[i]; i++){p[i] = str[i];}while (i <= n){p[i++] = '\0';}return p;
}

执行结果:

        前面创建了一个left()函数,它返回一个指针,指向字符串的前 n个字符。下面添加另-个left()函数,它返回整数的前n位。例如,可以使用该函数来查看被存储为整数的、美国邮政编码的前3位一如果要根据城区分拣邮件,则这种操作很有用。

        该函数的整数版本编写起来比字符串版本更困难些,因为并不是整数的每一位被存储在相应的数组元素中。一种方法是,先计算数字含多少位。将数字除以10便可以去掉一位因此可以使用除法来计算数位。更准确地说,可以用下面的循环完成这种工作:

        ungigned digits = 1;
        while(n/=10)
            digits++;

        上述循环计算每次删除n中的一位时,需要多少次才能删除所有的位。前面讲过 n/=10是 n=n/10的缩写。例如,如果n为8,则该测试条将8/10的值(0,由于这是整数除法)赋给n。这将结束循环, digits的值仍然为1。但如果n为238,第一轮循环测试将n设置为238/10即23。这个值不为零,因此循环将digits增加到2。下一轮循环将n设置为23/10,即2。这个值还是不为零,因此digits将增加到3下一
        轮循环将n设置为2/10,即0,从而结束循环而,digits被设置为正确的值——3。

        现在假设知道数字共有5位,并要返回前3位,则将这个数除以10 后再除以10,便可以得到所需的值。每除以 10 次就删除数字的最后一位。要知道需要除多少位,只需将总位数减去要获得的位数即可。例如,要获得9位数的前4位,需要删除后面的5位。可以这样编写代码:

        ct =digits - ct;
        while(t--)
            num/=10;
        return num;
        
        上面程序将代码放到了一个新的left()函数中。该函数还包含一些用于处理特殊情况的代码,如用户要求显示0位或要求显示的位数多于总位数。由于新left()的特征标不同于旧的 left(),因此可以在个程序中使用这两个函数。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/589221.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

华为云创新中心,引领浙南的数字化腾飞

编辑&#xff1a;阿冒 设计&#xff1a;沐由 县域经济是我国国民经济的重要组成部分&#xff0c;是推动经济社会全面发展的核心力量之一。在推进中国式现代化的征程中&#xff0c;县域经济扮演的角色也越来越重要。 毫无疑问&#xff0c;县域经济的良性发展&#xff0c;需要多方…

数据的确权、流通、入表与监管研究(一):数据与确权(下)

关注WX公众号&#xff1a; commindtech77&#xff0c; 获得数据资产相关白皮书下载地址 1. 回复关键字&#xff1a;数据资源入表白皮书 下载 《2023数据资源入表白皮书》 2. 回复关键字&#xff1a;光大银行 下载 光大银行-《商业银行数据资产会计核算研究报告》 3. 回复关键字…

【NLP论文】02 TF-IDF 关键词权值计算

之前写了一篇关于关键词词库构建的文章&#xff0c;没想到反响还不错&#xff0c;最近有空把接下来的两篇补完&#xff0c;也继续使用物流关键词词库举例&#xff0c;本篇文章承接关键词词库构建并以其为基础&#xff0c;将计算各关键词的 TF-IDF 权值&#xff0c;TF-IDF 权值主…

软件工程PPT 笔记摘录(2)

分析软件需求 UML 提供了用例图来分析和描述用例视角的软件需求模型 UML 提供了交互图和状态图来描述行为视角的软件需求模型 UML 提供了类图来描述和分析业务领域的概念模型 顺序图&#xff1a;强调消息传递的时间序 通信图&#xff1a;突出对象间的合作 类图&#xff0…

掌握C++11标准库(STL):理解STL的核心概念

深入探索C11标准库STL&#xff1a;新特性和优化技巧 一、前言二、容器简介三、迭代器简介四、map与unordered_map&#xff08;红黑树VS哈希表&#xff09;4.1、map和unordered_map的差别4.2、优缺点以及适用处4.3、小结 五、总结 一、前言 STL定义了强大的、基于模板的、可复用…

python设计模式:模板方法模式

更多Python学习内容&#xff1a;ipengtao.com 软件设计和编程中&#xff0c;设计模式是一种有助于解决常见问题的强大工具。其中之一是"模板方法模式"&#xff0c;它是一种行为型设计模式&#xff0c;允许你定义一个算法的骨架&#xff0c;但将一些步骤的具体实现延迟…

win11 电脑睡眠功能失效了如何修复 win11 禁止鼠标唤醒

1、win11睡眠不管用怎么办&#xff0c;win11电脑睡眠功能失效了如何修复 在win11系统中拥有许多令人激动的新功能和改进&#xff0c;有些用户在使用win11电脑时可能会遇到一个问题&#xff1a;睡眠模式不起作用。当他们尝试将计算机置于睡眠状态时&#xff0c;却发现系统无法进…

HarmonyOS应用程序包快速修复

快速修复概述 快速修复是HarmonyOS系统提供给开发者的一种技术手段&#xff0c;支持开发者以远快于应用升级的方式对应用程序包进行缺陷修复。和全量应用升级软件版本相比&#xff0c;快速修复的主要优势在小、快和用户体验好。在较短的时间内不中断正在运行的应用的情况下&am…

SpringBoot + Vue 抖音全平台项目

简介 本项目是一个短视频平台&#xff0c;拥有热度排行榜&#xff0c;热门视频&#xff0c;兴趣推送&#xff0c;关注推送&#xff0c;内容审核等功能。 源码下载 网盘 (访问密码: 8418) 登录/注册 首页 创作中心 架构设计 上传视频业务流程 视频推送流程 1.用户订阅分类后…

聚类:聚类的介绍及k-means算法

聚类&#xff1a;聚类的介绍及k-means算法 什么是聚类 聚类就是在输入为多个数据时&#xff0c;将“相似”的数据分为一组的操作。1 个组就叫作 1 个 “簇”。下面的示例中每个点都代表1 个数据&#xff0c;在平面上位置较为相近、被圈起来的点就代表一 类相似的数据。也就是…

如何为开源项目和社区做贡献 -- 你应该知道的十件事(一)

1. 前言 大家好&#xff0c;我叫颜国进&#xff0c;现为英特尔边缘计算创新大使、百度飞桨开发者专家。回溯至2021年12月&#xff0c;那时的我&#xff0c;身为机械专业研一新生&#xff0c;仅在C和Python编程语言上有些许基础&#xff0c;对于深度学习的殿堂还只敢在门口窥探&…

【动态规划】【字符串】C++算法:正则表达式匹配

作者推荐 视频算法专题 涉及知识点 动态规划 字符串 LeetCode10:正则表达式匹配 给你一个字符串 s 和一个字符规律 p&#xff0c;请你来实现一个支持 ‘.’ 和 ‘’ 的正则表达式匹配。 ‘.’ 匹配任意单个字符 ’ 匹配零个或多个前面的那一个元素 所谓匹配&#xff0c;是…

web网站的工作流程和开发模式

web网站的工作流程和开发模式 基于Java Script封装的高级技术&#xff1a;Vue、Element、Nginx(前端程序部署的服务器) 初识Web前端 Web标准

Conv2Former:一种transformer风格的卷积特征提取方式

一、前言 昨天读到了一篇有意思的文章&#xff0c;文章提出通过利用卷积调制操作来简化self-attention。还证明了这种简单的方法可以更好地利用卷积层中嵌套的大核(≥7 7)。我们都知道ViTs推动了设计识别模型的发展&#xff0c;近几年使用的也相当的多&#xff0c;通常就是CN…

《ORANGE’S:一个操作系统的实现》读书笔记(十九)输入输出系统(一)

我们刚刚实现了简单的进程&#xff0c;你现在可能很想把它做得更加完善&#xff0c;比如进一步改进调度算法、增加通信机制等。但是这些工作不但做起来没有尽头&#xff0c;而且有些也是难以实现的&#xff0c;因为进程必须与I/O、内存管理等其它模块一起工作。而且&#xff0c…

Linux系统---进程程序替换

顾得泉&#xff1a;个人主页 个人专栏&#xff1a;《Linux操作系统》 《C/C》 《LeedCode刷题》 键盘敲烂&#xff0c;年薪百万&#xff01; 一、进程程序替换 一、替换原理 用fork 创建子进程后执行的是和父进程相同的程序 ( 但有可能执行不同的代码分支 ), 子进程往往要…

02 HAL库驱动按键响应外部中断

引言&#xff1a;这里我采用的实验平台可能跟大家的不太一样&#xff0c;文章的图像是一块资源拓展板&#xff0c; 主控板式fs_mp1a, 该板子的SOC是stm32mp157a&#xff0c; 有两个内核一个A7&#xff0c; 一个M4.但是实验的流程肯定都是一样的&#xff0c; 因为都是裸机程序嘛…

【PTA-C语言】实验七-函数与指针I

如果代码存在问题&#xff0c;麻烦大家指正 ~ ~有帮助麻烦点个赞 ~ ~ 目录——实验七-函数与指针I 6-1 弹球距离&#xff08;分数 10&#xff09;6-2 使用函数输出一个整数的逆序数&#xff08;分数 10&#xff09;6-3 使用函数求最大公约数&#xff08;分数 10&#xff09;6-4…

Visual Studio 2015 中 SDL2 开发环境的搭建

Visual Studio 2015 中 SDL2 开发环境的搭建 Visual Studio 2015 中 SDL2 开发环境的搭建新建控制台工程拷贝并配置 SDL2 开发文件拷贝 SDL2 开发文件配置 SDL2 开发文件 测试SDL2 开发文件的下载链接 Visual Studio 2015 中 SDL2 开发环境的搭建 新建控制台工程 新建 Win32 …

从2023看2024前端发展趋势

前言 流光溯影&#xff0c;纵观2023全年&#xff0c;整个前端业界呈现出百业凋零之状&#xff0c;更不乏有“前端已死”等论调甚嚣尘上。从全局视角看IT行业&#xff0c;除了AI领域的大语言模型爆发外&#xff0c;整体都鲜有特别亮眼及突出的技术展现。故而&#xff0c;作为IT…