C++用new与不用new创建对象的区别

192 阅读2分钟
C++创建对象一、Alignment问题
重新发现这个问题是因为在体系结构课上提到的一个概念,alignment对齐的概念。
class MyClass {
public
:
char
c;
// 1 byte
int
i;
// 4 byte
};
像上面这个C++的类,它有两个变量。
MyClass myclass;
cout
<<
sizeof
(myclass) <<
endl
;
或者直接算它的大小
cout
<<
sizeof
(MyClass) <<
endl
;
然后按照道理,myclass的大小,应该是5 byte才对,可是最后输出的结果是8,显然这就说明系统在创建对象的时候,将对象的变量进行了对齐操作,不足4 byte的按照4 byte补足。
2.1 虚函数引起的空间占用:虚函数管理
此外,普通成员函数不会占用对象空间的大小,也不会影响sizeof的结果。
而如果是虚函数就会占用空间了。
class MyClass {
public
:
char
c;
// 1 byte
int
i;
// 4 byte
void func(); virtual void func2() //4 byte {}};
结果是占用12 byte的大小。
这种的差别是由于c++的类所有的虚函数都是由一个虚函数指针所管理,它的虚函数被放在别的内存空间中管理。而普通函数是由this指针所管理,this指针并不是对象本身的一部分,所以不会影响到sizeof的结果。
2.2 一些有趣的现象
  • 空类的sizeof为1
  • 只有一个char的类的大小为1
  • 当有一个为char,另一个是虚函数或者int类型成员变量,会出现对齐现象,大小为8。对齐现象很有意思,它的规则是:以最长的变量长度为对齐长度
class MyClass {
public
:
char
c;
// 1 byte
long
long
t;};
long long为8 byte,所以总共占用16 byte。
在分析完各种情况之后,只有第一个的结果显得特别诡异,为什么空类的大小不是0,而是1?
每个类的实例,在内存中都有一个独一无二的地址,为了达到这个目的,编译器往往会给一个空类隐含的加一个字节,这样空类在实例化后在内存得到了独一无二的地址。------《深度探索c++对象模型》
二、用new与不用new创建对象的区别
内存分配有三种方式:
  • 从静态存储区域分配,内存在编译期间就已经分配好,这块内存在整个运行期间都存在,例如static变量。
  • 从栈上创建:函数内局部变量,自动分配与回收,效率高,但是分配的内存量有限。
  • 从堆上创建:由程序员控制,malloc、new,free、delete。