用汇编的眼光看C++(之类静态变量、静态函数)13

【 声明:版权所有,欢迎转载,请勿用于商业用途。  联系信箱: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指针。这就是成员函数和静态函数的区别,仅此而已。

 

 

【预告: 下面一篇博客介绍类的算术重载符】

你可能感兴趣的:(静态变量)