`
yeshaoting
  • 浏览: 686187 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

[不同编译器]函数参数求值顺序分析(附前置与后置++执行效率)

阅读更多

[不同编译器]函数参数求值顺序分析(附前置与后置++执行效率)


函数参数的求值顺序

 

函数参数的求值顺序跟不同语言编译器的处理方式有关,有些语言编译器从左向右处理,而有时从右往左处理.
这种编译器求值顺序的不同会相应地产生不同的输出结果.

测试程序:
向测试方法method(int a,int b)中传入类似于++x,x+y,x++的实参
.
e.g. method(++x,x+y),method(x+y,++x),...
根据输出结果的不同推出参数求值顺序.


. C语言编译环境

---------------------------------------前绪分析---------------------------------------

 

前置++程序:

 

// 前置++程序
// author: jarg
// http://jarg.iteye.com/

#include "stdio.h"

void main()
{
	int x = 2;
	printf("%d",++x);
	printf("%d",x);
}

 

C语言程序反汇编后的指令:

10:       int x = 2;
00401028   mov         dword ptr [ebp-4],2
11:       printf("%d",++x);
0040102F   mov         eax,dword ptr [ebp-4]
00401032   add         eax,1
00401035   mov         dword ptr [ebp-4],eax
00401038   mov         ecx,dword ptr [ebp-4]
0040103B   push        ecx
0040103C   push        offset string "%d" (00422fa4)
00401041   call        printf (00401090)
00401046   add         esp,8
12:       printf("%d",x);
00401049   mov         edx,dword ptr [ebp-4]
0040104C   push        edx
0040104D   push        offset string "%d" (00422fa4)
00401052   call        printf (00401090)
00401057   add         esp,8
13:   }

 

汇编指令解释:
第十条语句: int x = 2;对应汇编信息是mov dword ptr [ebp-4],2(dword ptr [ebp-4]是变量x的存储单元,语句是将值2放到变量x的存储单元中
).
第十一条语句(总共8条指令
):
0040102F~00401038:
变量x存储单元中的数值存入ax寄存器,然后ax寄存器中数值加1,再将ax寄存器中的数值存入变量x的存储单元,最后将变量x的存储单元中的数值存入cx寄存器中(总共4条指令
)
0040103B:
cx寄存器中的数值压入栈,作为函数的第二个实参值

0040103C:
将字符串"%d"压入栈,作为函数第一个实参值
00401041:
调用printf函数,执行函数过程

 

 

后置++程序:

// 后置++程序
// author: jarg
// http://jarg.iteye.com/

#include "stdio.h"

void main()
{
	int x = 2;
	printf("%d",x++);
	printf("%d",x);
}

 

 

 

 

 

C语言程序反汇编后的指令:

10:       int x = 2;
00401028   mov         dword ptr [ebp-4],2
11:       printf("%d",x++);
0040102F   mov         eax,dword ptr [ebp-4]
00401032   mov         dword ptr [ebp-8],eax
00401035   mov         ecx,dword ptr [ebp-8]
00401038   push        ecx
00401039   push        offset string "%d" (00422fa4)
0040103E   mov         edx,dword ptr [ebp-4]
00401041   add         edx,1
00401044   mov         dword ptr [ebp-4],edx
00401047   call        printf (00401090)
0040104C   add         esp,8
12:       printf("%d",x);
0040104F   mov         eax,dword ptr [ebp-4]
00401052   push        eax
00401053   push        offset string "%d" (00422fa4)
00401058   call        printf (00401090)
0040105D   add         esp,8
13:   } 

 

汇编指令解释:
第十条语句: int x = 2;对应汇编信息是mov dword ptr [ebp-4],2(dword ptr [ebp-4]是变量x的存储单元,语句是将值2放到变量x的存储单元中
).
第十一条语句(总共10条指令
):
0040102F~00401035:
变量x存储单元中的数值存入ax寄存器,再将ax寄存器中的数值存入变量x的存储单元的下一个存储单元,最后将变量x的存储单元的下一个存储单元中的数值存入cx寄存器中(总共3条指令
)
00401038:
cx寄存器中的数值压入栈,作为函数的第二个实参值

0040103C:
将字符串"%d"压入栈,作为函数第一个实参值
0040103E~00401044:
将变量x的存储单元中的数值存入dx寄存器,再将dx寄存器中的数值加1,最后最dx寄存器中的数值存入变量x的存储单元中
00401047:
调用printf函数,执行函数过程

 

结论(C语言编译环境下):
前置++与后置++程序不同就在于一个是++x,另一个是
x++;
这一区别造成了执行过程相差2条指令
.
由此可见,++x的执行效率高于x++.另外,前置++先执行x=x+1,然后将x值压入栈,做为函数实参值,调用函数;而后置++先将x值压入栈,做为函数实参值,然后执行x=x+1,再调用函数.

二个程序的printf函数的都是先处理第二个实参,压入栈;然后处理第一个实参,压入栈.
由此可见,C语言编译环境下函数求值顺序是从右向左
.
栈是一个先入后出的数据结构,最后一个实参先入栈,则会最后一个出栈,这样在函数从左向右顺序依次读取形参的时候,做出栈操作给对应的形参赋值.

 

---------------------------------------前绪分析---------------------------------------

 

 

 

 

 

C语言测试程序:

// 函数参数的求值顺序
// author: jarg
// http://jarg.iteye.com/
 
#include "stdio.h"

void main()
{
	int x = 2;
	printf("%d\t%d\t%d",x,++x,x++);
	printf("\t%d\n",x);
}

 

由前绪分析的结论已经得出: C语言编译环境下函数求值顺序是从右向左
初始化时x = 2,求值顺序是从右向左,则输出为3 3 2 4

输出结果:
3       3       2 4

 

 

. Java语言编译环境

 

Java语言测试程序:

// 函数参数的求值顺序
// author: jarg
// http://jarg.iteye.com/

public class Test
{
	public static void main(String[] args)
	{
		int x = 2;
		System.out.print("" + (x) + "\t" + (++x) + "\t" + (x++));
		System.out.print("\t" + x);
	}
}

 

初始化时x = 2,假设:
 
假设求值顺序是从左向右,则输出为
2 3 3 4
 
假设求值顺序是从右向左,则输出为3 3 2 4

 

输出结果:
2 3 3 4

 

结论(Java语言编译环境下):
由此可知,Java编译环境对函数求值顺序是按照函数参数声明顺序(符合我们正常思维逻辑)的从左向右
.
解释
:
虚拟机为每一个调用的Java(非本地)方法建立一个新的栈帧.栈帧包括:为方法的局部变量所预留的空间,该方法的操作数栈,以及特定虚拟机实现需要的其他所有的信息.局部变量和操作数栈大小在编译时计算出来,并设置到class文件中去,然后虚拟机就能够了解到方法的栈帧需要多少内存.当虚拟机调用一个方法的时候,它为该方法创建恰当大小的栈帆,再将新的栈帧压入Java
.
值得一提的Java编译器将Java方法的参数严格按照它们的声明顺序放到"局部变量数组"
.
处理方法时,虚拟机从所调用帧内的操作数栈中弹出实参值.虚拟机把对应的值存入局部变量数组参数对应位置,这种存入可以随机存入
.
C语言需要将后面参数得先入栈(先入后出结构),然后需要参数值的时候出栈供方法使用
.
这种不同导致编译器求值顺序的差异.

 

C语言编译环境下前置后置++执行过程类似,Java编译环境函数从左向右顺序求值情况下,++x在执行完x=x+1操作后才做传值操作,这就导致后面的x++中的x值变成3;表达式的输出结果也是3,不难看出x++是先做传值操作再完成的x=x+1操作,这一情况体现在下一条语句:System.out.print("\t" + x);,x输出的是4.
但是由于本人能力不够,不知道如何查看Java程序编译后类似于C语言汇编后的指令代码.因此,对于Java语言中二者执行效率孰高孰低不敢妄自定夺.姑且类推猜测前置++效率高于后置++.

 

 

本人才疏学浅,汇编描述以及Java编译环境结论解释的地方有不准确的地方,望海涵.

 

 

分享到:
评论

相关推荐

    C语言中的自加自减运算.pdf

    特别是在一些老旧的编译器(如文中提到的Turbo C)中,表达式参数的求值是从右向左进行的,这可能导致与现代编译器不同的行为。在现代编译器中,尽管大多数时候参数也是从右到左求值,但这种行为可能会依赖于具体的...

    C 面试问与答攻略

    在上述例子中,需要注意函数参数的求值顺序是未定义的,不同的编译器可能会有不同的结果。因此,在实际编程中应尽量避免这类复杂的表达式,以免引起混淆或错误。 ### 5. 参数求值顺序的不确定性 在C++中,函数参数...

    C语言程序自增自减问题的几点思考.pdf

    当表达式作为函数参数传递(如`printf`函数的参数)时,编译器可能采取不同的求值策略。 在TurboC环境下,如果表达式作为赋值给变量的一部分,那么求值顺序可能与表达式直接输出时不同。这会导致输出结果出现差异。...

    c语言自增与自减运算符共2页.pdf.zip

    这两种运算符有前缀形式(前置++或前置--)和后缀形式(后置++或后置--),它们在语义上略有不同。 1. 前缀运算符:前置自增(++var)和前置自减(--var) 当自增或自减运算符放在变量前面时,它们会在表达式计算...

    C++教程第07章 类与对象-8 自定义类中的运算符.docx

    - 成员函数重载时,单目运算符通常作用于当前对象,不需要额外参数(除了后置++和--)。 - 友元函数重载时,需要明确指定操作数,通常是类的引用或指针。 4. **二目运算符重载**: - 成员函数重载时,左操作数...

    详解C++编程中一元运算符的重载.doc

    前置形式的重载与其它一元运算符类似,而后置形式则需要多一个 `int` 类型的参数,这主要是为了与标准库中的实现保持一致,尽管这个参数通常未被实际使用。例如: ```cpp class Point { public: Point& operator++...

    C语言疑惑经验谈.pdf

    此外,`printf`函数的参数求值顺序在某些编译器中是从右到左,而在其他编译器中可能是从左到右,这可能导致输出结果的变化。 3. 字符和字符串的定界符 C语言中,单引号`' '`用于定义字符常量,而双引号`" "`用于...

    Java se 面试题.docx

    在给出的字节码中,我们可以看到自增操作的顺序与源代码中的顺序不完全相同,导致了不同的结果。 ### 单例设计模式 单例模式是一种确保一个类只有一个实例并且提供全局访问点的设计模式。它的主要目的是限制对象的...

    深信服+面试题+往年 (2).docx

    - `++`和`--`操作分为前置和后置,前置操作会先执行操作再返回值,后置操作会返回操作前的值。 - 使用异或交换两个变量的值:`A^=B; B^=A; A^=B;` 7. 语句和循环: - `for`循环中的`C`语句会在每次循环后执行。 ...

    c语言——常见错误PPT学习教案.pptx

    27. 忽略函数参数求值顺序:C语言中,函数参数的求值顺序是未定义的,可能导致意外结果。 28. 数组名与指针变量混淆:数组名是常量指针,不能直接赋值。 29. 结构体类型与结构体变量的误用:结构体类型不能直接赋值...

    探讨C语言中自增、自减运算符.pdf

    这是因为“先用后变”和“先变后用”的概念在顺序和时间上很难给出准确的定义,导致不同编译器对其处理方式可能会有所不同。 在教学过程中,王仕勋老师通过实际编程的例子发现了在使用自增、自减运算符时遇到的一些...

    你必须知道的495个C语言问题.pdf

    C语言标准未明确规定此类表达式的计算顺序,因此结果可能在不同的编译器上有所不同。 **3.2 表达式`i++*i++;`的结果** - 该表达式的结果取决于表达式中副作用的顺序。在C语言中,表达式中副作用的顺序是未定义的,...

Global site tag (gtag.js) - Google Analytics