C++类成员布局

在C++中对象的内存布局与类成员声明的顺序一致,静态成员放在数据区(Data Section)而非对象内存中,若多个类静态成员名称相同,C++则按照name mangling技术进行重命名保证名称的唯一性。若类之间发生了继承关系(无虚拟指针无虚继承)时,按照基类、子类成员顺序排列,另在C++对象复制中,有一个规则:基类子对象(base class object)在派生类(derived class member)中成员的原样性

测试代码如下:

#include <iostream>
#include <iomanip>
#include <cassert>

using namespace std;

class Concrete1 {
public:
	void foo(void) {
		cout << "concrete1:" << bit1 << endl;
	}
	Concrete1():val(1),bit1(‘a‘) {
	}
public:
	int val;
	char bit1;
};

class Concrete2:public Concrete1 {
public:
	char bit2;
public:
	void foo(void) {
		cout << "concreate2" << bit2 << endl;
	}

	Concrete2():bit2(‘b‘),Concrete1() {
	}
};

class Concrete3: public Concrete2 {
public:
	void foo(void) {
		cout << "concrete3:" << bit3 << endl;
	}

	Concrete3():bit3(‘c‘),Concrete2() {
	}

public:
	char bit3;
};

template <class data_type1,class data_type2>
const char *access_order(data_type1 *mem1,data_type2 *mem2)
{
	unsigned long p1,p2;

	p1 = (unsigned long)(mem1);
	p2 = (unsigned long)(mem2);

	assert(p1 != p2);

	return p1 < p2
				? "member 1 occurs first"
				:"member 2 occurs first";
}

int main(int argc, char *argv[])
{
	Concrete2 c2,*pc2;
	Concrete1 c1,*pc1_1,*pc1_2;
	Concrete3 c3;

	pc2 = &c3;
	pc1_1 = pc2;
	pc1_2 = &c1;
	char *b;

	//access_order(&Concrete1::bit1,&Concrete1::bit);
	cout << "address val  and bit1:" << access_order(&c1.val,&c1.bit1) << endl;
	cout << "Concrete3:" << sizeof(Concrete3) << endl;

	//b = *(char *)((char *)(&(pc1_2->bit1)) + 1);
	b = &(pc1_2->bit1);

	cout << "Before address:" << hex << static_cast<void *>(b) << ",b:" << *b << endl;
	b++;
	cout << "After address:" << hex << static_cast<void *>(b) << endl;
	cout << "b:" << *b << "b+1:" << *(b+1)<< endl;
	cout << "Memberwise copy." << endl;
	*pc1_2 = *pc1_1;
	//b = *(char *)((char *)(&(pc1_2->bit1)) + 1);
	b = &(pc1_2->bit1);

	cout << "Before address:" << hex << static_cast<void *>(b) << ",b:" << *b << endl;
	b++;
	cout << "After address:" << hex << static_cast<void *>(b) << endl;
	cout << "b:" << *b << "b+1:" << *(b+1)<< endl;
	pc1_2->foo();

	return 0;
}

g++ 运行情况如下:

f:\code\C++\study>concrete
concrete
address val  and bit1:member 1 occurs first
Concrete1:8//三个类大小均为8字节
Concrete2:8
Concrete3:8
Before address:0x22ff34,b:a
After address:0x22ff35
b: b+1:"
Memberwise copy. //复制前后Concrete1对象填充字节内容未变化
Before address:0x22ff34,b:a
After address:0x22ff35
b: b+1:"
concrete1:a

vs2010运行如下:

testlayout
address val  and bit1:member 1 occurs first
Concrete1:8//三个类大小均不同,采用了对齐方式处理
Concrete2:12
Concrete3:16
Before address:0012FF40,b:a
After address:0012FF41
b:蘠+1:
Memberwise copy. //复制前后Concrete1对象填充字节内容未变化
Before address:0012FF40,b:a
After address:0012FF41
b:蘠+1:
concrete1:a
时间: 2024-11-25 06:52:12

C++类成员布局的相关文章

C++类内存布局图(成员函数和成员变量分开讨论)

一.成员函数 成员函数可以被看作是类作用域的全局函数,不在对象分配的空间里,只有虚函数才会在类对象里有一个指针,存放虚函数的地址等相关信息. 成员函数的地址,编译期就已确定,并静态绑定或动态的绑定在对应的对象上.对象调用成员函数时,早在编译期间,编译器就可以确定这些函数的地址,并通过传入this指针和其他参数,完成函数的调用,所以类中就没有必要存储成员函数的信息. 二.成员变量 转自:http://www.cnblogs.com/jerry19880126/p/3616999.html 书上类继

理解ATL中的一些汇编代码(通过Thunk技术来调用类成员函数)

我们知道ATL(活动模板库)是一套很小巧高效的COM开发库,它本身的核心文件其实没几个,COM相关的(主要是atlbase.h, atlcom.h),另外还有一个窗口相关的(atlwin.h), 所以拿来学习应该是很方便的.但是因为ATL的代码充满了模板和宏,内部还夹杂着汇编,所以如果没有比较丰富的C++模板和系统底层的知识,一般人会看得一头雾水. 下面我们主要分析一下ATL中的一些汇编代码. ATL中出现汇编代码主要是2处,一处是通过Thunk技术来调用类成员函数处理消息:还有一处是通过打开_

C#定义类成员

1.成员定义 public--成员可以由任何代码访问. private--成员只能由类中的代码访问(如果没有使用任何关键字,就默认使用这个关键字). internal--成员只能由定义它的程序集(项目)内部的代码访问. protected--成员只能由类或派生类中的代码访问. 后两个关键字可以合并使用,所以也有protected internal成员.他们只能有项目(更确切地讲,是程序集)中派生类的代码来访问. 也可以使用关键字 static 来声明字段.方法和属性,这表示它们是类的静态成员,而

Objective-C类成员变量深度剖析

目录 Non Fragile ivars 为什么Non Fragile ivars很关键 如何寻址类成员变量 真正的“如何寻址类成员变量” Non Fragile ivars布局调整 为什么Objective-C类不能动态添加成员变量 总结 看下面的代码,考虑Objective-C里最常见的操作之一——类成员变量访问. - (void)doSomething:(SomeClass *)obj { obj->ivar1 = 42; // 访问obj对象的public成员变量 int n = sel

C++语言学习(十四)——C++类成员函数调用分析

C++语言学习(十四)--C++类成员函数调用分析 一.C++成员函数 1.C++成员函数的编译 C++中的函数在编译时会根据命名空间.类.参数签名等信息进行重新命名,形成新的函数名.函数重命名的过程通过一个特殊的Name Mangling(名字编码)算法来实现.Name Mangling算法是一种可逆的算法,既可以通过现有函数名计算出新函数名,也可以通过新函数名逆向推导出原有函数名.Name Mangling算法可以确保新函数名的唯一性,只要命名空间.所属的类.参数签名等有一个不同,那么产生的

类成员函数的指针与多态性

1 类成员函数的指针 2 类成员函数的指针数组 3 指向类成员函数的指针的指针 1 类成员函数的指针 auto func1 = &com::jia;//C++函数指针 int (com::*p)(int, int) = &com::jia;//C函数指针 1 #include <iostream> 2 3 class com 4 { 5 private: 6 int a; 7 int b; 8 public: 9 com(int x, int y) :a(x), b(y) 10

C++ 获取类成员函数地址方法 浅析

C语言中可以用函数地址直接调用函数: void print () { printf ("function print"); } typdef void (*fun)(); fun f = print; f(); C++中类非静态成员函数必须通过实例去调用,C++中类成员函数调用: class test { public: void print () { printf ("function print"); } }; 我们同样可以通过定义函数指针来调用如下: type

C++的继承操作---基类指针访问派生类问题---基类成员恢复访问属性问题

#include "stdafx.h" #include <iostream> #include <algorithm> using namespace std; class Base { public: int num; virtual void func() { cout<<"Do something in Base"<<endl; } }; class Derived:private Base { public:

类成员变量初始化的问题

class window{ window(int maker){ System.out.println("window"+maker); } } class House{ window w1 ;//new window(1); window w3 = new window(3); House(){ System.out.print("House"); w3 = new window(33); } window w4 = new window(4); } class