C++抽象基类多重继承时,成员函数地址不一致问题(详见程序)

Intel0011 2017-11-14 03:35:55
#define  INITGUID
#include <Windows.h> // Define HRESULT
#include <iostream>
#include <objbase.h> // Define interface
#include <iomanip>
using namespace std;

interface IDictionary : public IUnknown
{
virtual void _stdcall Initialize() = 0;
virtual void _stdcall Uninitialize() = 0;
};

interface ISpellCheck : public IUnknown
{
virtual void _stdcall CheckWord() = 0;
};

class CDictionary : public IDictionary, public ISpellCheck
{
public:
virtual HRESULT _stdcall QueryInterface(const IID&, void** ppv)
{
static int i;
cout << "QueryInterface " << i++ << endl;
return 0;
};
virtual ULONG _stdcall AddRef() { cout << "AddRef" << endl; return 0; };
virtual ULONG _stdcall Release() { cout << "Release" << endl; return 0; };

virtual void _stdcall Initialize() { cout << "Initialize" << endl; };
virtual void _stdcall Uninitialize() { cout << "Uninitialize" << endl; };

virtual void _stdcall CheckWord() { cout << "CheckWord" << endl; };
int a;
};

// {3340DDDC-490B-4a3c-9CC9-0E3DB217D36A}
DEFINE_GUID(MyIID, 0x3340dddc, 0x490b, 0x4a3c, 0x9c, 0xc9, 0xe, 0x3d, 0xb2, 0x17, 0xd3, 0x6a);


int main(void)
{
CDictionary *pA = new CDictionary;

cout << "对象地址:" << (int*)pA << endl;
cout << "第1个ppv地址:" << (int*)pA << endl;
cout << "第1个ppv地址内容:" << setiosflags(ios::uppercase) << setw(8) << setfill('0') << hex << *(int*)pA << endl;

cout << "第2个ppv地址:" << (int*)pA+1 << endl;
cout << "第2个ppv地址内容:" << setiosflags(ios::uppercase) << setw(8) << setfill('0') << hex << *((int*)pA+1) << endl;

cout << "虚函数表 — 第1个函数指针:" << setiosflags(ios::uppercase) << setw(8) << setfill('0') << hex << ((int*)*(int*)pA+0) << endl;

cout << "虚函数表 — 第1个函数地址:" << setiosflags(ios::uppercase) << setw(8) << setfill('0') << hex << *((int*)*(int*)pA+0) << endl;
cout << "虚函数表 — 第2个函数地址:" << setiosflags(ios::uppercase) << setw(8) << setfill('0') << hex << *((int*)*(int*)pA+1) << endl; // 为啥地址和下面的不一样
cout << "虚函数表 — 第3个函数地址:" << setiosflags(ios::uppercase) << setw(8) << setfill('0') << hex << *((int*)*(int*)pA+2) << endl;
cout << "虚函数表 — 第4个函数地址:" << setiosflags(ios::uppercase) << setw(8) << setfill('0') << hex << *((int*)*(int*)pA+3) << endl;
cout << "虚函数表 — 第5个函数地址:" << setiosflags(ios::uppercase) << setw(8) << setfill('0') << hex << *((int*)*(int*)pA+4) << endl;
cout << "虚函数表 — 第6个函数地址:" << setiosflags(ios::uppercase) << setw(8) << setfill('0') << hex << *((int*)*(int*)pA+5) << endl;

cout << "虚函数表 — 第2个函数指针:" << setiosflags(ios::uppercase) << setw(8) << setfill('0') << hex << ((int*)*((int*)pA+1)) << endl;

cout << "虚函数表 — 第1个函数地址:" << setiosflags(ios::uppercase) << setw(8) << setfill('0') << hex << *((int*)*((int*)pA+1)+0) << endl; // 为啥地址和上面的不一样
cout << "虚函数表 — 第2个函数地址:" << setiosflags(ios::uppercase) << setw(8) << setfill('0') << hex << *((int*)*((int*)pA+1)+1) << endl;
cout << "虚函数表 — 第3个函数地址:" << setiosflags(ios::uppercase) << setw(8) << setfill('0') << hex << *((int*)*((int*)pA+1)+2) << endl;
cout << "虚函数表 — 第4个函数地址:" << setiosflags(ios::uppercase) << setw(8) << setfill('0') << hex << *((int*)*((int*)pA+1)+3) << endl;
cout << "虚函数表 — 第5个函数地址:" << setiosflags(ios::uppercase) << setw(8) << setfill('0') << hex << *((int*)*((int*)pA+1)+4) << endl;

cout << "数据成员地址: " << &pA->a << endl;

IDictionary *e = pA;
ISpellCheck *f = pA;

e->QueryInterface(MyIID, NULL); // 确实是调用同一个函数
f->QueryInterface(MyIID, NULL);

delete pA;

return 0;
}
...全文
249 7 打赏 收藏 转发到动态 举报
AI 作业
写回复
用AI写文章
7 条回复
切换为时间正序
请发表友善的回复…
发表回复
Intel0011 2017-11-15
  • 打赏
  • 举报
回复
引用 3 楼 Saleayas 的回复:
当然不一样的。 他们都在自己的虚表中,都是自己的虚函数。 而你看到的 QI 函数是 这个类的实例函数,这个函数不是虚函数的。是一个类的实实在在的函数。 每一个定义到续表中的函数,都在 C++ 中有自己的定义的。这些函数入口被写入虚表中。这些函数在修正 this 之后再调用 QI 这个函数。 也就是说 IUnknown 的函数每一个接口都有自己的,而类本身也有自己的。
能否画个简单的内存布局示例图来讲解一下,尤其是几个虚表,对C++不熟悉,感觉编译器在幕后干了太多的事情
mLee79 2017-11-15
  • 打赏
  • 举报
回复
COM 是二进制规范, 不使用虚表的 C++ 编译器没法愉快的玩, 现在活着的 C++ 编译器没有不用虚表的 。 你没见 两个类型的 QI 函数参数列表都不一样么, this 指针都要调整为 CDictionary* 类型再调用你写的 QI函数, 只是在单继承的时候, 这个调整为 0
Saleayas 2017-11-15
  • 打赏
  • 举报
回复
当然不一样的。 他们都在自己的虚表中,都是自己的虚函数。 而你看到的 QI 函数是 这个类的实例函数,这个函数不是虚函数的。是一个类的实实在在的函数。 每一个定义到续表中的函数,都在 C++ 中有自己的定义的。这些函数入口被写入虚表中。这些函数在修正 this 之后再调用 QI 这个函数。 也就是说 IUnknown 的函数每一个接口都有自己的,而类本身也有自己的。
开心秋水 2017-11-15
  • 打赏
  • 举报
回复
虚函数表只是虚函数的一种实现方式而已,没有明确的证据能够证明你使用的编译器就是使用虚函数表的形式。
Intel0011 2017-11-15
  • 打赏
  • 举报
回复
引用 6 楼 iloveyou418 的回复:
一起学习~https://www.cnblogs.com/hushpa/p/5707475.html看看这个是不是你想要的呀
多谢,和我的问题不太一样
CT8100 2017-11-15
  • 打赏
  • 举报
回复
一起学习~https://www.cnblogs.com/hushpa/p/5707475.html看看这个是不是你想要的呀
赵4老师 2017-11-14
  • 打赏
  • 举报
回复
《深度探索C++对象模型》 《C++反汇编与逆向分析技术揭秘》
部分程序未完成所有功能,,,东华理工2011届课程设计,,汇总 1. 有理数运算 有理数是一个可以化为一个分数的数,例如2/3,533/920,-12/49都是有理数,而就为无理数。在C++中,并没有预先定义有理数,需要可以定义一个有理数类,将有理数的分子和分母分别存放在两个整型变量中。对有理数的各种操作都可以用重载运算符来实现。 定义并实现一个有理数类,通过重载运算符+、-、*、/对有理数进行算术运算,通过重载运算符==实现判定两个有理数是否相等。写一个优化函数,它的作用是使有理数约去公分母,也即是使保存的有理数分子和分母之间没有公约数(除去1以外)。此外,还要定义一个将有理数转换为实数的函数,再加上构造函数和有理数输出函数。 2. 模拟计算器 设计一个程序来模拟一个简单的手持计算器。程序支持算术运算+、-、*、/、=、以及C(清除)、A(全清除)操作。 定义一个计算器类,该类包括两个组件对象,一个计算引擎和一个用户接口,用户接口对象处理接受的键盘输入信息,并显示答案,计算引擎对象对给出的数据执行相应操作,并存储操作的结果。 3. 设计一个Database类 设计一个Database类。Database类是一个表的集合,而表又由行和列组成。例如,下面的雇员信息表包含三个记录,每个记录有四个字段( Employee、Name、Department和Boss)。 雇员 姓名 部门 部门经理 111-11-1234 Cruz ACC Warder 213-44-5649 Johnston MIS Michaels 321-88-7895 Tom FIN Bearskin 能够完成对数据库的基本操作;包括创建数据库,实现对数据库里面的表的添加,删除;以及能够完成对表结构的修改(如添加或删除字段),以及对表中的记录进行添加和删除;能够完成对使用适当的查询语言从一个或多个表中查找相关信息。 4、 矩阵类设计 定义Matrix类,参照实现: (1) 任意行数,列数矩阵的构建 (2) 常数矩阵 (3) 转置矩阵 (4) 矩阵加法,减法,乘法 (5) 矩阵与数组间的转换 (6) 逆矩阵 (7) 矩阵的输入与输出 (8) 完成相应应用程序设计 5、 表格类设计 定义Table类,参照实现: a) 任意行数,列数表格的构建 b) 表格标题设置 c) 表头的定义 d) 表格数据的输入及修改 e) 表格框线的绘制 f) 表格数据显示 g) 表格数据的统计计算 h) 完成相应应用程序设计 6、 堆栈类设计* 定义Stack类,参照实现: a) 用数组实现先进后出的数据结构 b) 栈顶指针的设置 c) 进栈操作 d) 出栈操作 e) 空栈与栈满溢出判定 f) 完成相应应用程序设计 7、 矢量设计* 定义Vector类,参照实现: a) 矢量的构建 b) 矢量的加法,减法 c) 矢量的点积 d) 矢量的叉积 e) 矢量的输出 f) 完成相应应用程序的设计 8、 链表类设计 定义Link类,参照实现: a) 建立链表的数据结构 b) 输入链表结点的数据 c) 显示链表结点数据 d) 插入链表结点 e) 删除链表结点 f) 完成相应应用程序设计 9、通信录的设计 一、定义人员(person)类,其中至少包括姓名、性别、电话、地址、邮政编码、邮箱、QQ号和类别(例如:同学、朋友等)。 二、实现下面的功能 1、设计菜单实现功能选择; 2、输入功能:输入人员信息,并保存到文件中; 3、查询功能: 1)能够根据姓名、电话精确查询人员信息; 2)能够根据地址进行模糊查询人员信息; 3)根据人员类别查询人员信息 4、根据姓名对人员信息排序输出 5、能根据姓名、电话修改人员信息 6、能根据姓名、电话删除人员信息 10、职工工资管理 一、定义职工(employee )类,其中至少包括姓名、性别、工号、电话、所在科室和工资。 二、实现下面的功能 1、设计菜单实现功能选择; 2、输入功能:输入职工信息,并保存到文件中; 3、查询功能: 1)能够根据工号精确查询职工信息; 2)能够根据姓名、科室查询职工信息 3)分科室进行工资统计,计算各科室的平均工资 4、根据职工的工资排序输出 5、根据工号修改职工信息 6、根据工号删除职工信息 11、三角形的种类与面积 一、定义点(point)类,包含点的坐标x和y;通过继承点类派生出线段(line)类;通

65,199

社区成员

发帖
与我相关
我的任务
社区描述
C++ 语言相关问题讨论,技术干货分享,前沿动态等
c++ 技术论坛(原bbs)
社区管理员
  • C++ 语言社区
  • encoderlee
  • paschen
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
  1. 请不要发布与C++技术无关的贴子
  2. 请不要发布与技术无关的招聘、广告的帖子
  3. 请尽可能的描述清楚你的问题,如果涉及到代码请尽可能的格式化一下

试试用AI创作助手写篇文章吧