转自:https://blog.csdn.net/allen_tony/article/details/76973906

https://blog.csdn.net/zzwdkxx/article/details/53635173

关于多个父类,虚继承,类本身自己的虚函数,情况比较复杂,后续时间再研究。

类所占内存的大小是由成员变量(静态变量除外)决定的,成员函数是不计算在内的。摘抄部分:
    成员函数还是以一般的函数一样的存在。a.fun()是通过fun(a.this)来调用的。所谓成员函数只是在名义上是类里的。其实成员函数的大小不在类的对象里面,同一个类的多个对象共享函数代码。而我们访问成员函数是通过类里面的一个指针实现,而这个指针指向的是一个table,table里面记录的各个成员函数的地址(当然不同的编译可能略有不同的实现)。所以我们访问成员函数是间接获得地址的。所以这样也就增加了一定的时间开销,这也就是为什么我们提倡把一些简短的,调用频率高的函数声明为inline形式(内联函数)。

如果类定义了虚函数,该类及其派生类就要生成一张虚拟函数表,即vtable。而在类的对象地址空间中存储一个该虚表的入口,占4个字节,这个入口地址是在构造对象时由编译器写入的。

所以,由于对象的内存空间包含了虚表入口,编译器能够由这个入口找到恰当的虚函数,这个函数的地址不再由数据类型决定了。

故对于一个父类的对象指针,调用虚拟函数,如果给他赋父类对象的指针,那么他就调用父类中的函数,如果给他赋子类对象的指针,他就调用子类中的函数(取决于对象的内存地址)。

(一)
class CBase
{
};
sizeof(CBase)=1;

为什么空的什么都没有是1呢?

C++要求每个实例在内存中都有独一无二的地址。    //important

空类也会被实例化,所以编译器会给空类隐含的添加一个字节,这样空类实例化之后就有了独一无二的地址了。所以空类的sizeof为1.

(二)
class CBase
{
    int a;
    char p;
};
sizeof(CBase)=8;

记得对齐问题,64位是8字节对齐,32位是4字节对齐。int占4字节    //注意这点和struct的对齐原则很像!!!
char占一个字节,补齐3字节

(三)
class CBase
{
public:
    CBase(void);
    virtual ~CBase(void);
private:
    int a;
    char* p;
};

再运行:sizeof(CBase)=12

C++类中有虚函数的时候有一个指向虚函数的指针(vptr),在32位系统分配指针大小为4字节。无论多少个虚函数,只有这一个指针,4字节。//注意一般的函数是没有这个指针的,而且也不占类的内存。

(四)
class CChild:public CBase
{
public:
    CChild(void);
    ~CChild(void);
    virtual void test();
private:
    int b;
};

输出:sizeof(CChild)=16;

可见子类的大小是本身成员变量的大小加上父类的大小。//其中有一部分是虚函数表的原因,一定要知道
        父类子类共享一个虚函数指针。

(五)
#include <iostream>
class a{};
class b{};
class c:public a{
virtual void fun() = 0;
};
class d:[public b,public c{};
int main()
{
cout << "sizeof(a)=" << sizeof(a)<<endl;
cout << "sizeof(b)=" << sizeof(b)<<endl;
cout << "sizeof(c)=" << sizeof(c)<<endl;
cout << "sizeof(d)=" << sizeof(d)<<endl;
retrun 0;
}

程序执行的输出结果为:
sizeof(a)=1
sizeof(b)=1
sizeof(c)=4
sizeof(d)=8

前三种情况比较常见,注意第四种情况。类d的大小更让初学者疑惑,类d是由类b,c派生而来的,它所大小应该是二者之和5,为什么却是8呢?这是因为为了提高实例在内存中的存取效率。类的大小往往被调整到系统的整数倍,并采取就近的法则,离哪最近的倍数就是该类的大小,所以类d的大小为8字节。

(六)

class MyClass
{
public:
MyClass();
~MyClass(); void test(); private:
char p;
}; MyClass::MyClass()
{
} MyClass::~MyClass()
{
} void MyClass::test()
{ } class MyClassA
{
public:
MyClassA();
~MyClassA(); public:
static int m_a; }; MyClassA::MyClassA()
{
} MyClassA::~MyClassA()
{
} int MyClassA::m_a = ; class MyClassB
{
public:
MyClassB();
~MyClassB(); virtual void test1();
virtual void test2(); private: }; MyClassB::MyClassB()
{
} void MyClassB::test1()
{
} void MyClassB::test2()
{
} MyClassB::~MyClassB()
{
} int main()
{ printf("MyClass's size is %d\n", sizeof(MyClass)); printf("MyClassA's size is %d\n", sizeof(MyClassA)); printf("MyClassB's size is %d\n", sizeof(MyClassB)); return ;
}

输出

MyClass's size is 1
MyClassA's size is 1
MyClassB's size is 4

(七)继承多个父类,父类都存在虚函数:

// y1.cpp : 定义控制台应用程序的入口点。
// #include "stdafx.h" class MyClassA
{
public:
MyClassA();
~MyClassA(); virtual void test1(); }; MyClassA::MyClassA()
{
} MyClassA::~MyClassA()
{
} void MyClassA::test1()
{ } class MyClassB
{
public:
MyClassB();
~MyClassB(); virtual void test1(); private: }; MyClassB::MyClassB()
{
} void MyClassB::test1()
{
} MyClassB::~MyClassB()
{
} class MyClassC:public MyClassA
{
public:
MyClassC();
~MyClassC(); private: }; MyClassC::MyClassC()
{
} MyClassC::~MyClassC()
{
} class MyClassD: public MyClassA, MyClassB
{
public:
MyClassD();
~MyClassD(); private: }; MyClassD::MyClassD()
{ } MyClassD::~MyClassD()
{ } int main()
{ printf("MyClass's size is %d\n", sizeof(MyClassA)); printf("MyClassA's size is %d\n", sizeof(MyClassB)); printf("MyClassB's size is %d\n", sizeof(MyClassC)); printf("MyClassB's size is %d\n", sizeof(MyClassD)); return ;
}

输出:

MyClassA's size is 4
MyClassB's size is 4
MyClassC's size is 4
MyClassD's size is 8

MyClassA,MyClassB都有虚函数。MyClassD继承了MyClassA和MyClassB,所以

MyClassD占用的空间为8

总结:
空的类是会占用内存空间的,而且大小是1,原因是C++要求每个实例在内存中都有独一无二的地址。
(一)类内部的成员变量:
普通的变量:是要占用内存的,但是要注意对齐原则(这点和struct类型很相似)。
static修饰的静态变量:不占用内容,原因是编译器将其放在全局变量区。
(二)类内部的成员函数:
普通函数:不占用内存。虚函数:要占用4个字节,用来指定虚函数的虚拟函数表的入口地址。所以一个类的虚函数所占用的地址是不变的,和虚函数的个数是没有关系的。

c++中各类型数据所占字节数(二)的更多相关文章

  1. c++中各类型数据所占字节数(一)

    转自: https://blog.csdn.net/hi_baymax/article/details/82415896 和机器字长及编译器有关系: 所以,int,long int,short int ...

  2. 在C语言中,double、long、unsigned、int、char类型数据所占字节数

    和机器字长及编译器有关系: 所以,int,long int,short int的宽度都可能随编译器而异.但有几条铁定的原则(ANSI/ISO制订的): 1 sizeof(short int)<= ...

  3. 整型变量修饰符,char类型数据存储原理,字节数,

    //------------------整型变量修饰符 修饰符(int short long longlong signed unsigned)所有修饰符都是用来修整形 int 4short %hd ...

  4. c++中各类型数据占据的字节长度

    c++中各种类型数据类型占据字节长度 首先罗列一下C++中的数据类型都有哪些: 1.整形:int.long 2.字符型:char.wchar_t 3.布尔型:bool 4.浮点型:float.doub ...

  5. java中不同类型的数值占用字节数

    在Java中一共有8种基本数据类型,其中有4种整型,2种浮点类型,1种用于表示Unicode编码的字符单元的字符类型和1种用于表示真值的boolean类型.(一个字节等于8个bit) 1.整型 类型 ...

  6. java中8种数据类型和默认值所占字节数

    java 8种基本数据类型的默认值及所占字节数 通过一段代码来测试一下 8种基本数据类型的默认值 1 package dierge; 2 3 public class Ceshi { 4 int a; ...

  7. java 8种基本数据类型的默认值及所占字节数

    通过一段代码来测试一下 8种基本数据类型的默认值 package dierge; public class Ceshi { int a; double b; boolean c; char d; fl ...

  8. - Java中boolean类型占用多少个字节 MD

    目录 目录 Java中boolean类型占用多少个字节 1个bit(1位) 1个Byte(1字节,8位) 4个Byte(4字节,32位) 分析 官方文档中的描述 Markdown版本笔记 我的GitH ...

  9. .SQL Server中 image类型数据的比较

    原文:.SQL Server中 image类型数据的比较 在SQL Server中如果你对text.ntext或者image数据类型的数据进行比较.将会提示:不能比较或排序 text.ntext 和 ...

随机推荐

  1. MIUI系统如何获取ROOT权限

    MIUI系统有么好方法启用了Root超级权限?各位都清楚,Android手机有Root超级权限,一旦手机启用了root相关权限,就能够实现更多的功能,举例子,各位公司的营销部门的同事,使用大多数营销工 ...

  2. ARM LCD屏调试3--屏的应用编程

    2011-06-25 19:20:47 驱动自己写完了,应用函数自己就不写了,找了一点代码参考,移植并修改了一下,配合之前的定义的接口文档,我贴出部分代码.目录: 一,开发环境... 1 二,底层函数 ...

  3. yum仓库源搭建

    本地yum源搭建 cd源 mount /dev/sr0 /mnt vim  /etc/yum.repos.d/base.repo [centos-base]name=centos7#baseurl=f ...

  4. [dart学习]第三篇:dart变量介绍 (二)

    本篇继续介绍dart变量类型,可参考前文:第二篇:dart变量介绍 (一) (一)final和const类型 如果你不打算修改一个变量的值,那么就把它定义为final或const类型.其中:final ...

  5. Gradle Repository

    // 该init.gradle文件,请保存到${USER_HOME}/.gradle/文件夹下,如:C:\Users\Administrator\.gradle allprojects { repos ...

  6. isolate demo

    dependencies dependencies: flutter: sdk: flutter # The following adds the Cupertino Icons font to yo ...

  7. golang设置title并获取窗口句柄

    package main import ( "fmt" "syscall" "github.com/lxn/win") func main( ...

  8. linux启动httpd服务出现 Could not reliably determine the server`s fully qualified domain name.

    安装好apache启动httpd服务时,出现httpd: Could not reliably determine the server's fully qualified domain name,  ...

  9. 单元测试系列之十:Sonar 常用代码规则整理(二)

    摘要:帮助公司部署了一套sonar平台,经过一段时间运行,发现有一些问题出现频率很高,因此有必要将这些问题进行整理总结和分析,避免再次出现类似问题. 作者原创技术文章,转载请注明出处 ======== ...

  10. xshell终端向远程服务器上传文件方法

    centos-7下在本地终端里向远程服务器上传文件,在命令行中执行的软件. 安装命令如下: 在终端里输入如下命令: 会弹出如下窗口 选择你要上传的文件即可上传成功.