zoukankan      html  css  js  c++  java
  • Delphi 循环的秘密

    技术交流,DH讲解.

    哈哈,我承认我是标题党.
    最近看天书夜读里面是对C语言的代码反汇编,不错,可惜Delphi没有这样的书,那我们自己动手吧.
    首先进入For循环了.

    Function TestFor( a, b: Integer ): Integer ;
    Var
      I: Integer ;
    Begin
      Result := a + b ;
      For I := 0 To 49 Do
        Result := Result + I ;
    End ;
    {$R *.dfm}
    
    Procedure TForm1.btn1Click( Sender: TObject ) ;
    Begin
      ShowMessage( IntToStr( TestFor( 1, 2 ) ) ) ;
    End ;

    反汇编代码:

    Unit4.pas.28: Result := a + b ;
    00523AE0 03D0             add edx,eax //执行a+b,但是我很奇怪为什么不用add eax,edx 这样最后就不用执行那句mov eax,edx了,节约一句
    Unit4.pas.29: For I := 0 To 49 Do
    00523AE2 33C0             xor eax,eax //清除eax,来充当I这个循环变量
    Unit4.pas.30: Result := Result + I ;
    00523AE4 03D0             add edx,eax //累计和,并把循环变量+1,这里就是循环体的执行代码
    00523AE6 40               inc eax
    Unit4.pas.29: For I := 0 To 49 Do
    00523AE7 83F832           cmp eax,$32 //边界比较,不等于就跳回去,继续执行循环体代码.
    00523AEA 75F8             jnz $00523ae4 //看来还是再写个函数来看看
    Unit4.pas.31: End ;
    00523AEC 8BC2             mov eax,edx
    00523AEE C3               ret 

    对比书中C的反汇编代码,Delphi反汇编出来的代码很精简,虽然声明了局部变量,但是函数直接用寄存器代替了.很好.
    书中C循环体先直接用Jmp跳过去与临界变量比较,然后再来看是否执行循环体,而改变循环变量它用了add eax,1,然后再把eax传给栈上面的I.
    后来我尝试改成0 to -1 Delphi编译器会识别出来,然后不会添加任何循环代码到函数中去的.

    接下来是do循环,也就是Repeat

    function TestRepeat(a,b:Integer):Integer ;
    var
      I:Integer;
    begin
      Result:=a + b;
      I:=0;
      repeat
        Result:= Result + I;
        Inc(I);
      until I = 50;
    end;

    反汇编代码:

    Unit4.pas.30: Result:=a + b;
    00523AE0 03D0             add edx,eax
    Unit4.pas.31: I:=0;
    00523AE2 33C0             xor eax,eax
    Unit4.pas.33: Result:= Result + I;
    00523AE4 03D0             add edx,eax
    Unit4.pas.34: Inc(I);
    00523AE6 40               inc eax
    Unit4.pas.35: until I = 50;
    00523AE7 83F832           cmp eax,$32
    00523AEA 75F8             jnz $00523ae4
    Unit4.pas.36: end;
    00523AEC 8BC2             mov eax,edx
    00523AEE C3               ret 

    我们发现反汇编代码和For是一样的.
    这段代码就和VC反编译出来的差不多了,先执行循环体然后再比较边界.
    好,看看while呢?

    Function TestWhile(A, B: Integer): Integer;
    Var
      I: Integer;
    Begin
      Result:= A+ B;
      I:= 0;
      While I< 50 Do
      Begin
        Result:= Result+ I;
        Inc(I);
      End;
    End;

    反汇编代码:

    Unit4.pas.38: Result:= A+ B;
    00523AE0 03D0             add edx,eax
    Unit4.pas.39: I:= 0;
    00523AE2 33C0             xor eax,eax
    Unit4.pas.42: Result:= Result+ I;
    00523AE4 03D0             add edx,eax
    Unit4.pas.43: Inc(I);
    00523AE6 40               inc eax
    Unit4.pas.40: While I< 50 Do
    00523AE7 83F832           cmp eax,$32
    00523AEA 7CF8             jl $00523ae4
    Unit4.pas.45: End;
    00523AEC 8BC2             mov eax,edx
    00523AEE C3               ret 

    我晕,居然还是一样的.也就是Delphi里面3种循环执行效率是一样的吧.
    而VC同样是先初始化循环变量,然后无条件跳转到边界检测,然后才执行循环体代码.
    循环就这么多,以后该是条件判断

  • 相关阅读:
    sql语句技巧
    逻辑查询处理的步骤
    left join 和 left outer join的区别
    SQL 笛卡尔积
    SQL 分类
    显示数据库中的表
    数据库备份 恢复
    增删主键及修改表名
    Securing Data笔记
    System Monitoring之"文件系统"
  • 原文地址:https://www.cnblogs.com/huangjacky/p/1663526.html
Copyright © 2011-2022 走看看