c++对象模型03——C++对象内存大小模型

引入

  • 下面的Point3d是一个类,我们有一个问题:影响x的存取效率的因素有哪些?
Point3d origin;
origin.x = 0.0;
  • 下面我们有分别定义了类对象的变量形式与指针形式,那么通过origin和pt对数据成员存取有什么差异吗? 
Point3d origin,*pt = &origin;
origin.x = 0.0;
pt->x = 0.0;

1. 静态数据成员(Static Data Members)

1.1 特点

  • 静态数据成员并不属于某一特定类对象,而是属于整个类;
  • 静态数据成员可以通过“::”操作符或“.”操作符用类名直接访问,或者用“.”操作用类对象进行访问;
  • static成员变量可以在类内声明并初始化。但是建议在类内定义、类外初始化
  • 如果有继承关系,那么静态数据成员会存在于整个继承体系中;

1.2 静态数据成员在内存中的定义

  • C++会把类的静态数据成员当做一种全局(global)变量,只在class生命范围内可见;
  • 静态数据成员是存储在程序的数据段(data segment)之中;
  • 静态成员编码:因为静态数据成员是存储在程序的数据段的,所以如果有两个不同类的静态数据成员如果相同可能会产生冲突,为了防止名称冲突,C++编译器的解决办法是对每一个静态数据数据成员进行编码(这个手法叫做name-mangling),以获得一个独一无二的程序识别代码。不同的编译器,其name-mangling做法不同;

1.3 静态数据成员的存取

  • 因为静态数据成员并不存在于类对象之中因此存取静态成员并不需要通过类对象,相比类非静态成员,效率高一些;
  • 继承体系中:如果一个派生类存取其基类的静态成员,消耗的精力与上面的也类似,因为程序中对于静态成员还是只有一个唯一的实体,对齐存取还是那么直接;
  • 如果静态数据成员的存取是经由函数调用(或其他某些语法)而被存取,则C++标准明确要求函数必须被求值;
  • 指针操作:如果要取一个静态数据成员的地址,会得到一个指向其数据类型的指针,而不是指向于类对象的指针(本质还是因为静态数据成员并包含在一个class object中);

1.4 示例

示例1:

#include <iostream>
using namespace std;

class MyClass
{
public:
	static int a;
};

int MyClass::a = 10;

int main()
{
	MyClass test;

	cout << "The value of a is " << test.a << endl;
	//cout << "The value of a is" << MyClass.a <,endl;  //error
	cout << "The value of a is " << MyClass::a <<endl;

	system("pause");
	return 0;
}

打印结果:

示例2:

#include <iostream>
using namespace std;

class MyClass
{
public:
	static int a;
};

int MyClass::a = 10;

// 拷贝返回一个MyClass类变量
MyClass foobar()
{
	MyClass a;
	return a;
}

int main()
{
	cout << "The value of a is " << MyClass::a << endl;

	// 返回的类变量的静态成员必须被初始化
	foobar().a = 20;
	cout << "The value of a is " << MyClass::a << endl;

	system("pause");
	return 0;
}

 打印结果:

 示例3:

#include <iostream>
using namespace std;

class MyClass
{
public:
	static int a;
};
int MyClass::a = 10;

int main()
{
	int* pa = &MyClass::a;

	cout << "The value of a is " << *pa << endl;

	system("pause");
	return 0;
}

打印结果:

2. 非静态数据成员(Nonstatic Data Members)

2.1 非静态数据成员的存取

  • 非静态数据成员存在于每一个类对象之中,因此我们对非静态数据成员的存取必须经过类对象
  • 地址偏移:对一个非静态成员进行存取操作,编译器需要把类对象起始地址加上数据成员的地址偏移量才可以对数据成员进行操作
  • 虚继承之下:如果非静态数据成员是一个类成员,在单一继承、多重继承的情况下存取都是相同的。但如果非静态数据成员是一个虚基类的成员,存取速度可能会慢一些;
  • 通过“.”、“->”访问非静态数据成员的差异:如果一个非静态数据成员是从一个虚基类中继承而来的数据成员时,会有不同的差异;

2.2 示例

示例1

class Point3d{
public:
    translate(const Point3d &pt){
        x += pt.x;  //this->x += pt.x
        y += pt.y;  //this->y += pt.x
        z += pt.z;  //this->z += pt.z
    }
private:
    int x;
    int y;
    int z;
};

示例2

class Point3d{
public:
    float _y;
}
 
int main()
{
    Point3d origin;
    origin._y = 0.0;
}

示例3

如果成员x是从虚基类继承而来的:

  • 那么pt在编译使其不知道这个x属于哪一种类类型(class type),因为其是一个指针,会产生“动态绑定”的效果,并不知道该指针实际指向哪个类,于是也就不知道其真正的偏移位置,所以这个存取操作也就延迟至执行器,经由一个额外的间接引导;
  • 但是如果使用origin,那么就不会有上面的间接引导操作,因为其类型在编译器就可以判断为Point3d类类型,则其成员的偏移位置也就在编译器固定下来了;
Point3d origin, *pt = &pt;
 
origin.x = 0.0;
pt->x = 0.0.;
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值