美文网首页
Boolan微专业-面向对象高级编程学习笔记(Week02)

Boolan微专业-面向对象高级编程学习笔记(Week02)

作者: GoMomi | 来源:发表于2018-01-15 03:14 被阅读0次

    Class with pointer member(stirng)

    1. Big Three

    • 拷贝构造 String(const String& str);
    • 拷贝赋值 String& operator=(const String& str);
    • 析构函数 ~String();

    构造函数和析构函数

    • 构造函数
      • String(const char* cstr = 0);
      • 字符串构造要注意检查是否为 nullptr
    • 析构函数
      • 带有指针的 Class 多半会动态分配内存,因此在析构函数中要主动将动态分配的内存 delete 掉
      • inline String::~String(){
          delete[] m_data;
        }
        
    • Class with pointer members 必须要有 copy ctor 和 copy op=
      • 默认的copy ctor 和 copy op= 只会进行浅拷贝,会导致两个指针指向同一内存块,形成 alias 和 memory leak
      • 拷贝构造&拷贝赋值

    拷贝构造

    • inline String::String(const String& str){
        m_data = new char[ strlen(str.m_data) + 1 ];
        strcpy(m_data, str.m_data);
      }
      
    • String s2(s1) <===> String s2 = s1 两者调用的都是拷贝构造函数

    拷贝赋值

    • inline String& String::operator=(const String& str){
        if (this == &str)
          return *this;           // 检测自我赋值
        delete[] m_data;
        m_data = new char[ strlen(str.m_data) + 1 ];
        strcpy(m_data, str.m_data);
        return *this;
      }
      
    • 步骤:
      1. 先释放自己 delete[] m_data
      2. 分配足够的内存 m_data = new char[ strlen(str.m_data) + 1 ];
      3. 拷贝数据 strcpy(m_data, str.m_data);
    • 要点:
      • 返回类型要是String&,用于s1 = s2 = s3的使用情景
      • 一定要检测自我赋值,否则会造成undefined behavior
      • self assignment

    2. Stack(栈) 与 Heap(堆)

    Stack

    • Complex c1(1,2);
    • 内存由编译器在需要时自动分配和释放。通常用来存储局部变量和函数参数。(为运行函数而分配的局部变量、函数参数、返回地址等存放在栈区)。
    • 栈运算分配内置于处理器的指令集中,效率很高,但是分配的内存容量有限。
    • Stack 对象的生命在作用域 Scope 结束之际结束,自动调用其析构函数

    Heap

    • Complex* p = new Comples(3);
    • System Heap, 由操作系统提供的一块 global 内存空间,程序可动态分配从中获得若干区块
    • 要配合使用 delete 或 delete[] 进行释放,否则会造成内存泄漏

    其他内存块 [参考]

    • 自由存储区:就是那些由malloc等分配的内存块,他和堆是十分相似的,使用free来释放内存
    • 全局/静态存储区:全局变量和静态变量被分配到同一块内存中(在以前的C语言中,全局变量又分为初始化的和未初始化的,在C++里面没有这个区分了,他们共同占用同一块内存区)
    • 常量存储区:一块比较特殊的存储区,里面存放的是常量,不允许修改。

    对象

    • Stack Objects
      1. Complex c1(1,2);
      2. 生命在作用域(Scope)结束之际结束
      3. 编译器自动调用其析构函数,因此又称为 auto object
    • Static Stack Objects
      1. static Complex c2(1,2);
      2. 生命在作用域(Scope)结束之后仍存在,直至整个程序结束
    • Global Object
      1. Complex c3(1,2);
        int main(){
        ...
        }
      2. 生命在作用域(Scope)结束之后仍存在,直至整个程序结束
    • Heap Objects
      1. Complex* p = new Complex;
      2. 生命在它被 delete 之际结束,调用其析构函数
      3. 若未 delete 则会造成 memory leak(指针p的生命已经结束了,但所致的heap object仍存在)

    3. new & delete

    new 与 构造

    • 先分配 memory, 再调用 ctor

    • Complex* pc = new Complex(1,2); ==>

      void* mem = operator new(sizeof(Complex)); // 分配内存,内部调用 malloc(n)
      pc = static_cast<Complex*>(mem);           // 转型
      pc->Complex::Complex(1,2);                 // 调用构造函数
      

    delete 与 析构

    • 先调用 dtor, 再释放 memory

    • delete pc; ==>

      Complex::~Complex(pc);                    // 析构函数
      operator delete(pc);                      // 释放内存,内部调用 free(pc)
      

    内存详情

    Single Object

    • Single Object Memory In VC
    • 灰色部分为Debug模式额外添加的信息

    • 绿色部分为对象数据所占空间,此处需要满足内存4字节对齐(青绿色标出)

    • 内存块收尾为标记为,其值代表整个内存块大小。最后1位用于指示内存块的用途,1:送出 0:回收

    Array Object

    • Array Object Memory In VC

    Array New 一定要搭配 Array Delete

    • Array New
    • Array new 分配的内存的使用 delete 释放时,编译器仅会施放申请的内存,且只调用1次析构函数,会导致部分对象未正确析构

    4. Static

    Static Data Member

    • 将数据与对象分离,与类绑定
    • 一定要在类外初始化(真正的分配内存),赋不赋值均可
      • class Account {
        public:
        static double m_rate;
        static void set_rate(const double& x) { m_rate =x; };
        }
        double Accont::m_rate = 8.0;

    Static Function Member

    • 用于操作Static Data
    • 调用方式:
      • 通过 object 调用 Accout a; a.set_rate(7.0);
      • 通过 class name 调用 Account::set_rate(5.0);

    Singleton(单例) [参考] --- 把 ctor 放在 private 区

    class Singleton {
      public:
        static Singleton& Instance() {
        static Singleton theSingleton;
          return theSingleton;
      }
    
    /* more (non-static) functions here */
    
    private:
      Singleton();                            // ctor hidden
      Singleton(Singleton const&);            // copy ctor hidden
      Singleton& operator=(Singleton const&); // assign op. hidden
      ~Singleton();                           // dtor hidden
    };
    
    • 在 Instance() 调用前,不会存在 theSingleton对象,没有内存的浪费

    概念辨析

    • Static 全局变量 vs. 普通全局变量
      • 全局变量本身就是静态存储方式, 静态全局变量当然也是静态存储方式。 这两者在存储方式上并无不同
      • 非静态的全局变量在各个源文件中都是有效的
      • 而静态全局变量则限制了其作用域, 只在定义该变量的源文件内有效, 在同一源程序的其它源文件中不能使用它
      • 全局变量改为静态后改变了它的作用域
    • Static 局部变量 vs. 普通局部变量
      • Static 局部变量存储在静态区,生命直至整个程序结束
      • 普通局部变量存储在栈区,生命仅在 Scope 内有效
      • 局部变量改为静态后改变了它的生命期

    相关文章

      网友评论

          本文标题:Boolan微专业-面向对象高级编程学习笔记(Week02)

          本文链接:https://www.haomeiwen.com/subject/btfroxtx.html