七:拷贝构造、拷贝复制、析构
1.编译器自动给的BIG THREE,其中默认拷贝构造函数、拷贝赋值函数(参数是本类的对象或引用),会忠实地对每一个bit进行拷贝,因此如果类中有指针的时候,必须自己实现可以深拷贝的拷贝构造、拷贝赋值;
拷贝构造函数的两种调用方式:
ØString s2(s1);//以s1为蓝本,调用拷贝构造函数,创建s2
ØString s2=s1;//创建新的s2,内部如果有指针成员,则指针成员需要调用其构造函数完成深拷贝
2.深拷贝:把指针指向的内容拷贝过去;浅拷贝:只拷贝了指针,两个指针指向相同的内容,会导致内存泄露;
3.拷贝赋值函数中,先把自己的指针指向的内容清空,再重新创建一块和拷贝源侧一样大的内存,最后把拷贝源侧的内容拷贝到这块内存上;一定要进行自我赋值检测:否则,如果进行自我赋值,先清空内容,无法再继续赋值;
拷贝赋值的步骤:
Ø自我赋值检测
Ø清空自身内存块
Ø分配与源侧一样大的内存 ?????
ØmemoryCopy ??????
4、拷贝构造函数、赋值操作符编写的时 候,两个要点
1、对于父类部分,首先考虑要调用父类的拷贝构造函数、赋值操作符进行初始化
2、对于指针成员,考虑其深度拷贝问题:在构造函数中,在赋值操作符中,不完全一样
八:堆、栈与内存管理
1.Stack:是存在于某作用域内的一块内存空间,调用函数,会形成一个stack,其中局部变量的生命周期仅限于该作用域。
2.Heap:操作系统提供的一块global内存空间,可以动态分配,生命在delete后结束
3.Static:静态对象,生命周期是整个程序的执行期
4.Global:全局对象,生命周期也是整个程序的执行期,可以看做static
5.New:用于在heap中动态分配空间,编译器分解为三个动作:
Complex* pc=new Complex(1,2)
ØVoid* mem=operator new(sizeof(Complex));//operator new调用malloc分配内存
ØPc=static_cast(mem);//void*指针转型
ØPc->Complex::Complex(1,2);//调用构造函数
6.Delete:释放new分配的动态内存空间,编译器分解为两个动作:清内存,放指针
String* ps=new string(“hello”);
Delete ps;
ØString::~strting(ps);//调用析构函数,会把动态分配的内存清零,指针还存在
ØOperator delete(ps);//释放内存,调用free(ps),删除string中的指针
7.VC中动态分配内存块
Debug模式:28+4*2(所用内存上下的0xfd)+4*2(全部内存首尾的cookie)
Release模式:4*2(只会多两个cookie)
分配的内存块必须是16字节的整数倍
Cookie的末位为1表示该内存被分配,为0表示内存未被分配,倒数第二位*16为内存字节数
8.动态分配数组内存
多开辟一个字节用以保存数组大小,以在delete[]的时候,多次唤起dtor
9.Array new一定要搭配array delete
如果数组中的对象不带pointer,则使用delete pointer和delete[] pointer都一样,并不会造成内存泄露,如果数组中的对象携带pointer,则使用delete pointer会泄露内存。
EX:string *s=new string[10];
Delete s;
ØString::~string()//只数组第一个成员调用析构函数,在该析构函数中会清空内存,释放*c_string指针。但是其余9个数组成员的析构函数没有被调用,在堆上分配的内存没有被回收
ØFree(s);//释放s指针,s指针指向的内存被成功回收
十、类模板、函数模板
1.Static:
Østatic data存储在静态数据区,只有一份,所有对象共享。
ØStatic member func没有this指针,所以不能处理普通数据成员,只能处理静态数据成员。可以通过对象、类名调用。
2.This:成员函数在内存空间中只有一份,不同的对象在调用成员函数的时候,成员函数怎么找到对象的数据成员呢?::成员函数有一个隐藏参数:this指针,编译在第一个位置,成员函数通过this指针找到对象的数据成员。
3.类模板使用时传入类型参数,编译期具现化为模板类,模板类运行时生成对象。
4.类模板在使用时需明确指出参数,函数模板在使用时,编译期可以进行参数推导,参数推导时,类型参数需要隐式支持函数内对该参数的要求。
Q:1、m_data =newchar[strlen(c) + 1];//new typename[length]