c++入门学习④——对象的初始化和清理

news2024/11/18 9:24:57

目录

对象的初始化和清理:

why?

如何进行初始化和清理呢?

使用构造函数和析构函数​编辑

构造函数语法:

析构函数语法:

构造函数的分类:

两种分类方式:

三种调用方法:

括号法(默认构造函数调用)

显示法

隐式转换法

拷贝调用函数的调用时机

1.使用一个已经常见完毕的对象来初始化一个新对象

2.值传递的方式给函数参数传值。

3.以值方式返回局部对象

构造函数调用时机:

深拷贝浅拷贝:

有可能的错误:

错误原因:

解决方案:

 深拷贝和浅拷贝案例:

 初始化列表

基本初始化列表案例:

灵活定义初始化的数字时的初始化列表案例:

类对象作为类成员

静态成员

静态成员变量:

 静态成员变量访问案例:

访问方式:

静态成员函数

静态成员函数创建调用基本示例

why?

结语:


本篇博客讲解c++入门学习④有关对象的初始化以及清理的知识点💪

传送门:

c++入门学习①-CSDN博客

c++入门学习②-CSDN博客

c++入门学习③——封装-CSDN博客

对象的初始化和清理:

why?

为了保证数据安全👉一个对象或者是变量没有初始化,则后面的后果是未知的,同样,没有清理也是可能会造成一系列的安全问题。

如何进行初始化和清理呢?

使用构造函数和析构函数
构造函数语法:

类名(){}
1.构造函数,没有返回值也不写void
2.函数名称与类名相同
3.构造函数可以有参数,因此可以发生重载
4.程序在调用对象时候会自动调用构造,无须手动调用,而且只会调用一次

自己不写,程序会自动帮你实现,不过函数里面的代码是空的,要是自己写,程序运行你自己写的构造函数

析构函数语法:

~类名(){}
1.析构函数,没有返回值也不写void
2.函数名称与类名相同,在名称前加上符号
3.析构函数不可以有参数,因此不可以发生重载
4.程序在对象销毁前会自动调用析构,无须手动调用,而且只会调用一次

自己不写,程序会自动帮你实现,不过函数里面的代码是空的,要是自己写,程序运行你自己写的析构函数

代码:

#include<iostream>
#include<string>
using namespace std;
class person{
     public:
         //创造构造函数
         person() {
             cout << "构造函数" << endl;
         }
         ~person() {
             cout << "析构函数调用" << endl;
         }
};
void test1() {
       person a;
}//局部变量,在栈区上创建,test执行完毕后,释放对象,会自动实现析构函数,不过是空的
//当自己创建了一个,那么就执行自己创建的析构函数

int main()
{
    test1();
    system("pause");
    return 0;
}

构造函数的分类:

两种分类方式:

按照参数分类:有参数和无参数(默认情况下为无参)

按照类型分类:普通和拷贝

拷贝:copy、复制,

例如:person(const person &a){name=a.name;};把其他的属性拷贝到这个上面

拷贝方式创建,注意括号里写要copy的对象,并加上const,以引用的方式写入。

三种构造函数的具体表示:

#include<iostream>
#include<string>
using namespace std;
class person{
     public:
         //创造构造函数
         person() {
             cout << "默认构造函数" << endl;
         }
         person(int age) {
             cout << "有参构造函数" << endl;
             Age = age;
         }
         person(const person& p) {
             cout << "拷贝构造函数" << endl;
             Age = p.Age;
         }
         ~person() {
             cout << "析构函数调用" << endl;
         }
         int Age;
};

引用:c++引用-CSDN博客

三种调用方法:

括号法(默认构造函数调用)

person p1;调用无参构造函数,也是默认构造函数调用;

person p2(12);调用有参构造函数

person p3(p2);调用拷贝构造函数

注意事项:调用默认构造函数的时候不要加(),因为编译器会认为这个是一个函数声明,不会认为在创建对象

显示法

person p1;默认构造

person p2=person(12);有参构造

person p3=person(p2);拷贝构造

单独拿出来person(12),称为匿名对象,当前执行结束后,系统会立即回收掉匿名对象,马上析构掉

不单独看这个,匿名对象就有名字了,左边的就是它的名字

注意:不要利用拷贝构造函数初始化匿名对象。编译器会认为person(p3)==person p3;认为这是一个对象声明

隐式转换法

person p4=10;相当于写了person p4=person(10)有参构造

person p5=p4;拷贝构造

括号法最简单,这里推荐使用这个


拷贝调用函数的调用时机

1.使用一个已经常见完毕的对象来初始化一个新对象

#include<iostream>
using namespace std;
class person{
     public:
         //创造构造函数
         person() {
             cout << "默认构造函数" << endl;
         }
         person(int age) {
             Age = age;
             cout << "有参构造函数" << endl;
         }
         person(const person & a) {
             Age = a.Age;
             cout << "拷贝构造函数" << endl;
         }
         ~person() {
             cout << "析构函数调用" << endl;
         }
         int Age;
};
void test1() {
       person a;//局部变量,在栈区上创建,test执行完毕后,释放对象
}
//使用一个已经创建完毕的对象来初始化一个新的对象
void test2() {
    person p(20);//调用有参构造函数
    person p1(p);//调用拷贝构造函数
}

int main()
{
    //test1();
    test2();
    system("pause");
    return 0;
}

输出结果: 

2.值传递的方式给函数参数传值。

#define  _CRT_SECURE_NO_WARNINGS
#include<iostream>
#include<string>
using namespace std;
class person{
     public:
         //创造构造函数
         person() {
             cout << "默认构造函数" << endl;
         }
         person(int age) {
             Age = age;
             cout << "有参构造函数" << endl;
         }
         person(const person & a) {
             Age = a.Age;
             cout << "拷贝构造函数" << endl;
         }
         ~person() {
             cout << "析构函数调用" << endl;
         }
         int Age;
};
//值传递的方式给函数参数传值
void test(person p) 
{
}
void test3() {
    person p;//调用默认构造函数
    test(p);//调用拷贝构造函数,因为这里是以值传入,相当于拷贝一个副本传进去

}

int main()
{
    //test1();
    //test2();
    test3();
    return 0;
}

3.以值方式返回局部对象

示例:

#define  _CRT_SECURE_NO_WARNINGS
#include<iostream>
#include<string>
using namespace std;
class person{
     public:
         //创造构造函数
         person() {
             cout << "默认构造函数" << endl;
         }
         person(int age) {
             Age = age;
             cout << "有参构造函数" << endl;
         }
         person(const person & a) {
             Age = a.Age;
             cout << "拷贝构造函数" << endl;
         }
         ~person() {
             cout << "析构函数调用" << endl;
         }
         int Age;
};

//以值返回局部对象
person test0() {
    person p;//局部对象在完成函数之后会被自动释放掉,但是现在以值方式返回,则在返回的时候会自动给创建一个新的对象
    cout << "p地址" << (int*) & p << endl;
    return p;//返回的是p拷贝的一个新的对象
}
void test4() {
     person a=test0();//这里的函数a没有调用析构函数和构造函数
     cout << "a地址" << (int*)&a << endl;
}
int main()
{
    //test1();
    //test2();
    //test3();
    test4();
    system("pause");
    return 0;
}

构造函数调用时机:

默认情况添加的函数:

⭐注意:拷贝构造函数默认情况下只写值传递语句,不会出现输出语句

构造函数调用规则如下:
若用户定义有参构造函数,c++不再提供默认无参构造,但是会提供默认拷贝构造函数

当仅定义有参构造函数时不要进行默认无参构造,会发生错误显示没有默认无参构造函数。

若用户定义拷贝构造函数,c++不会再次提供其他构造函数

只定义拷贝构造函数时如果使用其他的构造函数,则会发生错误

深拷贝浅拷贝:

一般浅拷贝是逐个字节依次复制过去,而默认情况下系统创建拷贝构造函数是浅拷贝

有可能的错误:

一般使用不会出错,但是如果数据是程序员开发在堆区,就会发生释放重复的错误

错误原因:

因为浅拷贝使两个对象里的属性指向同一块空间,当释放一个对象中的属性之后,再去释放另一个也在堆区的数据,由于她们指向相同,所指向的空间已经被释放,但是系统不知道因此会发生错误

解决方案:

改进方法就是进行深拷贝,程序员自己创建一个函数来达到深拷贝的效果,深拷贝是在拷贝构造函数内进行自己再开辟一片空间,只复制要拷贝对象属性的内容而非空间地址。

 深拷贝和浅拷贝案例:

#include<iostream>

using namespace std;
class person{
     public:
         //创造构造函数
         person() {
             cout << "默认构造函数" << endl;
         }
         person(int age,int height) {
             Age = age;
             high = new int(height);
             cout << "有参构造函数" << endl;
         }
         person(const person & a) {
             Age = a.Age;
             cout << "person的拷贝构造函数" << endl;
             high=new int(*a.high);
         }
         ~person() {
             //析构代码,将堆区开辟数据做释放操作
             if (high != NULL) {
                 delete high;
                 high = NULL;
             }//加上这个代码后运行,发现程序崩溃了?
             cout << "析构函数调用" << endl;
         }
         int Age;
         int* high;//身高,使用指针,把这个参数存放在堆区
         //堆区的数据,由程序员手动开辟,也由程序员手动释放,销毁前释放,在析构之前释放
};
void test1() {
    person p1(18,160);
    cout << "p1年龄:" << p1.Age<<"p1身高为:"<<*p1.high << endl;
    person p2(p1);//编译器创建了一个拷贝函数,进行了一个浅拷贝的操作,简单的赋值操作,逐字节拷贝
    cout << "p2年龄:" << p2.Age<< "p2身高为:" <<*p2.high << endl;
    //栈区先进后出,所以p2先被释放,p2先执行析构函数,p2释放过了指针所指的内存,则再次用p1释放时发生错误
    //解决:浅拷贝用深拷贝改进,不是逐个字节拷贝了,是重新开辟一个空间,两个空间就不会存在重复释放了
    //自己写拷贝构造函数
}
int main()
{
    test1();
    system("pause");
    return 0;
}

 初始化列表

简介:
初始化类中属性:1创建构造函数2.初始化列表
构造函数():属性1(值1),属性2(值2)

基本初始化列表案例:

#include<iostream>
using namespace std; 
         //初始化列表
         //初始化类中属性:1创建构造函数2.初始化列表
         // 构造函数():属性1(值1),属性2(值2)
class person{
     public:
         //初始化列表给类属性赋初值;
         person() :Age(10),b(20)
         {

             
         }
         int Age;
         int b;
};
void test1() {
    person p1;
    cout << p1.Age <<"\t" << p1.b << endl;
}
int main()
{
    test1();
    system("pause");
    return 0;
}

灵活定义初始化的数字时的初始化列表案例:

#include<iostream>

using namespace std; 
         //初始化列表
         //初始化类中属性:1创建构造函数2.初始化列表
         // 构造函数():属性1(值1),属性2(值2)
class person{
     public:
         //初始化列表给类属性赋初值;
         person(int a,int n) :Age(a),b(n)//注意这个冒号的位置,别写错了
         {

             
         }
         int Age;
         int b;
};
void test1() {
    person p1(12,22);//12先传给了p1中的int a,然后传入到Age里的a,这样Age就被这个数字赋值了,被初始化了。
    cout << p1.Age <<"\t" << p1.b << endl;
}
int main()
{
    test1();
    system("pause");
    return 0;
}

类对象作为类成员

类定义的对象称为另一个类里的成员,则称为类成员

构造顺序和析构顺序相反

构造时,先构造类成员的类,再构造类成员所在的类

而析构顺序则正相反。

#include<iostream>
#include<string>
using namespace std; 
//类对象成为另一个对象的成员,称为类成员
//先有的什么?👉先构造其他的类的对象,再构造这个类
//那么析构的顺序呢?和构造的顺序是相反的。
class A {
public:
    // 可以用Phone a_phone=pname👉隐式转换法来初始化变量
    A(string Name, string pname):name(Name),a_phone(pname)
    {
        cout << "A的构造函数" << endl;
    }
    //姓名
    string name;
    //手机
    Phone a_phone;//类成员
};
//手机类
class Phone{
public:
    Phone(string pname) {//Phone  的构造函数
        Pname = pname;
        cout << "Phone的构造函数" << endl;
    }
    //手机品牌名
    string Pname;
};
void test1() {
    A p("黑菜钟","华为");
    cout << "名字:"<<p.name<< endl;
    cout <<"手机型号:"<< p.a_phone  << endl;
    
}
int main()
{
    test1();
    system("pause");
    return 0;
}

静态成员

定义:静态成员就是在成员变量和成员函数前加上关键字static,称为静态成员

分为:静态成员变量和静态成员函数

静态成员变量要求:所有对象共享同一份数据,编译阶段分配内存,类内声明,类外初始化

静态成员函数要求:

静态成员变量:

静态成员变量示例:

当静态成员变量像平常一样定义类对象并且输出所对应的值的时候,程序会怎么样?

结果:发生错误,无法解析的外部命令👉这是因为静态成员变量要类外进行初始化操作,这个程序显然没做

如何在类外初始化呢?👉int person::m = 100;(这里的person是类名,m是类内静态变量)

 静态成员变量访问案例:

⭐静态成员变量不属于某个对象,所有对象共享同一份

访问方式:

1.通过对象访问
person p;
cout<<p.m_A<<endl;
2.通过类名访问
cout << person::m_A << endl;

#include<iostream>
#include<string>
using namespace std; 
//静态成员变量
//所有对象共享同一份数据
// 编译阶段分配内存
// 类内声明,类外初始化
class person {
public:
    static int m_A;

};
int person::m_A = 10;
void test1() {
    person p;
    cout << p.m_A << endl;//10
    person p2;
    p2.m_A = 20;
    cout << p.m_A << endl;//结果为20,因为所有对象共享数据,一改全改
    //静态成员变量不属于某个对象,所有对象共享同一份
}
void test2() {
    //访问方式1.通过对象访问
    person p;
    cout<<p.m_A<<endl;//20
    //访问方式2.通过类名访问
    cout << person::m_A << endl;//用::来借助访问类对象
}

int main()
{
    test1();
    test2();
    system("pause");
    return 0;
}

注意:静态成员变量只能在公共权限内定义,在私有权限内定义不出错,但是不能共享数据。类外访问不到私有静态变量。

静态成员函数

特点:所有对象共享同一个函数,静态成员函数只能访问静态成员变量

静态成员函数创建调用基本示例

#include<iostream>
#include<string>
using namespace std; 
//静态成员函数
//所有对象共享同一个函数
// 静态成员函数只能访问静态成员变量

class person {
public:
    static void func() {
        cout << "static void func调用" << endl;
    }
};

void test1() 
{
    //访问方式和静态成员变量的访问方式相同
    //1.通过对象
    person p;
    p.func();
    //2.通过类名
    person::func();
}

int main()
{
    test1();
    system("pause");
    return 0;
}

如果静态成员函数访问非静态成员变量会出现👉这样的结果

why?

数据属于特定的,而函数体无法区分是哪个对象的,会出错

而静态成员变量是共享的,所有类的对象都用一份,函数体不需要知道这是哪个类对象对应的

⭐注意: 类外还是不可以访问私有的静态成员函数

结语:

希望这篇有关c++对象初始化和清理的博客对大家有所帮助,欢迎大佬们留言o(* ̄▽ ̄*)ブ

一起学习进步!!!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1435329.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

thinkphp数据批量提交(群发消息)

<form id="edit-form" class="form-horizontal" role="form" data-toggle<

PHP实现DESede/ECB/PKCS5Padding加密算法兼容Java SHA1PRNG

这里写自定义目录标题 背景JAVA代码解决思路PHP解密 背景 公司PHP开发对接一个Java项目接口&#xff0c;接口返回数据有用DESede/ECB/PKCS5Padding加密&#xff0c;并且key也使用了SHA1PRNG加密了&#xff0c;网上找了各种办法都不能解密&#xff0c;耗了一两天的时间&#xf…

【Qt学习笔记】(三)常用控件(持续更新)

Qt 常用控件 1 控件概述2 QWidget 控件核心属性2.1 enabled2.2 geometry2.3 window frame 的影响2.4 windowTitle2.5 window Icon2.6 windowOpacity2.7 cursor2.8 font2.9 toolTip2.10 focusPolicy2.11 stylesheet 1 控件概述 Widget是Qt中的核心概念英文原义是"小部件&q…

netty-websocket扩展协议及token鉴权补充

文章源码&#xff1a;gitee 源码部分可以看上一篇文章中的源码分析netty-websocket 鉴权token及统一请求和响应头&#xff08;鉴权控制器&#xff09; 最近刚好没事&#xff0c;看到有朋友说自定义协议好搞&#xff0c;我就想了想&#xff0c;发现上面那种方式实现确实麻烦&…

假期刷题打卡--Day24

1、MT1198阶乘差 求1!-2!-3!-…-n! 格式 输入格式&#xff1a; 输入为整型 输出格式&#xff1a; 输出为整型 样例 1 输入&#xff1a; 5输出&#xff1a; -151 分析过程 看到这个题目的时候&#xff0c;感觉这个题目出现的没有必要&#xff0c;就和前面阶乘和一样的…

jbdc的简单了解

JDBC JDBC所处的位置 JDBC的本质 Java操作数据库的一套接口。 补充 ddl:数据库定义语言,例如建表,创建数据库等。 dml:数据库操作语言,例如增删改。 dql:数据库查询语言,例如查询语句。 注意 在创建Java项目后的第一个步骤是导入jar包。 导入jar包的步骤 1 创建l…

华为视频监控接入到视频监控平台 (华为网路监控摄像机IPC和华为视频节点设备VCN)

目 录 一、设备介绍 1.1 华为VCN介绍 1.2 AS-V1000视频监控平台介绍 1.3 平台服务器配置说明 二、安装、配置HW_IVS软件 2.1下载安装HW_IVS软件 2.2登录HW_IVS 2.3共享到外域 三、配置华为外域参数 3.1 PCG模块设置 3.2通信协议GBT28181配置 3.3传…

ChatGPT 4.0 升级指南, ChatGPT Plus(GPT 4.0) 有何优势?

1.ChatGPT 是什么&#xff1f; ChatGPT 是由 OpenAI 开发的一种基于人工智能的聊天机器人&#xff0c;它基于强大的语言处理模型 GPT&#xff08;Generative Pre-trained Transformer&#xff09;构建。它能够理解人类语言&#xff0c;可以为我们解决实际的问题。 ChatGPT 4.…

网络选择流程分析(首选网络类型切换流程)

首先是界面,我在此平台的界面如下: 对应的入口源码位置在Settings的UniEnabledNetworkModePreferenceController中,当然其他平台可能在PreferredNetworkModePreferenceController中,流程上都是大同小异 然后点击切换按钮会调用到UniEnabledNetworkModePreferenceControlle…

【第三十五节】idea项目的创建以及setting和Project Structure的设置

项目创建 Project Structure的设置 点击file ~ Project Structure 进入

Python脚本之操作Elasticsearch【一】

本文为博主原创&#xff0c;未经授权&#xff0c;严禁转载及使用。 本文链接&#xff1a;https://blog.csdn.net/zyooooxie/article/details/109588072 前面刚写了 requests发请求 操作Elasticsearch - Search https://blog.csdn.net/zyooooxie/article/details/123730279&…

从零开始手写mmo游戏从框架到爆炸(七)— 消息封装

上一篇&#xff0c;我们初步把消息handler 注册到了服务中&#xff0c;在进行后续工作之前我们需要再做一些准备工作。 第一&#xff1a;把之前自己管理的bean放到spring中去管理&#xff0c;后面大部分的bean都通过spring来管理。 第二&#xff1a;为了方便路由消费&#xff0…

Golang-Map有序输出——使用orderedmap库实现

前言 工作中遇到一个问题&#xff1a;需要导出一个MySQL表格&#xff0c;表格内容由sql查询得来。但现在发现&#xff0c;所导出的表格中&#xff0c;各列的顺序不确定。多次导出&#xff0c; 每一次的序列顺序也是不定的。 因此确定是后端&#xff0c;Map使用相关导致的问题。…

别具一格,质感拉满 | PITAKA苹果Apple Watch彩碳表带开箱

别具一格&#xff0c;质感拉满 | PITAKA苹果Apple Watch彩碳表带开箱 我是在前年的时候购买的目前手头这款Apple Watch Series7&#xff0c;因为是购买的Nike版&#xff0c;所以可以看到它的表带标配为透气孔的运动型表带。 &#x1f53a;耐克版的透气孔表带虽说在一定程度上解…

时序预测 | MATLAB实现基于CNN-LSTM-AdaBoost卷积长短期记忆网络结合AdaBoost时间序列预测

时序预测 | MATLAB实现基于CNN-LSTM-AdaBoost卷积长短期记忆网络结合AdaBoost时间序列预测 目录 时序预测 | MATLAB实现基于CNN-LSTM-AdaBoost卷积长短期记忆网络结合AdaBoost时间序列预测预测效果基本介绍模型描述程序设计参考资料 预测效果 基本介绍 1.MATLAB实现基于CNN-LST…

电机控制系列模块解析(第六篇)—— 观测器

最近有上传一些入门的免积分的资料&#xff0c;方便大家上手进行仿真分析。注意查收。还在继续更新中。 继续回到咱们的电机控制系列模块解析&#xff08;第六篇&#xff09;—— 观测器 1、无位置传感器控制背景 这方面的文献比较多&#xff0c;直接引用一些文献里的背景知…

SSRF:服务端请求伪造攻击

目录 什么是SSRF&#xff1f; 攻击内网应用 端口扫描 攻击非web应用 pikachu中的SSRF curl&#xff08;端口扫描&#xff09; file_get_content&#xff08;读取文件&#xff09; 防御SSRF 什么是SSRF&#xff1f; 服务端请求伪造&#xff08;Server Side Request For…

Python __file__属性:查看模块的源文件路径

除可以查看模块的帮助信息之外&#xff0c;还可以直接阅读模块的源代码来掌握模块功能&#xff0c;提升 Python 编程能力。 不管学习哪种编程语言&#xff0c;认真阅读那些优秀的框架、库的源代码都是非常好的学习方法。 通过模块的 __file__ 属性即可查看到指定模块的源文件…

实践:微服务版本升级步骤以及maven仓库相关概念

进行微服务开发的时候&#xff0c;上层服务依赖于下层的服务的api&#xff0c;比如适配属于上层服务&#xff0c;用户属于下层服务。 例子: 上层服务 <!--订单管理微服务api依赖--> <dependency><groupId>com.jn.server</groupId><artifactId>…

docker部署自己的网站wordpress

目录 安装 1.创建目录 2.创建并启动mysql 3.创建并启动wordpress 使用 1.设置语言 2.设置基础信息 3.首页 安装 1.创建目录 mkdir -p /opt/wordpress/{db,data} 2.创建并启动mysql docker run -d --name my_mysql --restart always -e MYSQL_ROOT_PASSWORD123456 -e …