C++ ——构造函数

news/2025/2/21 2:36:50/

1、作用:创建对象时,给对象的属性进行初始化

2、特点

        (1)构造函数与类同名

        (2)如果没有显式给出构造函数,编译器会给出默认的构造函数(参数为空,并且函数体也为空);如果给出任意的构造函数,系统默认的构造函数就不存在了

        (3)有返回值,但是不写返回值类型,也不可以写void

        (4)构造函数是在创建对象时自动调用

3、构造函数支持函数重载

4、构造函数也支持函数默认值

1、无参构造函数

编译器默认的构造函数:

    类名(){
   }
#include <iostream>
using namespace std;
class Phone{
private:  
    string brand;  //品牌
    string color;  //颜色
    double price;  //价格
public:
    void play_music() {
        cout<<"播放音乐"<<endl;
    }
    void play_video(){
        cout<<"播放视频"<<endl;
    }
    //无参构造函数
    Phone(){
        cout<<"构造函数"<<endl;
    }
};
int main(){
    //无参构造函数
    Phone p1;
    return 0;
}

2、有参构造函数

#include <iostream>
using namespace std;
class Phone{
private: 
    string brand;  //品牌
    string color;  //颜色
    double price;  //价格
public:
    void play_music() {
        cout<<"播放音乐"<<endl;
    }
    void play_video(){
        cout<<"播放视频"<<endl;
    }
    //有参构造函数
    Phone(string a,string b,double c=2995){
        brand=a;
        color=b;
        price=c;
    }
};
int main(){
    //有参构造函数栈内存对象
    Phone p2("vivo","红",9);
    //有参构造函数堆内存对象
    Phone*p3=new Phone("oppo","蓝",4);
    delete p3;
    p3=NULL;
    return 0;
}

3、构造初始化列表

构造初始化列表:就是构造函数的一种简便写法

注意:构造初始化列表与构造函数不能同时出现

#include <iostream>
using namespace std;
class Person{
private: 
    int id;
    string name;
    string sex;
    bool flag;
public:
    //构造初始化列表
    Person(int id,string name,string sex,bool flag)
        :id(id),name(name),sex(sex),flag(flag){}
    void show(){
        cout<<id<<name<<sex<<endl;
    }
};
 
int main(){
    Person p1(1001,"张三","男",true);
    p1.show();
    return 0;
}

4、拷贝构造函数

作用:用于实现对象的拷贝创建 

特点:

        (1)拷贝构造函数与构造函数构成重载(拷贝构造函数也是与类同名)

        (2)如果不给出拷贝构造函数,编译器会给出默认的拷贝构造函数,完成对象之间的值复制;如果给出拷贝构造函数,编译器就不会提供默认的拷贝构造函数

        (3)拷贝构造函数的参数类型是对象的引用或者是const修饰的对象的引用

        (4)拷贝构造函数是在拷贝创建对象时自动调用

注意:对象之间是相互独立的,对象之间的属性也是相互独立的

4.1 浅拷贝 

编译器默认给出的拷贝构造函数,完成的就是浅拷贝,会完成对象之间简单的值复制 

#include <iostream>
using namespace std;
class Person{
private:  
    int id;
    string name;
    string sex;
    bool flag;
public:
    //构造初始化列表
    Person(int id,string name,string sex,bool flag)
        :id(id),name(name),sex(sex),flag(flag){}
    //编译器给出的默认的拷贝构造函数(手写出来的)----->浅拷贝 
    Person(const Person&p){
        id=p.id;
        name=p.name;
        sex=p.sex;
        flag=p.flag;
    }
    void show(){
        cout<<id<<name<<sex<<endl;
    }
};
int main(){
    Person p1(1001,"张三","男",true);  //调用有参的构造函数
    p1.show();
//    //方法一:
//    Person&pp=p1;
//    Person p2(pp);
    //方法二:
    Person p2(p1);  //调用拷贝构造函数
    p2.show();
    return 0;
}

默认拷贝构造函数存在安全隐患:如果成员变量是指针类型,两个对象的指针类型属性指向同一个内存空间,破坏了对象的独立性,那么这种拷贝叫浅拷贝 

解决方法:使用深拷贝

4.2 深拷贝

实现方式:创建对象时,指针属性要有自己独立的区域;拷贝对象时,由地址拷贝变成内容拷贝

#include <iostream>
#include <string.h>
using namespace std;
class Animal{
private:
    string kind;  //种类
    double weight;  //体重
    char*hobby;  //爱好
public:
    //构造函数---->深拷贝
    Animal(string k,double w,char*h){
        kind=k;
        weight=w;
        //创建对象时,指针属性要有自己独立的区域
        hobby=new char[20];
        strcpy(hobby,h);  //使用strcpy前,需要引入string.h头文件
    }
    //展示信息
    void show(){
        cout<<"种类:"<<kind<<",体重:"<<weight<<",爱好:"<<hobby<<endl;
    }
    //kind读接口
    string get_kind(){
        return kind;
    }
    //拷贝构造函数---->深拷贝
    Animal(const Animal&c){
        kind=c.kind;
        weight=c.weight;
        //拷贝对象时,由地址拷贝变成内容拷贝
        hobby=new char[20];
        strcpy(hobby,c.hobby);
    }
};
int main(){
    char h[20]="eat fish";
    Animal cat1("小猫",12,h);  //调用有参构造函数
    cat1.show();
    Animal cat2(cat1);  //调用拷贝构造函数
    cat2.show();
    return 0;
}

4.3 隐式调用构造函数

截止到目前,对于构造函数的调用都是显式调用

隐式调用构造函数的出现情况:

        (1)等号赋值时,等号左侧是对象类型,等号右边恰好是对象构造函数所需要的参数类型,这时就会把右侧值传入到构造函数中,相当于隐式调用构造函数

#include <iostream>
using namespace std;
class Test{
private:
    int number;
public:
    Test(int number)
        :number(number){}
    int get_number(){
        return number;
    }
};
int main(){
//    Test t1(100);  //显示调用构造函数
//    cout<<t1.get_number()<<endl;
//    Test*t2=new Test(8);  //显示调用构造函数
//    cout<<t2->get_number()<<endl;
//    delete t2;
//    t2=NULL;
//    Test t3(t1);  //显示调用构造函数
//    cout<<t3.get_number()<<endl;
    Test t4=9;  //隐式调用构造函数
    cout<<t4.get_number()<<endl;
    Test t5=t4;  //隐式调用构造函数
    cout<<t5.get_number()<<endl;
    return 0;
}

        (2)隐式调用构造函数,一般在程序员不自知的情况下产生的,需要规避掉,可以用explicit关键字屏蔽

class Test{
private:
    int number;
public:
    explicit Test(int number)
        :number(number){}
    int get_number(){
        return number;
    }
};

http://www.ppmy.cn/news/1572820.html

相关文章

Pygame中自定义事件处理的方法2-2

在《Pygame中自定义事件处理的方法2-1》中提到了处理自定义事件的方法。通过处理自定义事件&#xff0c;可以实现动画等效果。 1 弹跳小球程序 通过处理自定义事件&#xff0c;可以实现弹跳小球程序&#xff0c;如图1所示。 图1 弹跳小球程序 2 弹跳小球程序原理 实现弹跳小…

《探秘Windows 11驱动开发:从入门到实战》

《探秘Windows 11驱动开发:从入门到实战》 驱动开发初印象 在 Windows 11 这个充满活力与创新的操作系统世界里,驱动程序犹如幕后英雄,默默发挥着无可替代的关键作用。它是连接操作系统与硬件设备的桥梁,操作系统下达的指令,如播放音乐、读取硬盘数据等,都需要通过驱动…

排序算法衍生问题

排序算法衍生问题 引言 排序算法是计算机科学中的基本问题之一&#xff0c;广泛应用于各种实际场景中。尽管有多种排序算法可供选择&#xff0c;但每种算法都有其特定的优势和局限性。本文将探讨排序算法中的一些衍生问题&#xff0c;包括算法效率、稳定性、内存占用等方面&a…

[特殊字符] C语言中打开和关闭文件的两种方法:标准库 VS 系统调用

C语言中对文件打开关闭操作 前言方法一&#xff1a;标准输入输出库&#xff08;stdio.h&#xff09;—— 高级文件操作的利器打开文件&#x1f4a1; 关闭文件&#xff1a;fclose示例代码&#x1f4dd; 个人见解 方法一&#xff1a;系统调用&#xff08;fcntl.h 和 unistd.h&…

Python学习心得字符串的去重操作

一个字符串中可能包含许多相同的元素&#xff0c;为了保证字符串中的唯一性&#xff0c;下面介绍的是字符串的去重操作&#xff1a; 第一种方式&#xff1a;利用forif的结构进行去重 这个程序是对字符串中的每个元素进行判断&#xff0c;如果不在新建的空字符串中就把该元素添…

WeMos D1+PIR+Android 的小场景制作

最近在做一个有趣的小场景功能&#xff0c;其实已经有成熟产品&#xff0c;但是考虑到没法实现场景扩展&#xff0c;所以自己开始动手做。 场景描述&#xff1a;玄关人体感应&#xff0c;有人进门&#xff0c;致欢迎词&#xff0c;有人离开&#xff0c;致欢送词。 硬件设备&a…

Ai知识点总结

图层–》后面小圈–》释放到图层顺序 illustrator&#xff1a;插画 位图&#xff1a;可以制作色彩和色调变换丰富的图像&#xff0c;容易在不同软件之间变换文件。所占容量大&#xff0c;放大、缩小、旋转会导致失真&#xff0c;无法制作真正的3D图像。 矢量图&#xff1a;所占…

C++ 设计模式-桥接模式

C桥接模式的经典示例&#xff0c;包含测试代码&#xff1a; #include <iostream> #include <string>// 实现化接口 class Device { public:virtual ~Device() default;virtual bool isEnabled() const 0;virtual void enable() 0;virtual void disable() 0;vi…