C++类中六个默认的成员函数详解

作者:_Camille 时间:2022-11-25 10:46:30 

浅谈

先来说一下“this指针”:

C++中通过引入this指针解决该问题,暨:C++编译器给每个“非静态的成员函数”增加了一个隐藏的指针参数,让该指针指向当前对象(函数运行时调用该函数的对象),在函数体中所有成员变量的操作,都是通过该指针去访问,只不过所有的操作对用户是透明的,暨用户不需要来传递,编译器自动完成。

说了这么多其实编译器在生成程序时获取对象首地址的信息。然后将获取的对象的首地址存放在了寄存器中,成员函数的其它参数都是存放在栈中。而this指针参数则是存放在寄存器中。类的静态成员函数(用static修饰的成员函数)因为没有this指针这个参数,所以类的静态成员函数也就无法调用类的非静态成员变量。

构造函数

构造函数是一个特殊的成员函数,名字与类名相同且不能有返回值,创建类类型时由编译器自动调用,在对象的生命周期内只调用一次。**主要任务是初始化对象。

↓下面是一个简单的构造函数(全缺省):

C++类中六个默认的成员函数详解

主函数初始化时如果无参则以缺省值0给成员变量赋值。

默认构造函数:

C++类中六个默认的成员函数详解

C++类中六个默认的成员函数详解

C++类中六个默认的成员函数详解

Q:为什么会出现上面的报错——包含多个默认构造函数?

A:无参的构造函数和全缺省的构造函数都称为默认构造函数,并且默认构造函数只能有一个。注意:无参构造函数,全缺省构造函数,我们没写编译器默认生成的构造函数,都可以称为默认构造函数。

特征:

1.函数名与类名相同;

2.无返回值;

3.对象实例化时编译器自动调用对应的构造函数;

4.构造函数可以重载。

析构函数

析构函数:与构造函数功能相反,析构函数是完成对象的销毁,局部对象销毁工作是由编译器完成的。而对象在销毁时会自动调用析构函数,完成类的一些资源清理工作,和构造函数一样,如果我们没写析构函数,系统会生成一个默认析构函数,但这个析构函数什么都不会做。

如果类中的成员变量不需要动态开辟内存空间,则默认析构函数可以完成析构任务,比如下面这种,可以说不用析构。

C++类中六个默认的成员函数详解

但是像下面这种,默认析构函数已经不能够完成

C++类中六个默认的成员函数详解

特征:

1.函数名是在类名前加上字符~;

2.无参数(有一个隐藏参数*this指针)无返回值;

3.一个类有且仅有一个析构函数,若未显示定义,系统会自动生成默认的析构函数;

4.对象生命周期结束时,C++编译系统自动调用析构函数。

拷贝构造函数

拷贝构造函数:只有单个形参,该形参是对本类类型对象的引用(一般常用const修饰),在用已存在的类类型对象创建新对象时由编译器自动调用

C++类中六个默认的成员函数详解

从上图我们可以看出,关于给t2变量初始化时肯定不是调用构造函数。

下面这张图应该就可以解释上面的问题

C++类中六个默认的成员函数详解

关于对拷贝构造函数参数的说明:

C++类中六个默认的成员函数详解

当然也同构造函数一样,若未显示定义,系统生成默认的拷贝构造函数。默认的拷贝构造函数对象按内存存储按字节序完成拷贝,暨浅拷贝或值拷贝

特征:

1.拷贝构造函数是构造函数的一个重载形式;

2.拷贝构造函数的参数只有一个(当然还有个隐藏的*this)且必须使用引用传参,使用传值方式会引发死递归。

赋值重载函数

赋值重载函数:C++为了增强代码的可读性引入了运算符重载,运算符重载是具有特殊函数名的函数,也具有其返回值类型,函数名字以及参数列表,其返回值类型与普通的函数类似。

**函数名字为:**关键字operator后面接需要重载的运算符符号;

函数原型:返回值类型operator操作符(参数列表)

Test& operator= (const Test& t)

C++类中六个默认的成员函数详解

C++类中六个默认的成员函数详解

C++类中六个默认的成员函数详解

C++类中六个默认的成员函数详解

注意:

不能通过连接其他符号来创建新的操作符:比如operator@;

重载操作符必须有一个类类型或者枚举类型的操作数;

用于内置类型的操作符的操作符,其含义不能改变,例如:内置的整型+,不能改变其含义;

作为类成员的重载函数时,其形参看起来比操作数数目少1成员函数的,操作符有一个默认的形参this,限定为第一个形参;

*、::、sizeof、?:、. 以上5个运算符不能重载。

一个类如果没有显示定义赋值运算符重载,编译器也会生成一个,完成对象按字节序的值拷贝。

const成员函数

将const修饰的类成员函数称之为const成员函数,const修饰类成员函数,实际修饰该成员函数隐含的this指针,表明该成员函数中不能对类的任何成员进行修改。

C++类中六个默认的成员函数详解

multable关键字:当有必须要修改的成员变量时,需在成员变量声明时前加上该关键字,及时是const成员函数也依然可以修改。

取地址及const取地址重载函数

这两个默认成员函数一般不用重新定义,编译器默认生成;


class Date
{
public:
Date* operator&()
{
return this;
}
//因为对象被const修饰不能更改,所以返回值也要被const修饰
const Date* operator&() const
{
return this;
}
private:
int _year;
int _month;
int _day;
};

深挖

 构造函数

C++类中六个默认的成员函数详解

上面的代码居然可以正常运行并且赋值成功,这是为什么。实际上编译器在执行赋值语句前,将“100作为参数来构造一个无名的临时的类”然后进行赋值。

C++类中六个默认的成员函数详解

上面这种情况我们称为**“隐式转换”**。

explicit关键字:如果在构造函数前加上这个关键字,则要求显示转换,不能进行隐式转换。

C++类中六个默认的成员函数详解

因此我们进行强转之后再赋值;

C++类中六个默认的成员函数详解

我们换一种思路,将“类”给整型赋值,且先给出一个强转函数;

C++类中六个默认的成员函数详解

最后介绍另外一种初始化的方式:

C++类中六个默认的成员函数详解

给出一个复数类,参数列表初始化的效率要高于第一种初始化方式。

拷贝构造函数

下面介绍一下深拷贝,先看下面一段代码:

C++类中六个默认的成员函数详解

这个代码中我们没有写拷贝构造函数,可见默认拷贝函数这里出了问题;前面我们说过,默认拷贝方式是浅拷贝,也就是值拷贝。

上面s1使用hello初始化,s2使用s1的值来拷贝,因此s2并未开辟新空间,而是指向s1的空间,因此在最后调用析构函数的时候,对同一块空间free了两次。

如果将代码改成下面这种,那么就不会报错。

C++类中六个默认的成员函数详解

赋值运算符重载函数

这里我们继续引用上面构造函数部分最后给出的那个复数类

C++类中六个默认的成员函数详解

我们上面说了赋值重载函数,那么到底什么是赋值运算符重载呢?

运算符重载:对运算符赋予新的意义↓↓↓

C++类中六个默认的成员函数详解

由上图可知,上面这个对加法运算符符号“+”的重载函数是成员函数,因为它参数里面有一个隐藏的的this指针,所以需要用对象来调动它。

C++类中六个默认的成员函数详解

上面的代码整体的思路就是:

1.调用四次次构造函数,构造C1C2C3以及在operator+内部的tmp;

2.C1调用加法重载函数进行C1+C2;

3.加法重载函数返回时需要调动一次拷贝构造函数(tmp拷贝到临时对象);

4.调用赋值重载将C1+C2的值赋给C。

C++类中六个默认的成员函数详解

当写的类想做某种运算时,但编译器不支持,因此我们需要对运算符进行重载。

来源:https://blog.csdn.net/qq_43560037/article/details/114981742

标签:c++,类,成员函数
0
投稿

猜你喜欢

  • c#中XML解析文件出错解决方法

    2022-01-21 00:38:50
  • Java 图表类库详解

    2021-11-09 00:25:11
  • 浅谈BeanPostProcessor加载次序及其对Bean造成的影响分析

    2022-05-02 19:52:29
  • C#8.0新语法using declaration

    2023-10-23 00:57:54
  • 阿里、华为、腾讯Java技术面试题精选

    2023-11-25 02:29:39
  • Android车载多媒体开发MediaSession框架示例详解

    2022-08-10 19:49:20
  • 在Eclipse中运行Solr 基础知识

    2021-07-06 22:51:04
  • SpringCloud Eureka的使用教程

    2022-03-23 22:30:59
  • 一篇文章告诉你JAVA Mybatis框架的核心原理到底有多重要

    2023-11-13 06:20:10
  • android仿微信表情雨下落效果的实现方法

    2023-07-15 13:44:14
  • Java常用HASH算法总结【经典实例】

    2023-04-26 00:34:55
  • Android中Blade的使用方法

    2023-04-25 11:29:54
  • 如何设置Spring Boot测试时的日志级别

    2023-11-10 14:11:20
  • winform 调用摄像头扫码识别二维码的实现步骤

    2022-08-08 21:50:20
  • 一步步实现Viewpager卡片翻页效果

    2022-10-15 02:14:25
  • 如何解决通过spring-boot-maven-plugin package失败问题

    2021-10-16 15:43:45
  • Java InheritableThreadLocal用法详细介绍

    2023-01-27 04:30:11
  • android几种不同对话框的实现方式

    2022-09-13 06:31:42
  • SpringBoot实用小技巧之如何动态设置日志级别

    2023-02-09 04:27:58
  • 深入理解Kotlin的泛型系统

    2023-09-09 00:47:30
  • asp之家 软件编程 m.aspxhome.com