下面先来看c++的源码:

#include <cstdio>
using namespace std; class X {
public:
int get1() {
return ;
}
virtual int get2() {
return ;
}
virtual int get3() {
return ;
}
}; int main() {
X x;
X* xp = &x;
int(X::*gp1)() = &X::get1;
int(X::*gp2)() = &X::get2;
int(X::*gp3)() = &X::get3;
/*********************输出各个成员函数指针的值*****************/
printf("gp1 = %lu\n", gp1);
printf("gp2 = %lu\n", gp2);
printf("gp3 = %lu\n", gp3); /********************用成员函数指针调用虚函数*******************/
(x.*gp1)();
(xp->*gp1)();
(x.*gp2)();
(xp->*gp2)();
(x.*gp3)();
(xp->*gp3)();
}

类X有3个成员函数,其中get1是普通的成员函数,而get2和get3都分别是虚成员函数。在main函数里面分别定义了指向这三个成员函数的指针,并且将他们的值输出来。然后用成员指针对他们进行了调用。下面来看调用后的结果:

可以看到,gp1 gp2 gp3输出的好像都是地址。下面主要来看一下面函数里面,定义三个成员函数指针的汇编码:

; 20   :     int(X::*gp1)() = &X::get1;

    mov    DWORD PTR _gp1$[ebp], OFFSET ?get1@X@@QAEHXZ ; X::get1;取?get1@X@@QAEHXZ所代表的内存地址,即X::get1的地址给gp1

; 21   :     int(X::*gp2)() = &X::get2;

    mov    DWORD PTR _gp2$[ebp], OFFSET ??_9X@@$BA@AE ; X::`vcall'{0}'; 取??_9X@@$BA@AE所代表的内存地址,即X::`vcall'{0}'的地址给gp2

; 22   :     int(X::*gp3)() = &X::get3;

    mov    DWORD PTR _gp3$[ebp], OFFSET ??_9X@@$B3AE ; X::`vcall'{4}';取??_9X@@$B3AE所代表的内存地址,即X::`vcall'{4}'的地址给gp3

通过汇编码可以看到,gp1存储的确实是成员函数get1的地址,而gp2和gp3存储的确不是虚函数get2和get3的地址,而是X::vcall{0}和X::vcall{4}的地址。那么,X::vcall{0}和vcall{4}到底是什么呢?我们继续看汇编代码,接下来是X::vcall{0}的汇编码:

??_9X@@$BA@AE PROC                    ; X::`vcall'{0}', COMDAT
mov eax, DWORD PTR [ecx];寄存是ecx里面保存的是对象x的首地址,
;因此,这里是将对象x首地址处内存内容(即vftable首地址)给寄存器eax
jmp DWORD PTR [eax];跳转到vftable首地址处内存(里面存的是虚函数get2的地址)所存储的地址处执行
;这里就是跳转去执行虚函数get2
??_9X@@$BA@AE ENDP ; X::`vcall'{0}'

通过汇编码,我们可以发现,X::vcall{0}是一段代码,它的作用是跳转到相应的虚函数地址去执行。

下面是X::vcall{4}的汇编码:

??_9X@@$B3AE PROC                    ; X::`vcall'{4}', COMDAT
mov eax, DWORD PTR [ecx];寄存器ecx里面存储的是对象x的首地址
;因此,这里是将对象x首地址处内存内容(即vftable首地址)给寄存器eax
jmp DWORD PTR [eax+];跳转到偏移vftable首地址处4byte处内存(里面存的是虚函数get2的地址)所存储的地址处执行
;这里就是跳转去执行虚函数get3
??_9X@@$B3AE ENDP ; X::`vcall'{4}'

通过汇编码,我们返现,X::vcall{4}和X::vcall{0}的作用一样。

因此,gp2和gp3存储的不是虚函数get2和虚函数get3的地址,而是相应的vcall函数的地址。那么,通过这些成员函数的指针调用函数的时候,有发生了什么?

下面是调用函数的汇编代码:

; 28   :     /********************用成员函数指针调用虚函数*******************/
; 29 : (x.*gp1)(); lea ecx, DWORD PTR _x$[ebp];取对象x的首地址,给寄存器ecx,作为遗憾参数传递给get1
call DWORD PTR _gp1$[ebp];gp1中存有get1的地址,这里直接调用get1函数 ; 30 : (xp->*gp1)(); mov ecx, DWORD PTR _xp$[ebp];指针变量xp保存有对象x的首地址,这里将对象首地址给寄存器ecx 作为隐含参数传递给get1
call DWORD PTR _gp1$[ebp];gp1中存有get1的地址,这里直接调用get1函数
;用指针和对象操作成员变量指针效果一样 ; 31 : (x.*gp2)(); lea ecx, DWORD PTR _x$[ebp];取对象x的首地址,给寄存器ecx,作为隐含参数传递给X::vcall{0}
call DWORD PTR _gp2$[ebp];gp2中存有X::vcall{0}的地址,这里调用X::vcall{0},由vcall{0}查询虚表,执行get2 ; 32 : (xp->*gp2)(); mov ecx, DWORD PTR _xp$[ebp];用指针调用和用对象x调用效果一样
call DWORD PTR _gp2$[ebp] ; 33 : (x.*gp3)(); lea ecx, DWORD PTR _x$[ebp];将对象x首地址给寄存器ecx,作为隐含参数传递给X::vcall{4}
call DWORD PTR _gp3$[ebp];gp3中存有X::vcall{4}的地址,这里调用X::vcall{4},由vcall{4}查询虚表,调用get3 ; 34 : (xp->*gp3)(); mov ecx, DWORD PTR _xp$[ebp];指针调用和对象调用效果一样
call DWORD PTR _gp3$[ebp]

通过汇编码发现,普通成员函数时通过地址直接调用,而虚成员函数时通先调用vcall函数,然后由vcall函数查询虚表调用相应的虚函数

由此可以看出,一个类里面的每一给虚函数都有一个vcall函数与之对应,通过vcall函数来调用相应的虚函数。

从汇编看c++中成员函数指针(一)的更多相关文章

  1. 为什么 C++ 中成员函数指针是 16 字节?

    当我们讨论指针时,通常假设它是一种可以用 void * 指针来表示的东西,在 x86_64 平台下是 8 个字节大小.例如,下面是来自 维基百科中关于 x86_64 的文章 的摘录: Pushes a ...

  2. 从汇编看c++成员函数指针(三)

    前面的从汇编看c++中成员函数指针(一)和从汇编看c++成员函数指针(二)讨论的要么是单一类,要么是普通的多重继承,没有讨论虚拟继承,下面就来看一看,当引入虚拟继承之后,成员函数指针会有什么变化. 下 ...

  3. 从汇编看c++成员函数指针(二)

    下面先看一段c++源码: #include <cstdio> using namespace std; class X { public: virtual int get1() { ; } ...

  4. 成员函数指针与高效C++委托 (delegate)

    下载实例源代码 - 18.5 Kb 下载开发包库文件 - 18.6 Kb 概要 很遗憾, C++ 标准中没能提供面向对象的函数指针. 面向对象的函数指针也被称为闭包(closures) 或委托(del ...

  5. 成员函数指针与高性能C++委托

    1 引子 标准C++中没有真正的面向对象的函数指针.这一点对C++来说是不幸的,因为面向对象的指针(也叫做“闭包(closure)”或“委托(delegate)”)在一些语言中已经证明了它宝贵的价值. ...

  6. [转]成员函数指针与高性能的C++委托

    原文(作者:Don Clugston):Member Function Pointers and the Fastest Possible C++ Delegates 译文(作者:周翔): 成员函数指 ...

  7. 从汇编看c++中指向成员变量的指针(二)

    在从汇编看c++中指向成员变量的指针(一)中讨论的情形没有虚拟继承,下面来看看,当加入了虚拟继承的时候,指向成员变量的指针有什么变化. 下面是c++源码: #include <iostream& ...

  8. 函数指针和成员函数指针有什么不同,反汇编带看清成员函数指针的本尊(gcc@x64平台)

    函数指针是什么,可能会答指向函数的指针. 成员函数指针是什么,答指向成员函数的指针. 成员函数指针和函数指针有什么不同? 虚函数指针和非虚成员函数指针有什么不同? 你真正了解成员函数指针了吗? 本篇带 ...

  9. 关于C++中的非静态类成员函数指针

    昨天发现了一个问题,就是使用对类中的非静态成员函数使用std::bind时,不能像普通函数一样直接传递函数名,而是必须显式地调用&(取地址),于是引申出我们今天的问题:非静态类成员函数指针和普 ...

随机推荐

  1. MFC让控件随窗口大小而改变

    转载自http://blog.csdn.net/chw1989/article/details/7488711 大小和位置都改变(亲测可行) 1.首先为窗体类添加CRect m_rect,该成员变量用 ...

  2. Linux中Firefox——Firebug插件安装及使用

    Firebug的安装方法即打开方式同httpfox Firebug使用指南: Firebug可以随时编辑页面:在HTML标签中,点击窗口上方的"inspect"命令,然后再选择页面 ...

  3. 13年山东省赛 Mountain Subsequences(dp)

    转载请注明出处: http://www.cnblogs.com/fraud/          ——by fraud Mountain Subsequences Time Limit: 1 Sec   ...

  4. (原)caffe中fine tuning及使用snapshot时的sh命令

    转载请注明出处: http://www.cnblogs.com/darkknightzh/p/5946041.html 参考网址: http://caffe.berkeleyvision.org/tu ...

  5. Mysql开发技巧之删除重复数据

    Mysql利用联表查询和分组来删除重复数据 //删除表中重复的id,保留最大的id mysql> select * from user; +----+------+ | id | name | ...

  6. php缩放gif和png格式透明背景变成黑色的解决方法

    在对gif或png格式的图片进行缩放等操作时,原本透明背景的图片最后都变成黑色的,解决办法 $img = imagecreatetruecolor(, ); //2.上色 $color=imageco ...

  7. python3.4+pyspider爬58同城(二)

    之前使用python3.4+selenium实现了爬58同城的详细信息,这次用pyspider实现,网上搜了下,目前比较流行的爬虫框架就是pyspider和scrapy,但是scrapy不支持pyth ...

  8. 2015年网易考拉海淘android面试

    经朋友推荐,昨天下午去网易杭州公司参加了考拉海淘android客户端的面试.今天回忆一下面试题目,做个整理进行备案. 1.说说JVM垃圾回收机制. 1.1.画了JVM分代回收的图,大致说了下垃圾分代回 ...

  9. 懒猫们终究要付出代码(本领是一生的),鲸鱼们的短视(逐小利而暴死)——这么说我应该只去互联网公司:IM,云存储,邮箱(别的一概不考虑)

    摘自周鸿伟的书,好像:

  10. JS扩展方法

    JS扩展方法与C#的扩展方法非常相似,也是可以链式调用的,也是通过对某个类的扩展写法来实现.这个东西非常好用,如果将预先写好的方法放到一个js里面引用的话,那么后面写js将非常有趣. 下面给出一个例子 ...