从汇编看c++函数的默认参数的使用说明_C 语言

在c++中,可以为函数提供默认参数,这样,在调用函数的时候,如果不提供参数,编译器将为函数提供参数的默认值。下面从汇编看其原理。

下面是c++源码:

复制代码 代码如下:

int add(int a = 1, int b = 2) {//参数a b有默认值
    return a + b;
}
int main() {
   int c= add();//不提供参数

}

下面是mian函数里面的汇编码:

复制代码 代码如下:

; 4    : int main() {

    push    ebp
    mov    ebp, esp
    push    ecx;为局部变量c分配了4字节的存储空间 ecx为32位寄存器

; 5    :    int c= add();

    push    2;将2压栈,就是add函数中b参数的默认值 这里参数压栈方向是从右向左
    push    1;将1压栈,就是add函数中a参数的默认值
    call    ?add@@YAHHH@Z                ; 调用add函数
    add    esp, 8;释放刚才为add提供参数时的储存空间
    mov    DWORD PTR _c$[ebp], eax;eax寄存器里面存放add函数返回值,写入变量c里面

; 6    :   
; 7    : }

    xor    eax, eax
    mov    esp, ebp
    pop    ebp
    ret    0

下面是add函数的汇编码:

复制代码 代码如下:

?add@@YAHHH@Z PROC                    ; add

; 1    : int add(int a = 1, int b = 2) {

    push    ebp
    mov    ebp, esp

; 2    :     return a + b;

    mov    eax, DWORD PTR _a$[ebp];将参数a的值写入寄存器eax
    add    eax, DWORD PTR _b$[ebp];将参数b的值取出与eax里面的值相加,结果存入eax寄存器

; 3    : }

    pop    ebp
    ret    0
?add@@YAHHH@Z ENDP

下面是只提供一个参数值得情况

先看c++源码:

复制代码 代码如下:

int add(int a = 1, int b = 2) {//参数a b有默认值
    return a + b;
}
int main() {
   int a = 3;
   int c= add(a);//只为a提供参数

}

下面是main函数里面的汇编码:

复制代码 代码如下:

; 4    : int main() {

    push    ebp
    mov    ebp, esp
    sub    esp, 8;esp寄存器作为栈指针移动8字节,为局部变量a,c预留存储空间

; 5    :    int a = 3;

    mov    DWORD PTR _a$[ebp], 3;将3写入局部变量a所在存储空间

; 6    :    int c= add(a);//不提供参数

    push    2;将2压栈,提供b参数默认值
    mov    eax, DWORD PTR _a$[ebp];取出a的值,放入寄存器eax里面
    push    eax;将eax里面的值压栈,提供参数a的值,这里不是提供的默认值1
    call    ?add@@YAHHH@Z                ; 调用add函数
    add    esp, 8;释放刚才为调用函数add为参数分配的8byte空间
    mov    DWORD PTR _c$[ebp], eax;eax里面存放调用函数的结果,写到c所在存储空间里面

; 7    :   
; 8    : }

    xor    eax, eax
    mov    esp, ebp
    pop    ebp
    ret    0

可以看到,和上面的不同之处在于没有为a提供默认的参数值。

下面是add函数的汇编码,与第一种情况一样,没有变化:

复制代码 代码如下:

?add@@YAHHH@Z PROC                    ; add

; 1    : int add(int a = 1, int b = 2) {//参数a b有默认值

    push    ebp
    mov    ebp, esp

; 2    :     return a + b;

    mov    eax, DWORD PTR _a$[ebp]
    add    eax, DWORD PTR _b$[ebp]

; 3    : }

    pop    ebp
    ret    0
?add@@YAHHH@Z ENDP

这里由于为参数a给出了明确的值,因此编译器只为参数b提供默认值。可想而知,若为add函数都提供了明确的参数值,那么编译器就不会为参数a,b提供默认值了。

时间: 2024-09-19 17:43:22

从汇编看c++函数的默认参数的使用说明_C 语言的相关文章

C++中函数的默认参数详细解析_C 语言

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

实例讲解在C++的函数中变量参数及默认参数的使用_C 语言

包含变量参数列表的函数如果函数声明中最后一个成员是省略号 (...),则函数声明可采用数量可变的参数.在这些情况下,C++ 只为显式声明的参数提供类型检查.即使参数的数量和类型是可变的,在需要使函数泛化时也可使用变量参数列表.函数的系列是一个使用变量参数列表的函数的示例.printfargument-declaration-list 包含变量参数的函数 若要访问声明后的参数,请使用包含在标准包含文件 STDARG.H 中的宏(如下所述). 采用数量可变的参数的函数声明至少需要一个占位符参数(即使

详解C++编程中向函数传递引用参数的用法_C 语言

引用类型的函数参数向函数传递引用而非大型对象的效率通常更高. 这使编译器能够在保持已用于访问对象的语法的同时传递对象的地址. 请考虑以下使用了 Date 结构的示例: // reference_type_function_arguments.cpp struct Date { short DayOfWeek; short Month; short Day; short Year; }; // Create a Julian date of the form DDDYYYY // from a G

解析C++函数的默认参数和占位参数及较之C语言的拓展_C 语言

可以将占位参数与默认参数结合起来使用 意义: 为以后程序的扩展留下线索 兼容C语言程序中可能出现的不规范写法 //C++可以声明占位符参数,占位符参数一般用于程序扩展和对C代码的兼容 int func(int a, int b, int = 0) { return a + b; } void main() { //如果默认参数和占位参数在一起,都能调用起来 func(1, 2); func(1, 2, 3); system("pause"); } 默认参数对C的函数拓展1.C++中可以

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

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

详解C++编程中用数组名作函数参数的方法_C 语言

C++数组的概念 概括地说:数组是有序数据的集合.要寻找一个数组中的某一个元素必须给出两个要素,即数组名和下标.数组名和下标惟一地标识一个数组中的一个元素. 数组是有类型属性的.同一数组中的每一个元素都必须属于同一数据类型.一个数组在内存中占一片连续的存储单元.如果有一个整型数组a,假设数组的起始地址为2000,则该数组在内存中的存储情况如图所示. 引入数组就不需要在程序中定义大量的变量,大大减少程序中变量的数量,使程序精炼,而且数组含义清楚,使用方便,明确地反映了数据间的联系.许多好的算法都与

解析C++中的虚拟函数及其静态类型和动态类型_C 语言

虚拟函数是C++语言引入的一个很重要的特性,它提供了"动态绑定"机制,正是这一机制使得继承的语义变得相对明晰. (1)基类抽象了通用的数据及操作,就数据而言,如果该数据成员在各派生类中都需要用到,那么就需要将其声明在基类中:就操作而言,如果该操作对各派生类都有意义,无论其语义是否会被修改或扩展,那么就需要将其声明在基类中. (2)有些操作,如果对于各个派生类而言,语义保持完全一致,而无需修改或扩展,那么这些操作声明为基类的非虚拟成员函数.各派生类在声明为基类的派生类时,默认继承了这些非

C语言中函数与指针的应用总结_C 语言

1. 首先,在C语言中函数是一种function-to-pointer的方式,即对于一个函数,会将其自动转换成指针的类型. 复制代码 代码如下: #include<stdio.h> void fun(){} int main(void){   printf("%p %p %p\n", &fun, fun, *fun);   return 0;} -------------------------------------------------------------

Cocos2d-x学习笔记之CCScene、CCLayer、CCSprite的默认坐标和默认锚点实验_C 语言

结论:实践证明这三个东西的默认坐标都是0,0 默认锚点都是0.5,0.5. bool HelloWorld::init() { bool bRet = false; do { CC_BREAK_IF(! CCLayer::init()); CCSprite * sprite = CCSprite::create("image1.png"); this->addChild(sprite); //获得CCSprite的默认坐标,CCSprite设置坐标的时候是用自己的锚点位置占据在坐