zoukankan      html  css  js  c++  java
  • 用汇编的眼光看C++(之类静态变量、静态函数) 四

    【 声明:版权所有,欢迎转载,请勿用于商业用途。  联系信箱:feixiaoxing @163.com】


        看过设计模式的朋友都知道,我们在设计单件模式的时候离不开类的静态函数。和类的成员变量不同,类的静态变量属于全部类对象数据;同样和类的成员函数不同,类的静态函数属于全部类函数共有。这句话读来想来有一些拗口,不过没有关系,我们可以通过一段代码来说明问题。

        (1)静态变量

        老规矩,我们首先对类进行初步定义,如下所示:

    1. class employee  
    2. {  
    3. public:  
    4.     employee() { }  
    5.     ~employee() {}  
    6.   
    7.     static int value;  
    8. };  
        那么,这里出现的value是不是所共有的,我们可以看看相关的函数代码即可:

    1. 67:   int employee::value = 0;  
    2. 68:  
    3. 69:   void process()  
    4. 70:   {  
    5. 00401240   push        ebp  
    6. 00401241   mov         ebp,esp  
    7. 00401243   push        0FFh  
    8. 00401245   push        offset __ehhandler$?process@@YAXXZ (0041f469)  
    9. 0040124A   mov         eax,fs:[00000000]  
    10. 00401250   push        eax  
    11. 00401251   mov         dword ptr fs:[0],esp  
    12. 00401258   sub         esp,48h  
    13. 0040125B   push        ebx  
    14. 0040125C   push        esi  
    15. 0040125D   push        edi  
    16. 0040125E   lea         edi,[ebp-54h]  
    17. 00401261   mov         ecx,12h  
    18. 00401266   mov         eax,0CCCCCCCCh  
    19. 0040126B   rep stos    dword ptr [edi]  
    20. 71:       employee m;  
    21. 0040126D   lea         ecx,[ebp-10h]  
    22. 00401270   call        @ILT+35(employee::employee) (00401028)  
    23. 00401275   mov         dword ptr [ebp-4],0  
    24. 72:       employee n;  
    25. 0040127C   lea         ecx,[ebp-14h]  
    26. 0040127F   call        @ILT+35(employee::employee) (00401028)  
    27. 73:  
    28. 74:       m.value = 10;  
    29. 00401284   mov         dword ptr [employee::value (00438494)],0Ah  
    30. 75:       n.value = 100;  
    31. 0040128E   mov         dword ptr [employee::value (00438494)],64h  
    32. 76:   }  
    33. 00401298   lea         ecx,[ebp-14h]  
    34. 0040129B   call        @ILT+0(employee::~employee) (00401005)  
    35. 004012A0   mov         dword ptr [ebp-4],0FFFFFFFFh  
    36. 004012A7   lea         ecx,[ebp-10h]  
    37. 004012AA   call        @ILT+0(employee::~employee) (00401005)  
    38. 004012AF   mov         ecx,dword ptr [ebp-0Ch]  
    39. 004012B2   mov         dword ptr fs:[0],ecx  
    40. 004012B9   pop         edi  
    41. 004012BA   pop         esi  
    42. 004012BB   pop         ebx  
    43. 004012BC   add         esp,54h  
    44. 004012BF   cmp         ebp,esp  
    45. 004012C1   call        __chkesp (004086b0)  
    46. 004012C6   mov         esp,ebp  
    47. 004012C8   pop         ebp  
    48. 004012C9   ret  
        我们直接看74行和75行。我们看到复制的对象地址都是惊人的一致,这说明实际上m和n所指的value实际上是同一个地址0x401005。不过,类的静态对象有一个要求,那就是对象的静态变量必须像全局变量一样进行初始化操作。

        (2)静态函数

        静态函数和静态变量一样,实际上就是类的全局函数。它之所以和普通的成员函数不一样,就是因为它不需要定义类的类型就能使用这个函数。根据上面的信息,我们可以重新定义一下这个类:

    1. class employee  
    2. {  
    3. public:  
    4.     employee() { }  
    5.     ~employee() {}  
    6.   
    7.     static void print() { printf("employee::print()!\n");};  
    8. };  
        那么类的静态函数是这样使用的呢?大家看看下面这样一个函数:

    1. 68:   void process()  
    2. 69:   {  
    3. 00401230   push        ebp  
    4. 00401231   mov         ebp,esp  
    5. 00401233   sub         esp,40h  
    6. 00401236   push        ebx  
    7. 00401237   push        esi  
    8. 00401238   push        edi  
    9. 00401239   lea         edi,[ebp-40h]  
    10. 0040123C   mov         ecx,10h  
    11. 00401241   mov         eax,0CCCCCCCCh  
    12. 00401246   rep stos    dword ptr [edi]  
    13. 70:       employee::print();  
    14. 00401248   call        @ILT+0(employee::print) (00401005)  
    15. 71:   }  
    16. 0040124D   pop         edi  
    17. 0040124E   pop         esi  
    18. 0040124F   pop         ebx  
    19. 00401250   add         esp,40h  
    20. 00401253   cmp         ebp,esp  
    21. 00401255   call        __chkesp (00408620)  
    22. 0040125A   mov         esp,ebp  
    23. 0040125C   pop         ebp  
    24. 0040125D   ret  
        静态函数不需要对应的类对象,所以也就不需要this指针。这就是成员函数和静态函数的区别,仅此而已。


  • 相关阅读:
    快速入门各种跨域
    常用知识点
    比较少用的格式
    git
    “没有用var声明的为全局变量”这种说法不准确
    类数组对象
    函数上下文的变量对象实例
    var a =10 与 a = 10的区别
    原型链与作用域链、执行上下文
    闭包的作用
  • 原文地址:https://www.cnblogs.com/sier/p/5676486.html
Copyright © 2011-2022 走看看