C++入门09 -- 拷贝构造函数,深拷贝,浅拷贝,匿名对象,隐式构造,默认构造函数

拷贝构造函数

  • 拷贝构造函数是构造函数的一种;
  • 当利用一个已经存在的对象创建一个新的对象(类似于拷贝),就会调用新对象的拷贝构造函数进行初始化;
  • 拷贝构造函数的格式是固定的,以const引用作为参数,注意是引用reference,来接收传进来的对象,利用传进来的对象,创建出一个新的对象;
  • 如果不自定义拷贝构造函数,会调用C++系统默认的拷贝构造函数,如果自定义了拷贝构造函数,就调用自定义拷贝构造函数;
  • 针对成员变量是基本数据类型的,使用系统默认的拷贝构造函数就可以了,能将参数对象的所有成员变量的数据全部拷贝到新创建对象对应的成员变量中;
  • 针对成员变量是引用,指针类型的,必须自定义拷贝构造函数,实现深拷贝,而C++系统的数据拷贝默认是浅拷贝,会出现内存问题的,下面将会有详细的案例描述;
#include <iostream>

using namespace::std;

class Car {
    int m_price;
    int m_length;
    
public:
    Car(int price = 0,int length = 0) : m_price(price),m_length(length){
        cout << "Car(int price = 0,int length = 0)" << endl;
    }
    
    //1.拷贝构造函数 格式是固定的
    Car(const Car &car){
        this->m_price = car.m_price;
        this->m_length = car.m_length;
        cout << "Car(const Car &car)" <<endl;
    }
    
    //2.与系统的拷贝构造函数 功能相同 初始化列表写法
    Car(const Car &car) : m_price(car.m_price),m_length(car.m_length){
        cout << "Car(const Car &car)" <<endl;
    }
    
    void display(){
        cout << "price = " << this->m_price << endl;
        cout << "length = " << this->m_length << endl;
    }
    
};

int main(int argc, const char * argv[]) {
   
    Car car2(100,10);
    
    //利用car2对象创建了car3,会调用car3对象的拷贝构造函数进行初始化
    Car car3(car2);
    car3.display();
    
    //创建了car6对象,等价于Car car4(car2) 拷贝构造函数
    Car car4 = car2;
    car4.display();
    
    Car car5;
    //这里是赋值操作,没有创建新对象,所以不会调用拷贝构造函数
    car5 = car2;
    
    cout << "&car2 = " << &car2 << endl;
    cout << "&car3 = " << &car3 << endl;
    cout << "&car4 = " << &car4 << endl;
    
    return 0;
}
  • 子类调用父类的拷贝构造函数
#include <iostream>

using namespace::std;

class Person {
    int m_age;
    
public:
    //构造函数
    Person(int age = 0) : m_age(age){ }
    //拷贝构造函数
    Person(const Person &person) : m_age(person.m_age){ }
};

class Student : public Person {
    int m_score;
    
public:
    //构造函数
    Student(int age = 0,int score = 0) : Person(age),m_score(score){ }
    //拷贝构造函数 调用父类的拷贝构造函数 初始化成员变量m_age
    Student(const Student &student) : Person(student),m_score(student.m_score){ }
};

int main(int argc, const char * argv[]) {
    
    return 0;
}

浅拷贝,深拷贝

  • C++编译器默认提供的拷贝是浅拷贝;
  • 将一个对象中的所有成员变量的值拷贝到另一个对象;
  • 如果某个成员变量是指针,只会拷贝指针中所存储的地址值,并不会拷贝指针所指向的内存空间;
  • 可能会导致堆空间的多次释放问题;
  • 如果需要实现深拷贝,需要自定义拷贝构造函数;
  • 将指针类型的成员变量所指向的内存空间,拷贝到新的内存空间;
class Car {
public:
    int m_price;
    char *m_name;
    //构造函数
    Car(int price = 0,char *name = NULL) : m_price(price){
        cout << "Car(int price = 0,char *name = NULL)" << endl;
        //字符串要单独处理
        if (name == NULL) return;
        //申请堆空间 存储字符串内容
        this->m_name = new char[strlen(name)+1]{};
        //拷贝字符串内容到堆空间
        strcpy(this->m_name, name);
    }
    
    //拷贝构造函数
    Car(const Car &car) : m_price(car.m_price){
        cout << "Car(const Car &car)" << endl;
        //字符串要单独处理
        if (car.m_name == NULL) return;
        //申请堆空间 存储字符串内容
        this->m_name = new char[strlen(car.m_name)+1]{};
        //拷贝字符串内容到堆空间
        strcpy(this->m_name, car.m_name);
    }
    
    ~Car(){
        cout << "~Car()" << endl;
        if (this->m_name == NULL) return;
        delete [] this->m_name;
        this->m_name = NULL;
    }
    
    void display(){
        cout << "price = " << this->m_price << endl;
        cout << "name = " << this->m_name << endl;
    }
};
int main(int argc, const char * argv[]) {
    
    char name[] = {'b','m','w','\0'};
    //堆对象
    Car *car = new Car(100,name);
    
    cout << car->m_price << endl;
    cout << car->m_name << endl;
    
    delete car;
    
    //栈对象
    Car car1(100,"bmw");
    Car car2 = car1;
    
    return 0;
}
  • 针对构造函数中 有字符串成员变量的,如下图所示:
Snip20210814_158.png
  • 针对堆对象的构造函数,需要将name字符串数组的内容从栈区拷贝到堆区,因为栈区的name可能会随时被释放掉,所以构造函数在初始化字符串成员变量时要特殊处理,拷贝字符串内容到新的堆空间中;
  • 针对栈对象的拷贝构造函数,原理图如下所示:
    Snip20210814_160.png
  • 系统默认的拷贝构造函数是浅拷贝,那么拷贝的新对象,都指向同一块内存name,所以需自定义拷贝构造函数,实现name的深拷贝,保证每一个car对象都有自己独立的name;否则会造成name的多次释放以及修改一个car对象,其他car对象也会变动的问题;
  • 析构函数,针对name的堆空间也要做释放处理;

对象类型与返回值

  • 使用对象类型作为函数的参数或者返回值时,会产生一些不必要的中间对象;
#include <iostream>

using namespace::std;

class Car {
    int m_price;
    
public:
    Car(int price = 0) : m_price(price){
        cout << "Car(int price = 0)" << this << "-" << m_price << endl;
    }
    
    Car(const Car &car) : m_price(car.m_price){
        cout << "Car(const Car &car)" << this << "-" << m_price << endl;
    }
};

//对象形参
void test(Car car){
    
}

int main(int argc, const char * argv[]) {
    
    Car car1(10);
    //传参的 会产生一个新的对象 因为拷贝构造函数
    test(car1);
  
    return 0;
}
  • test(car1),car1传递给形参car,即Car car = car1会调用car的拷贝构造函数,生成一个新的临时对象,这个临时对象的创建完全没有必要;
  • 在定义函数形参时,为了避免调用拷贝构造函数,生成新的临时对象,定义形参使用对象的引用,就能避免,如下所示:
对象的引用形参
void test(Car &car){
    
}

匿名对象

  • 匿名对象:没有变量名,没有被指针指向的对象,用完之后马上析构销毁;
#include <iostream>

using namespace::std;

class Person {

public:
    
    Person(){
        cout << "Person()" << endl;
    }
    
    Person(const Person &person){
        cout << "Person(const Person &person)" << endl;
    }
    
    ~Person(){
        cout << "~Person()" << endl;
    }
    
    void display(){
        cout << "display()" << endl;
    }
};

void test1(Person person){
    
}


int main(int argc, const char * argv[]) {
    
    Person person;

    //匿名对象
    Person().display();

    test1(Person());
    
    return 0;
}

隐式构造

  • C++存在隐式构造的现象,某些情况下,会隐式的调用单参数的构造函数;
#include <iostream>

using namespace::std;

class Person {
    int m_age;
public:
    
    Person() : Person(0){
        cout << "Person()" << endl;
    }
    
    Person(int age = 0) : m_age(age){
        cout << "Person(int age = 0)" << endl;
    }
    
    Person(const Person &person){
        cout << "Person(const Person &person)" << endl;
    }
    
    ~Person(){
        cout << "~Person()" << endl;
    }
    
    void display(){
        cout << "display() age = " << this->m_age << endl;
    }
};

void test1(Person person){
    
}

Person test2(){
    return 30;
}
Snip20210814_163.png
  • Person person = 10 等价于 Person person = Person(10)

  • explicit关键字 可禁止使用单参数隐式构造函数;

#include <iostream>

using namespace::std;

class Person {
    int m_age;
public:
    
    Person() : Person(0){
        cout << "Person()" << endl;
    }
    
    explicit Person(int age = 0) : m_age(age){
        cout << "Person(int age = 0)" << endl;
    }
    
    Person(const Person &person){
        cout << "Person(const Person &person)" << endl;
    }
    
    ~Person(){
        cout << "~Person()" << endl;
    }
    
    void display(){
        cout << "display() age = " << this->m_age << endl;
    }
};

编译器自动生成的构造函数

  • C++编译器在某些特定的环境下,会默认生成无参的构造函数;
#include <iostream>

using namespace::std;

class Person {
    
public:
    int m_age;
    
    Person(int age = 0) : m_age(age){
        cout << "Person() " << endl;
    }
};

int main(int argc, const char * argv[]) {
    
    Person person;//没有生成构造函数 
    person.m_age = 10;
    
    return 0;
}
  • Person person; 没有生成构造函数,实例化对象没有调用函数;
  • 以下环境会生成无参构造函数:
    • 成员变量在声明的同时进行了初始化;
    • 有定义虚函数;
    • 虚继承其他类;
    • 包含了对象类型的成员,且对象成员有构造函数;
    • 父类有构造函数;
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,547评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,399评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,428评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,599评论 1 274
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,612评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,577评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,941评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,603评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,852评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,605评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,693评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,375评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,955评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,936评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,172评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,970评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,414评论 2 342

推荐阅读更多精彩内容