【C++ Priemr | 15】虚函数表剖析(三)

一、虚拟菱形继承

#include <iostream>
using namespace std;class B {
public:int _b;
};class C1 :virtual public B {
public:int _c1;
};class C2 :virtual public B {
public:int _c2;
};class D :public C1, public C2 {
public:int _d;
};int main()
{cout << sizeof(D); //24return 0;
}


二. 虚拟菱形继承

测试代码:

#include <iostream>
using namespace std;
struct A
{A(int v = 100) : X(v) {};virtual void foo(void) {}int X;
};struct B : virtual public A
{B(int v = 10) :Y(v), A(100) {};virtual void fooB(void) {}int Y;
};struct C : virtual public A
{C(int v = 20) :Z(v), A(100) {}virtual void fooC(void) {}int Z;
};struct D : public B, public C
{D(int v = 40) :B(10), C(20), A(100), L(v) {}virtual void fooD(void) {}int L;
};int main()
{A a;int* ptr;ptr = (int*)&a;cout << ptr << " sizeof = " << sizeof(a) << endl;for (int i = 0; i < sizeof(A) / sizeof(int); i++){if (ptr[i] < 10000){cout << dec << ptr[i] << endl;}elsecout << hex << ptr[i] << " = " << hex << *((int*)(ptr[i])) << endl;}cout << "--------------------------------------" << endl;B b;ptr = (int*)& b;cout << "addr:" << ptr << " sizeof = " << sizeof(b) << endl;for (int i = 0; i < sizeof(B) / sizeof(int); i++){if (ptr[i] < 10000){cout << dec << ptr[i] << endl;}elsecout << hex << ptr[i] << " = " << hex << *((int*)(ptr[i])) << endl;}cout << "--------------------------------------" << endl;D d;ptr = (int*)& d;cout << "addr:" << ptr << " sizeof = " << sizeof(d) << endl;for (int i = 0; i < sizeof(D) / sizeof(int); i++){if (ptr[i] < 10000){cout << dec << ptr[i] << endl;}elsecout << hex << ptr[i] << " = " << hex << *((int*)(ptr[i])) << endl;}return 0;
}

输出结果: 

参考资料

  •  图说C++对象模型:对象内存布局详解
  • 从内存布局看C++虚继承的实现原理
  • C++继承内存对象模型

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/385448.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

gcc的警告提示信息

gcc包含完整的出错检查和警告提示功能。采用-pedantic选项&#xff0c;对于不符合ANSI/ISO标准的源代码会产生相应的警告信息。如&#xff1a;gcc -pedantic hello.c -o hello (main函数返回类型为int&#xff0c;且函数体内要有return 语句&#xff0c;一般为 return 0;) -pe…

1037. 在霍格沃茨找零钱(20)

如果你是哈利波特迷&#xff0c;你会知道魔法世界有它自己的货币系统 —— 就如海格告诉哈利的&#xff1a;“十七个银西可(Sickle)兑一个加隆(Galleon)&#xff0c;二十九个纳特(Knut)兑一个西可&#xff0c;很容易。”现在&#xff0c;给定哈利应付的价钱P和他实付的钱A&…

【Leetcode | 6】136. 只出现一次的数字

给定一个非空整数数组&#xff0c;除了某个元素只出现一次以外&#xff0c;其余每个元素均出现两次。找出那个只出现了一次的元素。 说明&#xff1a; 你的算法应该具有线性时间复杂度。 你可以不使用额外空间来实现吗&#xff1f; 示例 1: 输入: [2,2,1] 输出: 1 示例 2: 输入…

gcc的优化功能

代码优化的目的是改善程序的执行性能。gcc提供的代码优化功能非常强大&#xff0c;它通过参数-On来控制优化代码的生成&#xff0c;其中n为优化级别的整数&#xff0c;比较典型的范围是从0变化到2或3&#xff08;与版本有关&#xff09;。 编译时通过使用选项-O可以告诉gcc同时…

gcc编译多个源代码文件的过程(引出makefile)

由foo1.c foo2.c foo3.c 3个源文件组成的源程序生成最终的可执行程序foo的命令&#xff1a; gcc foo1.c foo2.c foo3.c -o foo 如果处理的源文件不止一个&#xff0c;则gcc会依次对每个文件进行预处理、编译、汇编&#xff0c;最后将所有的目标代码和库文件进行&#xff0c;链…

观擦者模式

/********************************************************************created: 2006/07/20filename: Observer.hauthor: 李创http://www.cppblog.com/converse/purpose: Observer模式的演示代码 *********************************************************************/…

程序的装入和链接

注&#xff1a;这是本人学习汤小丹等编写的计算机操作系统&#xff08;西安电子科技大学出版社&#xff09;的学习笔记&#xff0c;因此许多引用来源于此书&#xff0c;在正文中就不注明了&#xff01; 程序在运行前需要经过以下步骤&#xff1a;编译程序对源程序进行编译生成…

内存对齐

1. 对齐原则&#xff1a; 数据成员对齐规则&#xff1a;结构(struct)(或联合(union))的数据成员&#xff0c;第一个数据成员放在offset为0的地方&#xff0c;以后每个数据成员的对齐按照#pragma pack指定的数值和这个数据成员自身长度中&#xff0c;比较小的那个进行。结构(或…

1006. 换个格式输出整数 (15)

让我们用字母B来表示“百”、字母S表示“十”&#xff0c;用“12...n”来表示个位数字n&#xff08;<10&#xff09;&#xff0c;换个格式来输出任一个不超过3位的正整数。例如234应该被输出为BBSSS1234&#xff0c;因为它有2个“百”、3个“十”、以及个位的4。 输入格式&a…

静态库的制作和使用

Linux下的静态库为lib*.a格式的二进制文件&#xff08;目标文件&#xff09;&#xff0c;对应于Windows下的.lib格式的文件。 &#xff08;1&#xff09;命名规则 lib库名字 .a libMytest.a &#xff0c;则库名字为mytest。下面以具体的代码为例介绍如何制作静态库。 //mai…

IO多路复用之select

int select(int nfds, fd_set *readfds, fd_set *writefds,fd_set *exceptfds, struct timeval *timeout); 分析&#xff1a; nfds: 监控的文件描述符集里最大文件描述符加1&#xff0c;因为此参数会告诉内核检测前多少个文件描述符的状态 readfds&#xff1a; …

1031. 查验身份证(15)

一个合法的身份证号码由17位地区、日期编号和顺序编号加1位校验码组成。校验码的计算规则如下&#xff1a; 首先对前17位数字加权求和&#xff0c;权重分配为&#xff1a;{7&#xff0c;9&#xff0c;10&#xff0c;5&#xff0c;8&#xff0c;4&#xff0c;2&#xff0c;1&…

虚拟地址空间

对于每一个进程都会对应一个虚拟地址空间&#xff0c;对于32位的操作系统&#xff08;其指令的位数最大为32位&#xff0c;因此地址码最多32位&#xff09;&#xff0c;虚拟地址空间的大小为B即0~4GB的虚拟地址空间&#xff0c;其中内核空间为1GB&#xff0c;如下所示&#xff…

Leecode 69. x 的平方根

实现 int sqrt(int x) 函数。 计算并返回 x 的平方根&#xff0c;其中 x 是非负整数。 由于返回类型是整数&#xff0c;结果只保留整数的部分&#xff0c;小数部分将被舍去。 示例 1: 输入: 4 输出: 2 示例 2: 输入: 8 输出: 2 说明: 8 的平方根是 2.82842..., 由于返回类…

1002. 写出这个数 (20)

读入一个自然数n&#xff0c;计算其各位数字之和&#xff0c;用汉语拼音写出和的每一位数字。 输入格式&#xff1a;每个测试输入包含1个测试用例&#xff0c;即给出自然数n的值。这里保证n小于10100。 输出格式&#xff1a;在一行内输出n的各位数字之和的每一位&#xff0c;拼…

C/C++中NULL指针

先谈一下C/C的强制类型转换Type cast。与强制类型转换相对应的是自动类型转换。或者强制类型转换叫显示类型转换&#xff0c;自动类型转换叫隐式类型转换。自动类型转换会在赋值运算、混合运算、参数传递、返回函数返回值、格式化输出时且当类型出现不一致时发生&#xff0c;转…

1009. 说反话 (20)

给定一句英语&#xff0c;要求你编写程序&#xff0c;将句中所有单词的顺序颠倒输出。 输入格式&#xff1a;测试输入包含一个测试用例&#xff0c;在一行内给出总长度不超过80的字符串。字符串由若干单词和若干空格组成&#xff0c;其中单词是由英文字母&#xff08;大小写有区…

动态库(共享库)的制作和使用

Linux下的动态库为lib*.so格式的二进制文件&#xff08;目标文件&#xff09;&#xff0c;对应于Windows下的.dll格式的文件。 &#xff08;1&#xff09;命名规则 lib库名.so &#xff08;2&#xff09;动态库的制作 1&#xff09;生成与位置无关的代码&#xff08;.o&…

孤儿进程、僵尸进进程

一、儿进程与僵尸进程 1、基本概念 我们知道在unix/linux中&#xff0c;正常情况下&#xff0c;子进程是通过父进程创建的&#xff0c;子进程在创建新的进程。子进程的结束和父进程的运行是一个异步过程,即父进程永远无法预测子进程 到底什么时候结束。 当一个 进程完成它的工…

管道 -pipe

gcc编译器将源代码编译成可执行程序的过程中&#xff0c;需要经过许多中间步骤&#xff08;预处理、编译、汇编、链接&#xff09;&#xff0c;这些过程实际上是由不同的程序来负责完成的&#xff08;/usr/bin/gcc、cpp、ccl、as和ld等&#xff09;。在这个过程的每一个阶段中&…