C++:this指针和构造与析构的运用

news2024/11/25 20:24:22

目录

一,this指针

二,构造函数

三,析构函数

四,析构与构造的调用


一,this指针

        首先,我们先观察以下类:

#include <iostream>
using namespace std;
class Date
{
public:
    void Init(int year, int month, int day)
    {
        _year = year;
        _month = month;
        _day = day;
    }
    void Print()
    {
        cout << _year << "-" << _month << "-" << _day << endl;
    }
private:
    int _year;     
    int _month;    
    int _day;      
};
int main()
{
    Date d1, d2;
    d1.Init(2022, 1, 11);
    d2.Init(2022, 1, 12);
    d1.Print();
    d2.Print();
    return 0;
}

        对于上述类,有这样的一个问题: Date类中有 Init 与 Print 两个成员函数,函数体中没有关于不同对象的区分,那当d1调用 Init 函 数时,该函数是如何知道应该设置d1对象,而不是设置d2对象呢? C++中是通过引入this指针解决该问题,即:C++编译器给每个“非静态的成员函数“增加了一个隐藏的指针参数,让该指针指向当前对象(即此类),在函数体中所有“成员变量” 的操作,都是通过该指针去访问。只不过所有的操作对用户是透明的,即用户不需要来传递,编译器自动完成。如下图:

class Date
{
public:
    void Init(int year, int month, int day)
    {
        _year = year;
        _month = month;
        _day = day;
    }
    //void Print(Data* this),错误,不能显示的写实参和形参,里面默认就有this指针
    void Print()
    {
        //我们不能在形参中写,但是可以在类里面使用,因为默认有this指针
        cout << this->_year << "-" << this->_month << "-" << this->_day << endl;
        cout << _year << "-" << _month << "-" << _day << endl;
    }
private:
    int _year;     
    int _month;    
    int _day;      
};

这里需要说明以下几个问题:

        1. this指针的类型:类类型* const,即成员函数中,不能给this指针赋值。

        2. this指针只能在“成员函数”的内部使用。

        3. this指针本质上是“成员函数”的形参,当对象调用成员函数时,将对象地址作为实参传递给 this形参。所以对象中不存储this指针。

        4. this指针是“成员函数”第一个隐含的指针形参,而形参跟函数一样存储在栈区中,一般情况由编译器通过ecx寄存器自动传递,不需要用户传递。


二,构造函数

介绍:

        构造函数:在创建对象时,自动的进行初始化工作。语法:类名() {...}

        构造函数是一个特殊的成员函数,名字与类名相同,创建类类型对象时,由编译器自动调用,以保证每个数据成员都有一个合适的初始值,并且在对象整个生命周期内只调用一次。

特征:

        1,构造函数的访问权限一般情况下要设置为public,设置为private将无法访问构造函数,导致建立类时自动访问将会出错。

        2,函数名必须与类名相同。

        3,没有返回值,也不写void。

        4,可以有参数,可以重载,可以有默认参数。

        5,创建对象时会自动调用一次,不能手工调用。

这里需注意以下几点:

        1,如果通过无参构造函数创建对象时,对象后面不用跟括号,否则就成了函数声明。 

        2,构造函数的权限若设置为private将会将会出错,因为权限不可访问。      

        3,如果定义了许多类,构造函数将会按照先后顺序调用。      

默认构造函数:

        如果类中没有显式定义构造函数,则C++编译器会自动生成一个无参的默认构造函数,一旦用户显式定义编译器将不再生成,默认构造函数将会往对象中放入随机值。

        C++把类型分成内置类型(基本类型)和自定义类型。内置类型就是语言提供的数据类 型,如:int/char...,自定义类型就是我们使用class/struct/union等自己定义的类型,而编译器生成的默认构造函数只会对自定类型成员调用它的默认构造函数,内置类型一般不做处理,但有些编译器会对其进行处理(这里建议都看成不处理),即只处理自定义类型,为解决这一问题,在C++11标准中,内置类型成员变量在类中声明时可以给默认值。

        内置类型是编译器自己定义的,所以一般都会自动给它们初始化,但内置类型不知是如何实现的,所以不会处理,需要默认构造函数处理。

样例示范:                       

#include <iostream>
using namespace std;
class Date
{
public://设置为公有权限,否则相当于建立时将会出错
    // 1.无参构造函数
    Date()
    {}
    // 2.带参构造函数
    Date(int year, int month, int day)
    {
        _year = year;
        _month = month;
        _day = day;
    }
private:
    int _year;
    int _month;
    int _day;
};
int main()
{
    Date d1; // 调用无参构造函数
    Date d2(2015, 1, 1); // 调用带参的构造函数
    Date d3();//函数的声明,声明了d3函数,该函数无参,返回一个日期类型的对象
    return 0;
}


三,析构函数

析构函数:在销毁对象前,自动的完成清理工作,即在对象生命周期结束时将会清理空间资源。

特征:

        1,析构函数的函数名是在类名前加上一个符号 ‘~’。

        2,析构函数无参数且无返回值类型。

        3,一个类中只能有一个析构函数,若没有自己定义,系统会自动生成一个默认析构函数。

        4,在对象的生命周期结束时,系统将会自动调用析构函数。

注意:

        1,由于析构函数没有参数,因此此函数不支持重载。

        2,析构函数的权限设置跟构造函数的设置一样,一般都要设置为public。

        3,类的析构函数调用一般按照构造函数调用的相反顺序进行调用,即倒序,但要注意static和全局变量的情况,存储静态区的将会在整个代码结束后才会调用。

        4,默认析构函数跟默认构造函数一样,只对自定义类型成员处理,即自定义类型成员会去调用他的析构函数,而内置类型成员不做任何处理。

        5,析构函数只能释放该对象本身所占用的空间,但对象内部所指向的内存空间或其它空间没有被释放,最终将会造成内存泄漏。

        6,由第2条可知,如果类中没有申请资源时,析构函数可不写,直接使用编译器生成的默认析构函数即可。因为当对象的生命周期结束后空间就自动销毁了,也就还给操作系统了,但是如果在堆区中开辟了空间就需要在析构函数中自己完成清理工作。

样例示范一:

class Time
{
public:
    //首先权限设置必须为public,否则结束时将不可调用,将会出错
    //其次,类对象的成员中没有在堆区中占用空间,可不写析构函数

    ~Time()
    {
        cout << "~Time()" << endl;
    }
private:
    int _hour;
    int _minute;
    int _second;
};

样例示范二:

class Stack
{
public:
    Stack(size_t capacity = 3)
    {
        _array = (int*)malloc(sizeof(int) * capacity);
        _capacity = capacity;
        _size = 0;
    }
    void Push(int data)
    {
        _array[_size] = data;
        _size++;
    }
    //首先权限要设置为public
    //动态开辟空间,需要在析构函数中释放动态空间

    ~Stack()//析构函数中不能有参数
    {
        if (_array)
        {
            free(_array);
            _array = NULL;
            _capacity = 0;
            _size = 0;
        }
    }
private:
    int* _array;
    int _capacity;
    int _size;
};


四,析构与构造的调用

构造函数的调用:

        当一个程序开始定义类时,程序就自动进入了此类中的构造函数阶段,构造完了之后系统就会自动往下运行,当又遇到一个类时,又将进入此类中的构造函数,也就是说当存在多个类定义时,系统会按照从前往后的顺序调用此类的构造函数。

析构函数的调用:

        析构函数是在一个函数即将结束时才会开始调用,一般情况下,调用的顺序是跟析构函数的调用顺序相反,即倒序,但是,当存在类似于static类时的情况就很不一样,因为此种情况是存储在系统的静态空间中,而静态空间不会随着函数栈帧的销毁而销毁,只有当一整个程序完毕之后才会系统才会做回收。

        上图中的构造函数的调用顺序为C,A,B,D。

代码演示:

#include <iostream>
using namespace std;
int i = 0;
class Data2
{
public:
    Data2()
    {
        i++;
        cout << "2构造中的i = " << i << endl;
    }
    void Print()
    {
        i++;
        cout << "2类输出中的 i = " << i << endl;
    }
    ~Data2()
    {
        i++;
        cout << "2析构中的i = " << i << endl;
    }
};
class Data1
{
public:
    Data1()
    {
        i++;
        cout << "1构造中的i = " << i << endl;
    }
    void Print(Data2 p)//类传入,当此函数结束时,会调用Data2类中的析构,即销毁形参中的类
    {
        i++;
        cout << "1类输出中的 i = " << i << endl;
    }
    ~Data1()
    {
        i++;
        cout << "1析构中的i = " << i << endl;
    }
};
int main()
{
    Data1 a;//调用Data1中的构造
    Data2 b;//调用Data2中的构造
    a.Print(b);
    cout << endl;
    //当系统运算到此步时就要开始调用析构函数了,在析构函数中,先析构Data2类型,再析构Data1的类型
    return 0;
}

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

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

相关文章

二十、W5100S/W5500+RP2040树莓派Pico<MQTT连接阿里云控制板载LED>

1. 前言 物联网平台提供安全可靠的设备连接通信能力&#xff0c;支持设备数据采集上云&#xff0c;规则引擎流转数据和云端数据下发设备端。此外&#xff0c;也提供方便快捷的设备管理能力&#xff0c;支持物模型定义&#xff0c;数据结构化存储&#xff0c;和远程调试、监控、…

mindspore mindcv图像分类算法;模型保存与加载

参考&#xff1a; https://www.mindspore.cn/tutorials/en/r1.3/save_load_model.html https://github.com/mindspore-lab/mindcv/blob/main/docs/zh/tutorials/finetune.md 1、mindspore mindcv图像分类算法 import os from mindcv.utils.download import DownLoad import o…

举个栗子!Tableau 技巧(259):文本表中省市县数据的灵活逐级下钻「方法一」

之前&#xff0c;我们分享过 &#x1f330;&#xff1a;实现地图中的省市县逐级下钻。有数据粉提出问题&#xff1a;如果不是地图&#xff0c;而是文本表&#xff0c;有什么办法可以像这样&#xff0c;实现地理位置逐级下钻呢&#xff1f; 文本表也是可以的。但是&#xff0c;…

Count-based exploration with neural density models论文笔记

Count-based exploration with neural density models[J]. International Conference on Machine Learning,International Conference on Machine Learning, 2017. 基于计数的神经密度模型探索 0、问题 这篇文章的关键在于弄懂pseudo-count的概念&#xff0c;以及是如何运用…

【Leetcode】202. 两数之和

给定一个整数数组 nums 和一个整数目标值 target&#xff0c;请你在该数组中找出 和为目标值 target 的那 两个 整数&#xff0c;并返回它们的数组下标。 你可以假设每种输入只会对应一个答案。但是&#xff0c;数组中同一个元素在答案里不能重复出现。 你可以按任意顺序返回…

Java 身份证号校验,根据身份证号识别出生地

Java 身份证号校验&#xff1a; import org.apache.commons.lang.StringUtils;import java.util.Calendar; import java.util.Collections; import java.util.HashMap; import java.util.Map;/*** desc 身份证工具类* auth llp* date 2022/7/7 16:13*/ public class IdCardNum…

Java算法(三): 判断两个数组是否为相等 → (要求:长度、顺序、元素)相等

Java算法&#xff08;三&#xff09; 需求&#xff1a; 1. 定义一个方法&#xff0c;用于比较两个数组是否相同2. 需求&#xff1a;长度&#xff0c;内容&#xff0c;顺序完全相同package com.liujintao.compare;public class SameArray {public static void main (String[] a…

JAVA微信端医院3D智能导诊系统源码

医院智能导诊系统利用高科技的信息化手段&#xff0c;优化就医流程。让广大患者有序、轻松就医&#xff0c;提升医疗服务水平。 随着人工智能技术的快速发展&#xff0c;语音识别与自然语言理解技术的成熟应用&#xff0c;基于人工智能的智能导诊导医逐渐出现在患者的生活视角中…

小红书达人投放比例是多少合适?品牌方必看

品牌做小红书种草推广想要产生更好的效果&#xff0c;是需要素人和达人按照一定比例去进行投放的&#xff0c;素人铺量可以让产品产生迅速曝光的效果&#xff0c;少量达人投放可以让产品产生更好的转化效果。 小红书达人投放具有较高的互动性和口碑传播效果。达人通过自身的影…

打开pr提示找不到vcomp100.dll无法继续执行代码怎么办?5种dll问题解决方案全解析

vcomp100.dll是一个由Microsoft开发的动态链接库&#xff08;DLL&#xff09;文件&#xff0c;它对于许多基于图形的应用程序&#xff08;如Photoshop&#xff09;和多个游戏&#xff08;如《巫师3》&#xff09;至关重要。以下是关于vcomp100.dll的属性介绍以及找不到vcomp100…

小程序如何部署SSL证书

微信小程序开发前提必须拥有一本SSL证书&#xff0c;办理SSL证书之前确保好指定的微信小程序开发接口使用的域名&#xff0c;如果没有域名的提前申请好&#xff0c;并且到国内服务器提供商去办理备案。 了解微信小程序使用SSL证书的作用&#xff0c;包括以下三个方面&#xff1…

[C语言基础]文件读取模式简析

文件操作 打开方式介绍r / rb模式w / wb模式 打开方式介绍 函数fopen可打开一个文件&#xff0c;返回值是文件指针FILE * 第一个参数是文件路径&#xff0c;第二个参数是打开方式mode 参数可为以下几种&#xff1a; r/w/a/r/w/a/rb/wb/ab/rb/wb/ab 其中&#xff0c; r 为只读&…

求臻医学MRD产品喜获北京市新技术新产品(服务)证书

近日&#xff0c;北京市科学技术委员会、中关村科技园区管理委员会、北京市发展和改革委员会等五大部门联合公示了2023年度第一批&#xff08;总第十八批&#xff09;北京市新技术新产品&#xff08;服务&#xff09;名单。凭借领先的技术能力、产品创新能力及质量可靠性等优势…

大数据毕业设计选题推荐-河长制大数据监测平台-Hadoop-Spark-Hive

✨作者主页&#xff1a;IT研究室✨ 个人简介&#xff1a;曾从事计算机专业培训教学&#xff0c;擅长Java、Python、微信小程序、Golang、安卓Android等项目实战。接项目定制开发、代码讲解、答辩教学、文档编写、降重等。 ☑文末获取源码☑ 精彩专栏推荐⬇⬇⬇ Java项目 Python…

VM虚拟机安装

想编译一个 c 代码 windows 转成 linux 安装一个vm 准备一个虚拟机安装包&#xff0c;双击&#xff0c;开始安装 下一步 缸盖安装位置路径&#xff0c;添加PATH&#xff0c;下一步 下一步 添加到桌面&#xff0c;加入开始菜单&#xff0c;下一步 打开桌面的软件图标&#…

Panorama SCADA平台的警报通知功能配置详解

1. 前言 SCADA系统的主要目标是采集与监控工业过程数据&#xff0c;以确保工业生产正常运行。通过实时警报通知功能&#xff0c;操作人员可以立即获取有关潜在问题的信息&#xff0c;从而能够快速采取行动解决问题&#xff0c;防止进一步的损害或生产中断。因此&#xff0c;及…

小程序版本审核未通过,需在开发者后台「版本管理—提交审核——小程序订单中心path」设置订单中心页path,请设置后再提交代码审核

小程序版本审核未通过&#xff0c;需在开发者后台「版本管理—提交审核——小程序订单中心path」设置订单中心页path&#xff0c;请设置后再提交代码审核 因小程序尚未发布&#xff0c;订单中心不能正常打开查看&#xff0c;请先发布小程序后再提交订单中心PATH申请 初次提交…

03【远程协作开发、TortoiseGit、IDEA绑定Git插件的使用】

上一篇&#xff1a;02【Git分支的使用、Git回退、还原】 下一篇&#xff1a;【已完结】 目录&#xff1a;【Git系列教程-目录大纲】 文章目录 一、远程协作开发1.1 远程仓库简介1.1.1 Github1.1.2 Gitee1.1.3 其他托管平台 1.2 发布远程仓库1.2.1 创建项目1&#xff09; 新…

deeplog中输出某个 event 的概率

1 实现之后效果 # import DeepLog and Preprocessor import numpy as np from deeplog import DeepLog import torch# Create DeepLog object deeplog DeepLog(input_size 10, # Number of different events to expecthidden_size 64 , # Hidden dimension, we suggest 64…

K8s----资源管理

目录 一、Secret 1、创建 Secret 1.1 用kubectl create secret命令创建Secret 1.2 内容用 base64 编码&#xff0c;创建Secret 2、使用方式 2.1 将 Secret 挂载到 Volume 中&#xff0c;以 Volume 的形式挂载到 Pod 的某个目录下 2.2 将 Secret 导出到环境变量中 二、Co…