C#中的函数(二) 有参有返回值的函数

接上一篇 C#中的函数(-) 无参无返回值的函数

http://www.cnblogs.com/fzxiaoyi/p/8502613.html

这次研究下C#中的函数(二) 有参有返回值的函数

依然写一个小例子,用来测试

跟上一个例子差不多,区别就是MyFunction有二个参数a,b,返回二个数相加的值

F5调试运行,中断后转到反汇编

这里很明显看到不同了

这里就得讲到参数传递的方式,参数从左向右依次存入寄存器ecx edx

但是不同的编程语言有不同的传递参数的方式,有空再写一篇文章介绍下

要是学过汇编的,到这里又有疑问了,要是每个实参都存储在寄存器中,那超过寄存器数量怎么办?

有疑惑就得测试,这是一贯原则,好吧再开个工程,写一个3个参数的相同函数

转到反汇编,Look一下, 第三个实参是直接压栈

那如果五个参数呢,再写一个函数测试下

到这就很明了了

总结一下: 如果传递的实参数量小于2个使用的ECX EDX寄存器,如果大于2个参数,

使用堆栈进行传递.

OK,明白了参数传递的方式回到最开始的例子,进一步分析MyFunction函数内部

F10单步执行二次,停在001F2DD5

然后F11进入函数内部查看

这里反汇编代码太长,就不截图了,直接贴反汇编代码

001F2DF8 push ebp         //前面这二句用来保存esp
001F2DF9 mov ebp,esp   //此时ebp指向栈顶
001F2DFB push edi
001F2DFC push esi
001F2DFD push ebx   //对edi esi ebx寄存器压栈进行保存

//这里开始就是我们研究的重点了,可以F10单步执行,顺带查看寄存器跟堆栈中值

//上一篇中这里是sub esp,2ch,而这里是3ch

//这里多出来的10h是返回值地址,二个形参地址,一个临时变量sum地址

001F2DFE sub esp,3Ch //接合上面那三句,此时esp移动了0xC + 0x3C = 0x48个字节

001F2E01 mov esi,ecx      //这个ecx 是1, 代表第一个实参,保存在esi中

001F2E03 lea edi,[ebp-38h]   // 取地址操作esp + 0x10 = 从栈顶往下第四个位置
001F2E06 mov ecx,0Bh           //这三行把0xB* 4 = 44个字节空间清零
001F2E0B xor eax,eax       
001F2E0D rep stos dword ptr es:[edi]
001F2E0F mov ecx,esi           //把参数1保存在ecx 
001F2E11 mov dword ptr [ebp-3Ch],ecx    // 把参数1保存在[ebp-0x3C] 处
001F2E14 mov dword ptr [ebp-40h],edx    //把参数2保存在[ebp-0x40h]处
001F2E17 cmp dword ptr ds:[0018C7A8h],0
001F2E1E je 001F2E25
001F2E20 call 71C6CB2D
001F2E25 xor edx,edx
001F2E27 mov dword ptr [ebp-48h],edx
001F2E2A xor edx,edx
001F2E2C mov dword ptr [ebp-44h],edx
001F2E2F nop

//这里就是具体计算的地方
22: int sum = a + b;

001F2E30 mov eax,dword ptr [ebp-3Ch] //把第一个参数存储在eax中
001F2E33 add eax,dword ptr [ebp-40h]  //然后通过add把eax + 第二个参数,结果就是1 + 2 = 3
001F2E36 mov dword ptr [ebp-44h],eax  //然后把相加的结果放在临时变量中
001F2E39 mov eax,dword ptr [ebp-44h]  //又从临时变量中取出结果放在eax(这句完全多余)
001F2E3C mov dword ptr [ebp-48h],eax  //把结果放在返回值地址中. 有返回值情况下eax存储的都是返回值

//这是当前堆栈中的情况

//这后面的代码是当前函数返回时必须要做的堆栈平衡处理

001F2E3F nop
001F2E40 jmp 001F2E42
001F2E42 mov eax,dword ptr [ebp-48h]
001F2E45 lea esp,[ebp-0Ch]
001F2E48 pop ebx
001F2E49 pop esi
001F2E4A pop edi
001F2E4B pop ebp
001F2E4C ret

多于二个参数时,也顺带看一下核心代码

前面说过,多余二个参数时,前二个参数存储在寄存器ECX EDX中,后面的参数存储在堆栈中

18: int sum = a + b + c + d;
001A34B0 mov eax,dword ptr [ebp-3Ch]    //eax = 第一个参数
001A34B3 add eax,dword ptr [ebp-40h]    //eax = eax + 第二个参数
001A34B6 add eax,dword ptr [ebp+0Ch]   //eax += 第三个参数 直接从堆栈中取不需要再开辟栈空间
001A34B9 add eax,dword ptr [ebp+8]       //eax += 第四个参数 直接从堆栈中取不需要再开辟栈空间
001A34BC mov dword ptr [ebp-44h],eax   // 临时变量sum = 前四个数之和
19: return sum;                          
001A34BF mov eax,dword ptr [ebp-44h]   //把临时变量sum给eax
001A34C2 mov dword ptr [ebp-48h],eax   //把结果放在返回值地址中. 有返回值情况下eax存储的都是返回值
001A34C5 nop
001A34C6 jmp 001A34C8

原文地址:https://www.cnblogs.com/fzxiaoyi/p/8503231.html

时间: 2024-08-08 09:41:53

C#中的函数(二) 有参有返回值的函数的相关文章

<10> 无参无返回值+ 无参有返回值函数的定义+有参无返回值函数定义+ 有参有返回值函数定义+函数的参数+函数的参数传递过程

无参无返回值: 1 #include <stdio.h> 2 3 4 /** 5 * 定义一个无参无返值函数 6 * 7 */ 8 void print_line(){ 9 10 printf("-----------------\n"); 11 12 } 13 14 15 16 int main(int argc, const char * argv[]) { 17 //调用函数 18 print_line(); 19 20 return 0; 21 } 无参有返回值函数

C#中的函数(-) 无参无返回值的函数

分析下C#中的函数 先写一个小例子,一个静态函数,无返回值,无形参 在第17行与20行分别下断点 F5调试运行,此时中断在第17行MyFunction(), 在第17行右键反汇编,看下反汇编代码 这里的Call 002D0C30  对应用就是MyFunction() 此时F11直接进入MyFunction函数内部查看我们函数 002D2CA0就是MyFunction的入口,这里就有疑问了,明明上面Call 002D0C30,这里为什么是002D2CA0呢? 在地址栏上转到002D0C30,结果V

python 最简单的函数(无参无返回值)

#定义了一个无参无返回值的函数def myPrint(): print("sunck is a very good man!") print("sunck is a nice man!") print("sunck is a handsome man!") myPrint()myPrint()myPrint()myPrint()myPrint() 原文地址:https://www.cnblogs.com/pygo/p/12243029.html

Swift2.0语言教程之函数的返回值与函数类型

Swift2.0语言教程之函数的返回值与函数类型 Swift2.0中函数的返回值 根据是否具有返回值,函数可以分为无返回值函数和有返回值函数.以下将会对这两种函数类型进行讲解. Swift2.0中具有一个返回值的函数 开发者希望在函数中返回某一数据类型的值,必须要在函数声明定义时为函数设定一个返回的数据类型,并使用return语句进行返回.其中,return语句的一般表示形式如下: return 表达式 其中,表达式可以是符合Swift标准的任意表达式.而具有返回值的函数声明定义形式如下: fu

使用在类中定义的有返回值的函数,如何得到执行结果反馈?

我们做开发时,程序执行完后最好会有个反馈结果,尤其是在庞大的程序中,如果有bug可以及时发现,不然很浪费时间...最近做sde的二次开发,创建数据集时要用到带有返回值的函数,因为在数据集上创建要素类需要用到这个数据集,声明一个全局变量,可以直接用这个返回值.但是在捕获错误方面受到了限制,因为有返回值的函数每一个节点都要有返回值,想直接得到反馈是不行的,想在方法执行后写结果反馈的代码是检测不到的. 只能向师傅求救,给我提供了两种方法解决,感觉很巧妙. 原来的代码: Form1中的代码: priva

javascript函数(声明,传参,返回值,递归)

javascript函数(声明,传参,返回值,递归) 1.函数的基本概念 函数:是由事件驱动的或者当他被调用时可执行的可重复使用的代码块. 空调是由遥控器控制或者当开关打开时,可运行的家用电器(工具) 特点: 封装代码----使代码更简洁 重复使用---在重复功能的时候直接调用就好 执行时机---随时可以在我们想要执行的时候执行 2.函数的创建和执行 1. 函数的创建 1.声明式 函数声明的关键字 : ==function== function 关键字 和 var 关键字的行为几乎一致,都会在内

Java 中带参带返回值方法的使用

如果方法既包含参数,又带有返回值,称为带参带返回值的方法. 例: 实现功能:将考试成绩排序并输出,返回成绩的个数 实现思路: 1. 定义一个包含整型数组参数的方法,用来传入成绩数组 2. 在方法体中使用 Arrays.sort( ) 方法对成绩数组进行排序,然后使用 Arrays.toString( ) 方法将数组转换为字符串并输出,最后使用 return 返回数组中元素的个数. 3. 调用方法时需要先创建对象,然后再调用.调用时为其传入成绩数组,并获取方法的返回值,保存在变量中,最后输出成绩的

Java 中无参带返回值方法的使用

如果方法不包含参数,但有返回值,我们称为无参带返回值的方法. 例如:下面的代码,定义了一个方法名为 calSum ,无参数,但返回值为 int 类型的方法,执行的操作为计算两数之和,并返回结果 在 calSum( ) 方法中,返回值类型为 int 类型,因此在方法体中必须使用 return 返回一个整数值. 调用带返回值的方法时需要注意,由于方法执行后会返回一个结果,因此在调用带返回值方法时一般都会接收其返回值并进行处理.如: 运行结果为: 两数之和为:17 不容忽视的“小陷阱”: 1. 如果方

慕课网-Java入门第一季-7-2 Java 中无参无返回值方法的使用

来源:http://www.imooc.com/code/1578 如果方法不包含参数,且没有返回值,我们称为无参无返回值的方法. 方法的使用分两步: 第一步,定义方法 例如:下面代码定义了一个方法名为 show ,没有参数,且没有返回值的方法,执行的操作为输出 “ welcome to imooc. ” 注意哦: 1. 方法体放在一对大括号中,实现特定的操作 2. 方法名主要在调用这个方法时使用,需要注意命名的规范,一般采用第一个单词首字母小写,其它单词首字母大写的形式 第二步,调用方法 当需