C++语法中的函数重载和默认参数

C语言中没有函数重载
C++语言中有函数重载

函数名相同,参数个数不同、参数类型不同、参数顺序不同

例如下面就是函数重载

void sum(int a, int b){
 cout << a+b << endl;
}

void sum(int a, double b){
 cout << a+b << endl;
}

返回值类型与函数重载无关

返回值类型与函数重载无关,下面代码不构成重载,编译会报错

//返回值类型与函数重载无关
int func(){
 return 0;
}

double func(){
 return 0;
}

实参的隐式类型转换可能会产生二义性

不同编译器有不同处理
下面代码在vs上编译不过,但是在Xcode中可以编译通过。

#include "iostream"
using namespace std;

void sum(double a){
 cout << a << endl;
}

void sum(int a){
 cout << a << endl;
}

int main(){
 sum(10);

 return 0;
}

函数重载的本质

采用了name mangling或者叫name decoration技术

  • C++编译器默认会对符号名(比如函数名)进行改编、修饰,有些地方翻译为“命名倾轧”
  • 重载时会生成多个不同的函数名,不同编译器(MSVC、g++)有不同的生成规则
  • 通过IDA打开【VS_Release_禁止优化】可以看到 或者通过hopper查看

源码

下面的代码

#include "iostream"
using namespace std;

void sum(double a){
 cout << a << endl;
}

void sum(int a){
 cout << a << endl;
}

int main(){
 return 0;
}

在代码中, void sum(double a){} 和 void sum(int a){} 是如何重载,调用函数的时候是如何能正确找到对应的函数呢?

汇编

我是用xcode的编译出可执行文件,放在hopper中查看

__Z3sumd:        // sum(double)
0000000100000ce0         push       rbp         ; CODE XREF=_main+23
0000000100000ce1         mov        rbp, rsp
0000000100000ce4         sub        rsp, 0x10
0000000100000ce8         mov        rdi, qword [__ZNSt3__14coutE_100001000]
0000000100000cef         movsd      qword [rbp+var_8], xmm0
0000000100000cf4         movsd      xmm0, qword [rbp+var_8]
0000000100000cf9         call       imp___stubs___ZNSt3__113basic_ostreamIcNS_11char_traitsIcEEElsEd ; std::__1::basic_ostream<char, std::__1::char_traits<char> >::operator<<(double)
0000000100000cfe         mov        rdi, rax
0000000100000d01         lea        rsi, qword [__ZNSt3__1L4endlIcNS_11char_traitsIcEEEERNS_13basic_ostreamIT_T0_EES7_]
0000000100000d08         call       __ZNSt3__113basic_ostreamIcNS_11char_traitsIcEEElsEPFRS3_S4_E ; std::__1::basic_ostream<char, std::__1::char_traits<char> >::operator<<(std::__1::basic_ostream<char, std::__1::char_traits<char> >& (*)(std::__1::basic_ostream<char, std::__1::char_traits<char> >&))
0000000100000d0d         mov        qword [rbp+var_10], rax
0000000100000d11         add        rsp, 0x10
0000000100000d15         pop        rbp
0000000100000d16         ret
                        ; endp
0000000100000d17         nop        word [rax+rax]

可知 void sum(double a){} 被编译器修改为函数 __Z3sumd

__Z3sumi:        // sum(int)
0000000100000da0         push       rbp
0000000100000da1         mov        rbp, rsp
0000000100000da4         sub        rsp, 0x10
0000000100000da8         mov        rax, qword [__ZNSt3__14coutE_100001000]
0000000100000daf         mov        dword [rbp+var_4], edi
0000000100000db2         mov        esi, dword [rbp+var_4]
0000000100000db5         mov        rdi, rax
0000000100000db8         call       imp___stubs___ZNSt3__113basic_ostreamIcNS_11char_traitsIcEEElsEi ; std::__1::basic_ostream<char, std::__1::char_traits<char> >::operator<<(int)
0000000100000dbd         mov        rdi, rax                                    ; argument #1 for method __ZNSt3__113basic_ostreamIcNS_11char_traitsIcEEElsEPFRS3_S4_E
0000000100000dc0         lea        rsi, qword [__ZNSt3__1L4endlIcNS_11char_traitsIcEEEERNS_13basic_ostreamIT_T0_EES7_]
0000000100000dc7         call       __ZNSt3__113basic_ostreamIcNS_11char_traitsIcEEElsEPFRS3_S4_E ; std::__1::basic_ostream<char, std::__1::char_traits<char> >::operator<<(std::__1::basic_ostream<char, std::__1::char_traits<char> >& (*)(std::__1::basic_ostream<char, std::__1::char_traits<char> >&))
0000000100000dcc         mov        qword [rbp+var_10], rax
0000000100000dd0         add        rsp, 0x10
0000000100000dd4         pop        rbp
0000000100000dd5         ret
                        ; endp
0000000100000dd6         nop        word [cs:rax+rax]

可知 void sum(int a){} 被编译器修改为函数 __Z3sumi

这样当我们调用的时候

int main(){
 sum(10.5);
 return 0;
}

汇编如下,可知:因为 10.5是double类型,调用函数的时候是调用 __Z3sumd

0000000100000de0         push       rbp
0000000100000de1         mov        rbp, rsp
0000000100000de4         sub        rsp, 0x10
0000000100000de8         movsd      xmm0, qword [0x100000f80]
0000000100000df0         mov        dword [rbp+var_4], 0x0
0000000100000df7         call       __Z3sumd        ; sum(double)
0000000100000dfc         xor        eax, eax
0000000100000dfe         add        rsp, 0x10
0000000100000e02         pop        rbp
0000000100000e03         ret
                        ; endp
0000000100000e04         nop        word [cs:rax+rax]
0000000100000e0e         nop

函数重载结论

由上面的汇编代码可知,当参数类型不同的时候,编译器会生成不同的函数名作为区别,这样就能实现函数重载。

默认参数

规则

C++允许函数设置默认参数,在调用时可以根据情况省略实参。规则如下:

  • 默认参数只能按照右到左的顺序
  • 如果函数同时有声明、实现,默认参数只能放在函数声明中
  • 默认参数的值可以是常量、全局符号(全局变量、函数名)

用法:如果函数的实参经常是同一个值,可以考虑使用默认参数

#include "iostream"
using namespace std;
void test(){
 cout << "test()" << endl;
}
// test2函数
// a没有默认值
// b 默认值是 10
// 最后一个参数默认值是个函数
void test2(int a, int b = 10, void (*func)() = test){
 cout << "a is " << a << endl;
 cout << "b is " << b << endl;
 func();
}
int main(){
 test2(3);
 return 0;
}

可能有冲突,二义性

函数重载、默认参数可能会产生冲突、二义性(建议优先选择使用默认参数)
例如下面的代码中, 调用 test(3); 会报错,因为不知道要执行哪个函数。

#include "iostream"
using namespace std;

void test(int a){
 cout << a << endl;
}

void test(int a,int b = 10){
 cout << a << endl;
}

int main(){
 test(3); // 这里报错,因为不知道要执行哪个函数
 test(10,20); //这一句可以正确
 return 0;
}

总结:如果函数的实参经常是同一个值,可以考虑使用默认参数

到此这篇关于C++语法之函数重载和默认参数的文章就介绍到这了,更多相关c++ 函数重载默认参数内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • C++中函数的默认参数详细解析

    使用方法:(1)在函数声明或定义时,直接对参数赋值,该参数就是默认参数.(2)在函数调用时,省略部分或全部参数,这时就会使用默认参数进行代替. 注意事项:(1)一般在声明函数是设置默认参数. 如果在函数声明和定义函数时都设置了默认参数,则以函数声明的默认参数为准. 复制代码 代码如下: #include<iostream>using namespace std;int main(){ double add(double a=3.2,double b=9.6);//在函数声明时设置默认参数 co

  • 解析C++中构造函数的默认参数和构造函数的重载

    C++构造函数的默认参数 和普通函数一样,构造函数中参数的值既可以通过实参传递,也可以指定为某些默认值,即如果用户不指定实参值,编译系统就使形参取默认值. [例] #include <iostream> using namespace std; class Box { public : Box(int h=10,int w=10,int len=10); //在声明构造函数时指定默认参数 int volume( ); private : int height; int width; int l

  • C++中函数重载实例详解

    C++中函数重载实例详解 函数重载: 1.具有相同的名称,执行基本相同的操作,但是使用不同的参数列表. 2.函数具有多态性. 3.编译器通过调用时参数的个数和类型确定调用重载函数的哪个定义. 4.只有对不同的数据集完成基本相同任务的函数才应重载. 函数重载的优 点 1.不必使用不同的函数名 2.有助于理解和调试代码 3.易于维护代码 接下来直接上代码: #include <iostream> using namespace std ; void say_hello(void) { cout &

  • C++默认参数与函数重载及注意事项

    一.默认参数 在C++中,可以为参数指定默认值.在函数调用时没有指定与形参相对应的实参时, 就自动使用默认参数. 默认参数的语法与使用: (1)在函数声明或定义时,直接对参数赋值.这就是默认参数: (2)在函数调用时,省略部分或全部参数.这时可以用默认参数来代替. 注意: (1)默认参数只可在函数声明中设定一次.只有在没有函数声明时,才可以在函数定义中设定.(#add ,此句意为存在函数声明和定义两部分的时候.验证表明有这个限制,可以随便,但出于规范,在声明中指定) (2)如果一个参数设定了缺省

  • 深入解析C++中的函数模板和函数的默认参数

    C++函数模板 我们知道,数据或数值可以通过函数参数传递,在函数定义时它们是未知的,只有在发生函数调用时才能确定其值.这就是数据的参数化. 其实,数据类型也可以通过参数来传递,在函数定义是可以不指明具体的数据类型,当发生函数调用时,编译器可以根据传入的参数自动确定数据类型.这就是数据类型参数化. 所谓函数模板,实际上是建立一个通用函数,其返回值类型和形参类型不具体指定,用一个虚拟的类型来代替(实际上是用一个标识符来占位).这个通用函数就称为函数模板(Function Template).凡是函数

  • 浅析C/C++中的可变参数与默认参数

    千万要注意,C不支持默认参数 C/C++支持可变参数个数的函数定义,这一点与C/C++语言函数参数调用时入栈顺序有关,首先引用其他网友的一段文字,来描述函数调用,及参数入栈: ------------ 引用开始 ------------ C支持可变参数的函数,这里的意思是C支持函数带有可变数量的参数,最常见的例子就是我们十分熟悉的printf()系列函数.我们还知道在函数调用时参数是自右向左压栈的.如果可变参数函数的一般形式是:    f(p1, p2, p3, -)那么参数进栈(以及出栈)的顺

  • C++语法中的函数重载和默认参数

    C语言中没有函数重载 C++语言中有函数重载 函数名相同,参数个数不同.参数类型不同.参数顺序不同 例如下面就是函数重载 void sum(int a, int b){ cout << a+b << endl; } void sum(int a, double b){ cout << a+b << endl; } 返回值类型与函数重载无关 返回值类型与函数重载无关,下面代码不构成重载,编译会报错 //返回值类型与函数重载无关 int func(){ retu

  • 深度探究C++中的函数重载的用法

    C++ 允许同一范围内具有相同名称的多个函数的规范.这些函数称为重载函数,"重载"中对其进行了详细介绍.利用重载函数,程序员可以根据参数的类型和数量为函数提供不同的语义. 例如,采用字符串(或 char *)参数的 print 函数执行的任务与采用"双精度"类型的参数的函数执行的任务截然不同.重载允许通用命名并使程序员无需创建名称,例如 print_sz 或 print_d.下表显示了 C++ 使用函数声明的哪些部分来区分同一范围内具有相同名称的函数组. 重载注意事

  • 通过实例理解javascript中没有函数重载的概念

    将函数名想象为指针,也有助于理解为什么ECMAScript中没有函数重载的概念.如下例子: 复制代码 代码如下: function addSomeNum(num) {     return num+100; } function addSomeNum(num) {     return num+200; } var result=addSomeNum(100);//300 显然,这个例子中声明了两个同名函数,而结果则是后面的函数覆盖了前面的函数.以上代码实际上与下面的代码是一致的. 复制代码 代

  • JavaScript中的函数重载深入理解

    在JavaScript中有一种特殊的数据类型---Function类型,JavaScript的每个函数都是Function类型的实例.由于函数是对象,因此函数名实际上也是一个指向函数对象的指针,不会与某个函数绑定. <pre name="code" class="html">function sum(num1,num2) { return num1 +num2; } alert(sum(10,10)); //20 var other = sum; ale

  • js模拟如何实现重载以及默认参数

    目录 模拟实现重载以及默认参数 下面先说实现重载的方法 接下来是实现默认参数的方法 下面进行一下测试 js的函数支持重载吗 模拟实现重载以及默认参数 众所周知,js是函数不支持重载和默认参数的,但是我们可以使用一些其他方法来模拟这个方法的实现. 首先看一下重载的定义:函数名相同,函数的参数列表不同(包括参数个数和参数类型),至于返回类型可同可不同. 以及默认参数的定义:默认参数指的是当函数调用中省略了实参时自动使用的一个值. 那么如何实现这两个功能呢,一个很简单的方法就是使用arguments来

  • PowerShell中的函数重载示例

    在PowerShell中,我们可以使用Function关键字来定义一个函数,比如: Function Get-Data { Param ( [int] $data = -1 ) return $data; } 这个名为Get-Data的函数接收一个int类型的参数$data,直接将其返回.那么,我同时还需要一个接收string类型的Get-Data函数怎么办?方法比较奇怪,至少和传统的C++之类的语言不同. 在PowerShell中定义重载函数,需要指定参数所属的Parameter Set,如下

  • 在Python中实现函数重载的示例代码

    假设你有一个函数connect,它有一个参数address,这个参数可能是一个字符串,也可能是一个元组.例如: connect('123.45.32.18:8080') connect(('123.45.32.18', 8080)) 你想在代码里面兼容这两种写法,于是你可能会这样写代码: def connect(address): if isinstance(address, str): ip, port = address.split(':') elif isinstance(address,

  • Python中使用partial改变方法默认参数实例

    Python 标准库中 functools库中有很多对方法很有有操作的封装,partial Objects就是其中之一,他是对方法参数默认值的修改. 下面就看下简单的应用测试. 复制代码 代码如下: #!/usr/bin/env python # -*- coding: utf-8 -*- #python2.7x #partial.py #authror: orangleliu ''' functools 中Partial可以用来改变一个方法默认参数 1 改变原有默认值参数的默认值 2 给原来没

  • 在VBScript中实现-函数/方法名作为参数传入另一个函数

    在JS中有这种用法,某个函数名可以当成参数的形式,传入到另外一个函数内部去,例如: <script type="text/javascript"> <!-- function myFuncA(str,myFuncB){  str = str + " 您好!";  str = myFuncB(str);  return str; } function myFuncB(str){  str = str + "欢迎来到IECN.NET"

随机推荐