C++学习8 构造函数的参数初始化表

构造函数是一种特殊的成员函数,在创建对象时自动执行,主要用来进行初始化工作,例如对 private 属性的成员变量赋值。

对成员变量的初始化,除了在构造函数的函数体中一一赋值,还可以采用参数初始化表。请看下面的代码:

class Student{
private:
    char *name;
    int age;
    float score;
public:
    Student(char *, int, float);
    void say();
};
//在构造函数中采用参数初始化表
Student::Student(char *name1, int age1, float score1):name(name1), age(age1), score(score1){}
void Student::say(){
    cout<<name<<"的年龄是 "<<age<<",成绩是 "<<score<<endl;
}

如本例所示,在定义带参构造函数 Student(char *name1, int age1, float score1) 时,不再是在函数体中对成员变量一一赋值,其函数体为空。在函数首部与函数体之间增添了一个冒并加上 name(name1), age(age1), score(score1) 语句,这个语句的意思相当于函数体内部的 name = name1; age = age1; score = score1; 语句。

参数初始化表还有一个很重要的作用,那就是为 const 成员变量初始化。例如:

class Array{
public:
    Array(){
        length = 0; //compile error
        num = NULL;
    };
private:
    const int length;
    int * num;
};

在本例中Array类声明了两个成员变量,length 和 num 指针,需要注意的是 length 加了 const 关键字修饰。此时默认构造函数再为 length 赋值为 0,这是无法通过编译的

初始化 const 成员变量的唯一方法只有利用参数初始化表。例如:

class Array{
public:
    Array(): length(0){
        num = NULL;
    };
private:
    const int length;
    int * num;
};

需要注意的是:参数初始化顺序与初始化表列出的变量的顺序无关,参数初始化顺序只与成员变量在类中声明的顺序有关。

C++析构函数

创建对象时系统会自动调用构造函数进行初始化工作,同样,销毁对象时系统也会自动调用一个函数来进行清理工作(例如回收创建对象时消耗的各种资源),这个函数被称为析构函数。

析构函数(Destructor)也是一种特殊的成员函数,没有返回值,不需要用户调用,而是在销毁对象时自动执行。与构造函数不同的是,析构函数的名字是在类名前面加一个”~“符号。

注意:析构函数没有参数不能被重载因此一个类只能有一个析构函数。如果用户没有定义,那么编译器会自动生成。

析构函数举例:

#include <iostream>
using namespace std;
class Student{
private:
    char *name;
    int age;
    float score;
public:
    //构造函数
    Student(char *, int, float);
    //析构函数
    ~Student();
    //普通成员函数
    void say();
};
Student::Student(char *name1, int age1, float score1):name(name1), age(age1), score(score1){}
Student::~Student(){
    cout<<name<<"再见"<<endl;
}
void Student::say(){
    cout<<name<<"的年龄是 "<<age<<",成绩是 "<<score<<endl;
}
int main(){
    Student stu1("小明", 15, 90.5f);
    stu1.say();
   
    Student stu2("李磊", 16, 95);
    stu2.say();
   
    Student stu3("王爽", 16, 80.5f);
    stu3.say();
    cout<<"main 函数即将运行结束"<<endl;
   
    return 0;
}

可以看出,析构函数在 main 函数运行结束前被执行,并且调用顺序和构造函数正好相反,为了方便记忆,我们可以将之理解为一个栈,先入后出

析构函数在对象被销毁前执行;要知道析构函数什么时候被调用,就要先知道对象什么时候被销毁。

对象可以认为是通过类这种数据类型定义的变量,它的很多特性和普通变量是一样的,例如作用域、生命周期等。由此可以推断,对象这种变量的销毁时机和普通变量是一样的。

总结起来,有下面几种情况:
1) 如果在一个函数中定义了一个对象(auto 局部变量),当这个函数运行结束时,对象就会被销毁,在对象被销毁前自动执行析构函数。

2) static 局部对象在函数调用结束时并不销毁,因此也不调用析构函数,只有在程序结束时(如 main 函数结束或调用 exit 函数)才调用 static 局部对象的析构函数。

3) 如果定义了一个全局对象,也只有在程序结束时才会调用该全局对象的析构函数。

4) 如果用 new 运算符动态地建立了一个对象,当用 delete 运算符释放该对象时,先调用该对象的析构函数

注意:析构函数的作用并不是删除对象,而是在撤销对象占用的内存之前完成一些清理工作,使这部分内存可以分配给新对象使用。

posted on 2016-08-21 12:20  白宫飘红旗  阅读(496)  评论(0编辑  收藏  举报