一、函数重载

普通函数重载的关键是参数列表---也称函数特征标。函数参数中有以下情况可以出现重载:

1、  形参个数不同

2、  形参的类型不同

3、  形参的类型和个数都不同

const形参和函数重载

重载无法区分const和非const

首先分清楚什么是顶层const,什么是底层const

前提是有一个指针或是引用,指向一个数据,顶层const表示该指针不能改变(不能重新指向另一个数据),底层const表示该指针指向的数据不能被改变。(顶层const可以理解成不同的类型)

底层const写法:

Int num = 1;

const int * p = #      //等价于int const * p = &num

顶层const写法:

Int num = 1;

int * const p = #

当指针作为函数参数时,顶层const可以区分,而底层const不可区分

下面的写法出现重复定义:

double cube(double x)

double cube(const double x)

下面的底层const可以区分

void dribble(char* bits);

void dribble(const char * bits);

下面的顶层const不可区分:

void dribble(char* bits);

void dribble(char * const bits);

同理,引用的情况是可以区分的

double cube(double& x)

double cube(const double & x)

const修饰函数

如果const是在函数结尾,表示整个函数体内代码不可修改时也可以用来标志一个函数

即一个函数即使参数个数、类型全相等,但一个使用const修饰函数,一个没有,则这两个函数还是可以被定义的

左值形参和右值形参

首先区分传引用和传复制传值,如下这两个定义会出现重复定义

double cube(double& x);

double cube(double x);

左值形参和右值形参情况有所不同

假设有如下定义

double cube(double&& x)

double cube(double x)

使用cube(x)调用的时候调用第二个,但使用cube(x+1)时会出现多个匹配问题(匹配问题说明函数定义是没有歧义的,但是在寻找对应的匹配的函数是有多个符合条件的结果,只是不知道选哪个)

二、函数匹配问题

定义了一组重载函数后,我们需要以合理的实参调用它们。函数匹配是指一个过程,在这个过程中我们把函数调用与一组重载函数中的某一个关联起来,函数匹配也叫做函数确定。编译器首先将调用的实参与重载集合中每一个函数的形参进行比较,然后根据比较的结果决定到底调用哪个函数。

现在我们需要掌握的是,当调用重载函数时有三种可能的结果:

  • 编译器找到一个与实参最佳匹配的函数,并生成调用该函数的代码;
  • 找不到任何一个函数与调用的实参匹配,此时编译器发出无匹配的错误信息;
  • 又多于一个函数可以匹配,但是每一个都不是明显的最佳选择。此时也将发生错误,称为二义性调用。

函数匹配过程分为三步

1、   确定候选函数,候选函数具备两个特征:一是与被调用的函数同名,二是其声明在调用点可见。

2、   确定可行函数,可行函数也具备两个特征:一是形参数量与本次调用的形参数量一致,二是每个实参的类型与对应的形参类型相同,或者能转化成形参的类型

3、   寻找最佳匹配,有多个可行函数的时候,需要选择一个最佳匹配,基本思想是实参类型与形参类型越接近,他们匹配得越好。

含有多个参数的函数匹配,有且只有一个函数满足下列条件,函数匹配成功:

l  该函数每个实参匹配都不劣于其他可行函数需要的匹配

l  至少有一个实参的匹配优于其他可行函数的匹配

比如使用f(2,3.5)匹配下面函数时会出现冲突:

void f(int x,int y)

void f(double x, double y)

实参类型转换

编译器将实参到形参的转换划分为几个等级

1、  精确匹配,包含如下情况

l  实参与形参类型相同

l  实参从数组类型或函数类型转换成对应的指针类型

l  向实参添加顶层const或者从实参中删除顶层const

2、  通过const转换实现的匹配

3、  通过类型提升实现的转换

4、  通过算术类型(int、short等的转换)或指针转换(0或字面值nullptr能转换成任意指针类型,指向任意非常量的指针能转换成void*,指向任意对象的指针能转换成void*)

5、  通过类类型转换实现的匹配

区别类型的提升和转换:

把char、unsigned char、short、unsigned short转换成int类型称为类型提升(promotion)

long double、double、float、unsigned long long、long long、unsigned long、long、unsigned int、int之间的转换称为类型转换

类型等级由高到低依次为:long double、double、float、 unsigned long long、long long、unsigned long、long、unsigned int、int

区分“忽略顶层const”和“const转换”

忽略顶层const的例子,他们单独都可以被double x = 9.0;cube(x);两条语句调用(即使x是非const,但仍旧可以调用double cube(const double x)),所以以下两种是同一定义:

double cube(const double x)

double cube(double x)

看一个指针的例子,下面两种同样是忽略顶层const的例子。他们单独都可以被double x = 9.0;cube(x);两条语句调用:

double cube(double* x)

double cube( double* const x)

如果有double cube(const double x),同样可以使用以下语句调用,因为

double x = 9.0

cube(&x)

以上都是顶层冲突的例子,但是const转换不同,它指非const的对象能被转换成const

double cube(const double * x)

能被以下的语句调用

double x = 9.0

cube(&x)

x不是const的,但是传给了const,出现了const转换

c++选择重载函数的更多相关文章

  1. C++:成员运算符重载函数和友元运算符重载函数的比较

    5.2.4 成员运算符重载函数和友元运算符重载函数的比较 (1)对双目运算符而言,成员运算符重载函数参数表中含有一个参数,而友元运算符重载函数参数表中有两个参数:对于单目运算符而言,成员运算符重载函数 ...

  2. 你好,C++(28)用空间换时间 5.2 内联函数 5.3 重载函数

    5.2  内联函数 通过5.1节的学习我们知道,系统为了实现函数调用会做很多额外的幕后工作:保存现场.对参数进行赋值.恢复现场等等.如果函数在程序内被多次调用,且其本身比较短小,可以很快执行完毕,那么 ...

  3. C++学习笔记(十二):重载函数

    1. 什么是重载函数 假设同一作用域内的几个函数名字同样但形參列表不同.那么这些函数就称之为--重载函数. 比如: void print( const char *cp); void print(co ...

  4. Item 27: 明白什么时候选择重载,什么时候选择universal引用

    本文翻译自<effective modern C++>,由于水平有限,故无法保证翻译完全正确,欢迎指出错误.谢谢! 博客已经迁移到这里啦 Item 26已经解释了,不管是对全局函数还是成员 ...

  5. C++ 重载运算符和重载函数

    C++ 重载运算符和重载函数 C++ 允许在同一作用域中的某个函数和运算符指定多个定义,分别称为函数重载和运算符重载. 重载声明是指一个与之前已经在该作用域内声明过的函数或方法具有相同名称的声明,但是 ...

  6. C++解析七-重载运算符和重载函数

    重载运算符和重载函数C++ 允许在同一作用域中的某个函数和运算符指定多个定义,分别称为函数重载和运算符重载.重载声明是指一个与之前已经在该作用域内声明过的函数或方法具有相同名称的声明,但是它们的参数列 ...

  7. C++ 基础语法 快速复习笔记(3)---重载函数,多态,虚函数

    1.重载运算符和重载函数: C++ 允许在同一作用域中的某个函数和运算符指定多个定义,分别称为函数重载和运算符重载. 重载声明是指一个与之前已经在该作用域内声明过的函数或方法具有相同名称的声明,但是它 ...

  8. Matlab匿名函数,子函数,私有函数,重载函数,eval和feval函数

    匿名函数,子函数,私有函数等函数类型 匿名函数: 匿名函数没有函数名,也不是.m文件,只包含一个表达式和输入输出参数. Fxy=@(x,y)x.^y+3*x*y x,y为输入输入参数,Fxy为函数名 ...

  9. 吴裕雄--天生自然C++语言学习笔记:C++ 重载运算符和重载函数

    C++ 允许在同一作用域中的某个函数和运算符指定多个定义,分别称为函数重载和运算符重载. 重载声明是指一个与之前已经在该作用域内声明过的函数或方法具有相同名称的声明,但是它们的参数列表和定义(实现)不 ...

随机推荐

  1. log4j的添加顺序

    1.log4j的架包 <dependency> <groupId>org.slf4j</groupId> <artifactId>slf4j-api&l ...

  2. mybatis框架的注意点

    1.在mapper.xml文件中 resultType:返回结果类型,只能用于单表 paramsType:接收参数类型,基本数据类型可省略 2.给实体类起别名 在mybatisConfig.xml配置 ...

  3. cvpr2018(转发一篇头条)

    CVPR 2018:腾讯图像去模糊.自动人像操纵最新研究 新智元 2018-05-29 14:13:04 新智元报道 来源:腾讯优图 编辑:江磊.克雷格 [新智元导读]即将在6月美国盐湖城举行的计算机 ...

  4. 【分布式缓存系列】集群环境下Redis分布式锁的正确姿势

    一.前言 在上一篇文章中,已经介绍了基于Redis实现分布式锁的正确姿势,但是上篇文章存在一定的缺陷——它加锁只作用在一个Redis节点上,如果通过sentinel保证高可用,如果master节点由于 ...

  5. Visual Studio 开发(一):安装配置Visual Studio Code

    一.为何使用Visual Studio Code 在学习音视频开发的时候,使用到了C和C++,在回顾复习C和C++的知识的时候,需要编写一些代码来加强理解. 虽然,有在线的语言编辑工具https:// ...

  6. 一次非常有趣的 SQL 优化经历

    阅读本文大概需要 6 分钟. 前言 在网上刷到一篇数据库优化的文章,自己也来研究一波. 场景 数据库版本:5.7.25 ,运行在虚拟机中. 课程表 #课程表 create table Course( ...

  7. 使用Consul 实现 MagicOnion(GRpc) 服务注册和发现

    1.下载打开Consul 笔者是windows下面开发的(也可以使用Docker). 官网下载windows的Consul https://www.consul.io/ 使用cmd窗口打开,输入con ...

  8. electron-builder 由于网络原因无法下载问题解决

    electron-builder 由于网络原因无法下载问题解决 在package.json的build中添加electron的镜像 "electronDownload": { &q ...

  9. Transaction rolled back because it has been marked as rollback-only

    出现这种错误的原因 1.接口A 调用了接口B 2.接口B报异常了,没有在B里面进行try catch捕获 3.接口A对 接口B进行了try catch捕获 因为接口B报异常 会把当前事物A接口的事物( ...

  10. 移动端点击事件300ms延迟问题解决方案——fastclick.js

    移动端点击事件300ms延迟的问题由来已久,如下截图 下面截图来自原文:https://www.jianshu.com/p/6e2b68a93c88 网上关于300ms延迟问题的解决方法,大致分为 3 ...