第3章 Data语意学
在C++中经常会遇到一个类的大小问题,关于一个类的大小一般受到三个方面的影响。
- 语言本身所造成的额外负担,如在虚拟继承中会遇到如派生类中会包含一个指针指向base class subobjec,这样会造成内存的额外负担。
- 编译器对于特殊情况所提供的的优化。如果编译器不提供优化,则会在一个空的基类中会形成至少一个char类型的内存空间,传统上这一个char类型的空间会被放在该类内存的固定尾部,但是有些编译器会提供优化的策略,在空类型中不会生成这一个char成员对象的空间。
- Alignment的限制。如果一个中含有虚函数,其中如果包含一个char类型的函数成员,则会发生内存空间对其的情况,从而出现sizeof class_name = 8的情况,相当于struct中的内存对齐。
3.1 Data Member的绑定
在类中如果有一个函数成员的名称和一个全局变量的名称相同,在类内的某些函数对其进行参阅(取用)操作,相信很多人会说该操作是对类内成员的操作,对于今天的编译器来说答案是正确,但是在很早的时候,这个操作是针对的global x object的操作。当然对于这种在当时的情况下出现了两种防御措施。
- 把所有的数据成员放在类的声明起头处,以确保正确的绑定:
class X{
float x, y, z;
public:
float get_X(){return x}
}
2. 把所有的内联函数不管大小都放在类声明之外。
class X{
float x, y, z;
public:
float get_X(){}
}
inline float get_X(){
return x;
}
上面的情况在C++2.0之后就已经消失了,但是关于成员函数的形参中关于类型的操作应该加以注意如:
typedef int length;
class Point3d{
void numble(length val){_val = val}//此处的length为int
length numble(){return _val} //此处的length为int
private:
typedef length float;
length _val;
};
在这种情况下以上两种防御型风格仍然需要。
3.2 Data Member的布局
非静态成员的顺序和声明的顺序一样(同一访问权限中(public, protected, private)较晚出现的成员在类对象的较高的地址处),静态成员不会放在一个对象的布局之中。
3.3 Data Member的存取
1. 静态成员
class_name::static_data_member;
class_obj.static_data_member;
2. 非静态成员
Point3d Point3d::translate(const Point3d &pt){
x += pt.x;
y += pt.y;
z += pt.z;
}
将被转化为
Point3d Point3d::translate(const Point3d &pt){
this->x += pt.x;
this->y += pt.y;
this->z += pt.z;
}
关于取地址操作
origin._y = 0;
&origin._y的操作为&origin + (&origin._y - 1)
在类内声明的变量只会按照声明顺序来进行安排其内存地址如:
#include<stdio.h>
//#include<string.h>
#include<string>
#include<windows.h>
class A;
void p_printf(); void print(char s[], int *b){
char szbuffer[] = {};
wsprintf(szbuffer, s, b);
printf(szbuffer);
} class A{ void friend print(char, A *);
void friend p_print();
public:
int public_A;
protected:
int protected_A;
private:
int private_A; public:
int public_B;
protected:
int protected_B;
private:
int private_B; }; void p_print(){
A a;
printf("the a address is %#x\n\0", &a);
print("the a.public_A address is %#X\n\0", &a.public_A);
print("the a.protected_A address is %#X\n\0", &a.protected_A);
print("the a.private_A address is %#X\n\0", &a.private_A);
print("the a.public_B address is %#X\n\0", &a.public_B);
print("the a.protected_B address is %#X\n\0", &a.protected_B);
print("the a.private_B address is %#X\n\0", &a.private_B); } int main(void){ p_print(); } 输出为:
the a address is 0x28fef8
the a.public_A address is 0X28FEF8
the a.protected_A address is 0X28FEFC
the a.private_A address is 0X28FF00
the a.public_B address is 0X28FF04
the a.protected_B address is 0X28FF08
the a.private_B address is 0X28FF0C Process returned (0x0) execution time : 0.021 s
Press any key to continue.
3.4 继承与数据成员
1. 只继承无多态

这是因为在进行向上转换的时候采用memberwise拷贝,如果不各自对其,当由基类向派生类拷贝时容易出现

2. 加上多态
多态作为C++的类的三大特性(封装,继承,多态),在内存中的布局为继承体系中每个类内存对其后的按照继承顺序叠加在一起,最后是该类自身的数据成员,由于含有虚函数,所以会产生一个虚函数表,至于虚函数指针(一个指向虚函数表的指针)(C++标准对虚函数指针的存放位置并没有严格规定,但是编译器一般情况是放在一个类的头部或者是尾部)。

3. 多重继承
如上所说,按照继承顺序进行叠加数据成员,每个类的自身虚函数指针仍然在自己类的相应部分。
4. 虚拟继承
在类的继承体系中,有的时候会出现某几个父类自于继承自同一个base class,如下图所示,为避免空间的浪费,采用虚拟继承

使得继承模型变为

但是在这种模型设计过程中发现会出现随着继承类的增加会增加指向虚拟类的指针内存负担,从而出现将这些指针转移至虚函数指针在-1处添加共享类的偏移地址,0处代表type_info for class, 正值为该类的virtual func的函数地址,如:

第3章 Data语意学的更多相关文章
- 第4章 Function语意学
第4章 Function语意学 目录 第4章 Function语意学 4.1 Member的各种调用方式 Nonstatic Member Function(非静态成员函数) virtual Memb ...
- 第 7 章 Data 类型
目录 第 7 章 Data 类型 一.创建方式 二.转时间戳 其他 第 7 章 Data 类型 @(es5) 参考了: 阮一峰javascript的标准.<javascript高级教程> ...
- 【C++对象模型】第六章 执行期语意学
执行期语意学,即在程序执行时,编译器产生额外的指令调用,确保对象的构造,内存的释放,以及类型转换与临时对象的生成的安全进行. 1.对象的构造和析构 对于类对象的构造,一般在定义之后则开始内部的构造过程 ...
- 《深度探索C++对象模型》第二章 | 构造函数语意学
默认构造函数的构建操作 默认构造函数在需要的时候被编译器合成出来.这里"在需要的时候"指的是编译器需要的时候. 带有默认构造函数的成员对象 如果一个类没有任何构造函数,但是它包含一 ...
- 《深度探索c++对象模型》chapter3 Data语意学
一个空的class:如 class X{} ; sizeof(X)==1; sizeof为什么为1,他有一个隐晦的1 byte,那是被编译器安插进去的一个char,这使得class2的两个object ...
- data语意学
引例: class X{}; class Y:public virtual X{}; class Z:public virtual X{}; class A:public Y,public Z{}; ...
- c++ data语意学
Data Member的绑定 extern float x; class Point3d { public: point3d(); //问题:被传回和被设定的x是哪一个x呢? float X() c ...
- 《深度探索C++对象模型》笔记——Data语意学
Data Member的绑定 inline member functin躯体之内的一个data member绑定操作会在整个class声明完成之后才发生. argument list中的名称还是会在它 ...
- 【C++】深度探索C++对象模型读书笔记--Data语意学(The Semantics of data)
1. 一个空类的大小是1 byte.这是为了让这一类的两个对象得以在内存中配置独一无二的地址. 2. Nonstatic data member 放置的是“个别的class object”感兴趣的数据 ...
随机推荐
- winform程序中chart图的使用经验(chart图的更新)
如何让chart图进行刷新并且根据数值重新绘制 首先初始化一个chart chart1.Titles.Add("柱状图数据分析"); chart1.ChartAreas[].Axi ...
- Beta冲刺 7
前言 队名:拖鞋旅游队 组长博客:https://www.cnblogs.com/Sulumer/p/10129067.html 作业博客:https://edu.cnblogs.com/campus ...
- c++ cin cin.getline() getline()用法
http://www.cnblogs.com/AndyJee/p/3821067.html 主要内容: 1.cin用法 2.cin.getline()用法 3.getline()用法 3.注意的问题 ...
- Linux常用命令之Yum
Linux Yum命令详解Yum全称Yellow dog Updater,Modified,是一个在Fedora和RedHat以及SUSE中提供的基于RPM包的软件包管理工具,能够从指定的服务器自动下 ...
- sdn交换机和普通交换机区别
SDN交换机基本具有普通交换机的所有功能.SDN交换机特别的功能在于支持OpenFlow协议(有些只支持OpenFlow1.0,有些强点支持1.0和1.3).不过你要连接交换机再手动将所需的端口改成支 ...
- [JsonSchema] 关于接口测试 Json 格式比对核心算法实现 (Java 版)
引言 为什么要自己重新造轮子,而不是采用第三方的JsonSchema方法进行实现存在以下痛点:1.我之前在网上找了很久,没有找到java版直接进行jsonschema生成的方法或直接比较的方法2.ht ...
- Sonar Java 规则插件开发 (基于阿里开发手册)
引言 最近在做Sonar静态代码扫描管理,以此顺手接了Sonar的插件开发,基于阿里开发手册进行开发,在整体开发过程中,其中还是遇到不少坑位,也以此给大家做相应借鉴官网Demo演示插件开发地址:htt ...
- 2018-4-25 初识html
第一天 简单了解web 路线 html css js 发布网站 web运作 html html是超文本语言,也就是标记语言.说白了也就规定了一些符号,然后赋予这些符号意义.比如规定a就是超链接标签,用 ...
- 转自CSDN-详述 Java 中的别名现象
在任何编程语言中,赋值操作都是最常见的操作之一,Java 自然也不例外.赋值时,使用赋值操作符=,它的意思是:“将等号右边的值(右值),复制给左边的值(左值)”.右值可以是任何常数.变量或者表达式(只 ...
- 路径定义前+r
定义文件路径时前面加个r 例如 firstfolder = r"C:\Users\1261\Desktop\" 不对其中的符号进行转义