C++类与对象(补)

 

感谢大佬的光临各位,希望和大家一起进步,望得到你的三连,互三支持,一起进步

个人主页:LaNzikinh-CSDN博客

文章目录

  • 前言
  • 一.默认成员函数
  • 二.static
  • 三.友元
  • 四.匿名对象
  • 总结

前言

类的默认成员函数,默认成员函数就是用户没有显式实现,编译器会自动生成的成员函数称为默认成员函数。一个类,我 们不写的情况下编译器会默认生成以下6个默认成员函数。


一.默认成员函数

1.1构造函数

因为前面对构造函数进行过了讲解,这里只做补充

注意:构造函数就是替代了以前C语言中的初始化,对象实例化时系统会自动调用对应的构造函数,构造函数是可以重载的

默认构造函数概念:1.无参构造函数,2.全缺省的,3.不写构造时编译器默认生成的。这三个函数有且只能有一个存在,总结:不传实参数就可以调用的构造叫做默认构造

对于编译器默认生成的构造,对内置类型成员变量初始化没有要求,但对于自定义类型成员变量,会调用这个成员变量的默认构造函数,如果这个成员变量没有构造函数就会报错,这时候就用初始化成员列表。

那么什么是内置类型和自定类型呢?C++把类型分为两类,内置类型和自定义类型,内置类型就是int,char,float,指针这种,自定义类型就是类,结构体,class,struct

这句话不是很好理解,我们来举个例子,方便理解一下

例子:

这段代码运行来是没有任何问题的,但是如果修改就会有问题

#include<iostream>
using namespace std;
class Date
{
public:Date(){_year = 1;_month = 1;_day = 1;}Date(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;Date d2(2025, 1, 1); Date d3();return 0;
}

我们如果把第二个构造函数给注释调的话,那么程序就会出错,因为我第二个对象创造的时候没有构造函数给他调用,所以就会报错

同理,如果我把第一个注释掉的话也会报错,因为第一个和第三个没有对应的构造函数

那如果我把两个构造函数全部注释掉可以吗?系统不是有一个默认的吗?系统会帮我们生成一个没有参数的无参构造函数,那么如果想要程序运行成功的话,就要把第二个对象也注释

那么默认的构造函数到底什么时候用呢?或者是说我什么时候要写构造函数什么时候不写呢?我们之前学过一个用栈来构造队列,如果把它用C++来写的话

class MyQueue
{
public:private:Stack pushst;Stack popst;
};

编译器默认生成MyQueue的构造函数调用了Stack的构造,完成了两个成员的初始化

如果我的成员对象是有构造函数的时候,他会先调用成员对象的构造函数,这种情况下如果帮我初始化了的话,我就不需要初始化了,我就不需要调用了,总结:大多数情况下,构造函数都需要我们自己来实现,少数情况下不需要

1.2析构函数

析构函数不想做太多的补充,跟之前一样,析构函数对应着就是C语言中的销毁,注意:自定义类型的成员也会调用它的析构,也就是说自定义类型成员无论什么情况都会自动调用他的析构函数。

1.3拷贝构造函数

拷贝构造函数是类与对象中的一个难点,拷贝构造函数是构造函数的一个重载,他有一个特别的地方,就是第一个参数必须是自身类型的引用,不然会引发无穷递归调用的报错

C++语法规定,传值传参会掉用拷贝构造函数。

我们之前讲过引用的本质是常量指针,所以说,如果不是引用的话,他就会无限递归

若未显示定义拷贝构造函数,编译器会自动生成一个拷贝构造函数,自动生成的拷贝构造对内置类型成员变量会完成浅拷贝,对自定义类型的成员变量会调用它的拷贝构造函数

这个地方和前面的有所差距,比如说我们再用那个站来实现队列的例子,这个地方就不能用系统自己的拷贝构造了

int main()
{Stack st1;st1.Push(1);st1.Push(2);// Stack不显示实现拷贝构造,用自动生成的拷贝构造完成浅拷贝// 会导致st1和st2里面的_a指针指向同⼀块资源,析构时会析构两次,程序崩溃Stack st2 = st1;MyQueue mq1;// MyQueue自动生成的拷贝构造,会自动调用Stack拷贝构造完成pushst/popst// 的拷贝,只要Stack拷贝构造自己实现了深拷贝,他就没问题MyQueue mq2 = mq1;return 0;
}

这里我们在前面的类与对象中讲了浅拷贝和深拷贝的问题,在这个地方就不多做解释了。

Date(const Date& d)
{
_year = d._year;
_month = d._month;
_day = d._day;
}

1.4赋值运算符重载

赋值运算符重载就是把一个对象复制给另一个对象。拷贝能分两种,一种是赋值重载拷贝,一种是拷贝构造,那到底怎么区分呢?赋值重载就是两个都存在的对象进行赋值,而拷贝构造是把初始化的对象去赋给没有初始化的对象。

int main()
{Date d1(2024, 7, 5);Date d2(2024, 7, 6);//赋值重载拷贝operator==(d1, d2);d1 == d2;//拷贝构造Date d3(d2);Date d4 = d2;return 0;
}

C++规定类类型对象使用运算符时,必须转换成调用对应运算符重载,若没有对应的运算符重载,则会编译报错。运算符重载是具有特名字的函数,他的名字是由operator和后⾯要定义的运算符共同构成。和其他函数一样,它也具有其返回类型和参数列表以及函数体。

bool operator==(const Date& d1, const Date& d2)
{
return d1._year == d2._year
&& d1._month == d2._month
&& d1._day == d2._day;
}

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

重载++运算符时,有前置++和后置++,运算符重载函数名都是operator++,无法很好的区分。
C++规定,后置++重载时,增加一个int形参,跟前置++构成函数重载,方便区分。

重载<<和>>时,需要重载为全局函数,因为重载为成员函数,this指针默认抢占了第一个个形参位置,第一个形参位置是左侧运算对象,调用时就变成了 对象<<cout,不符合使用习惯和可读性。重载为全局函数把ostream/istream放到第一个形参位置就可以了,第二个形参位置当类类型对象。

这个地方不做过多的讲解,到时候会专门来出一个日期类的实现,就专门运用这个赋值运算符重载来实现的。

1.5取地址运算符重载

将const修饰的成员函数称之为const成员函数,const修饰成员函数放到成员函数参数列表的后面。 • const实际修饰该成员函数隐含的this指针,表明在该成员函数中不能对类的任何成员进行修改。const修饰Date类的Print成员函数,Print隐含的this指针由 Date* const this 变为 const Date* const this

class Date
{public:Date(int year = 1, int month = 1, int day = 1){_year = year;_month = month;_day = day;}// void Print(const Date* const this) constvoid Print() const{cout << _year << "-" << _month << "-" << _day << endl;}
private:int _year;int _month;int _day;
};
int main()
{//这里非const也可以调用const成员函数这是⼀种权限的缩⼩Date d1(2024, 7, 5);d1.Print();const Date d2(2024, 8, 5);d2.Print();return 0;
}

1.6*取地址运算符重载

取地址运算符重载分为普通取地址运算符重载和const取地址运算符重载,⼀般这两个函数编译器自动生成的就可以够我们用了,不需要去显示实现。除非⼀些很特殊的场景,逼如我们不想让别人取到当 前类对象的地址,就可以自己实现⼀份,胡乱返回⼀个地址。

class Date
{
public:Date* operator&(){return this;// return nullptr;}const Date * operator&()const{return this;// return nullptr;}
private:int _year;int _month; int _day; 
};

二.static

用static修饰的成员变量,称之为静态成员变量,静态成员变量⼀定要在类外进行初始化,因为不存在对象中,存放在静态区用static修饰的成员函数,称之为静态成员函数,静态成员函数没有this指针,静态成员也是类的成员,受public、protected、private 访问限定符的限制。

class A
{static int GetACount()
{return _scount;
}
private:// 类⾥⾯声明static int _scount;
};
// 类外⾯初始化
int A::_scount = 0;
int main()
{return 0;}

例子:求1+2+3+...+n_牛客题霸_牛客网 (nowcoder.com)

描述

求1+2+3+...+n,要求不能使用乘除法、for、while、if、else、switch、case等关键字及条件判断语句(A?B:C)。

数据范围: 0<𝑛≤2000<n≤200
进阶: 空间复杂度 𝑂(1)O(1) ,时间复杂度 𝑂(𝑛)O(n)

这个题目就和静态成员变量有很大的关系了,如果不用静态成员变量的话,将很复杂,因为这个题目不能用循环这些语句来做,我们可以用C + +的构造函数的特性先创造一个类,专门用来求和,然后在类中初始化两个静态成员变量,一个初始化1,一个初始化0为什么要用到静态成员变量呢?因为它是一个累加的过程,我只有用到静态成员变量才不会出了,函数就销毁,就直接可以一直保存下来。在利用数组反复的调用构造函数完成题目,这里用了变长数组也可以用new来动态开辟。

class Sum
{
public:
Sum()
{
_ret += _i;
++_i;
}
static int GetRet()
{
return _ret;
}
private:
static int _i;
static int _ret;
};
int Sum::_i = 1;
int Sum::_ret = 0;
class Solution {
public:
int Sum_Solution(int n) {
// 变⻓数组
Sum arr[n];
return Sum::GetRet();
}
};

有static成员变量就肯定有static成员函数

1.静态成员函数中可以访问其他的静态成员,但是不能访问非静态的,因为没有this指针。2.非静态的成员函数,可以访问任意的静态成员变量和静态成员函数。 3.突破类域就可以访问静态成员,可以通过类名::静态成员 或者 对象.静态成员来访问静态成员变量和静态成员函数。

#include<iostream>
using namespace std;
class A
{
public:static int GetACount(){return _scount;}
private:static int _scount;
};
int A::_scount = 0;
int main()
{A a1;cout << A::GetACount() << endl;cout << a1.GetACount() << endl;return 0;
}

三.友元

如果我们需要访问内里的私有成员,该如何访问呢?友元提供了⼀种突破类访问限定符封装的方式,友元分为:友元函数和友元类,在函数声明或者类 声明的前面加friend,并且把友元声明放到⼀个类的里面。

class A
{// 友元声明friend void func(const A& aa);
private:int _a1 = 1;
};void func(const A& aa)
{cout << aa._a1 << endl;
}
int main()
{A aa;func(aa);return 0;
}

除了友元函数,还有友元类,友元类中的成员函数都可以是另⼀个类的友元函数,都可以访问另⼀个类中的私有和保护成员。友元类的关系是单向的,不具有交换性,比如A类是B类的友元,但是B类不是A类的友元。友元类关系不能传递,如果A是B的友元, B是C的友元,但是A不是B的友元。

class A
{// 友元声明friend class B;
private:int _a1 = 1;
};
class B
{
public:void func1(const A& aa){cout << aa._a1 << endl;cout << _b1 << endl;}
private:int _b1 = 3;
};
int main()
{A aa;B bb;bb.func1(aa);return 0;
}

内部类

如果⼀个类定义在另一个类的内部,这个内部类就叫做内部类。内部类是一个独立的类,跟定义在 全局相比,他只是受外部类类域限制和访问限定符限制,内部类默认是外部类的友元类

class A
{
private:static int _k;int _h = 1;
public:class B // B默认就是A的友元{public:void foo(const A & a){cout << _k << endl; cout << a._h << endl; }};
};
int A::_k = 1;

A是B的外步类,B默认就是A的友元,B是A的友元,B可以访问到A的私有,A不可以访问B的

四.匿名对象

匿名对象生命周期只在当前一行,一般临时定义一个对象当前用一下即可,就可以定义匿名对象,可以说明明对象就是一个一次性产品,比如有些时候我只需要用一次,就没有必要再创造一个类出来

class A
{private:int a = 2;int b = 1;
public:void print(){cout << print << endl;}
};
int main()
{A().print();return 0;
}

总结

这次是对之前的类与对象的一个补充,完善了一下类与对象的实现。

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

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

相关文章

linux centos limits.conf 修改错误,无法登陆问题修复 centos7.9

一、问题描述 由于修改/etc/security/limits.conf这个文件中的值不当&#xff0c;重启后会导致其账户无法远程登录&#xff0c;本机登录。 如改成这样《错误示范》&#xff1a; 会出现&#xff1a; 二、解决 现在知道是由于修改limits.conf文件不当造成的&#xff0c;那么就…

Android-- 集成谷歌地图

引言 项目需求需要在谷歌地图&#xff1a; 地图展示&#xff0c;设备点聚合&#xff0c;设备站点&#xff0c;绘制点和区域等功能。 我只针对我涉及到的技术做一下总结&#xff0c;希望能帮到开始接触谷歌地图的伙伴们。 集成步骤 1、在项目的modle的build.gradle中添加依赖如…

p19 C语言操作符详解

算术操作符 1.除了%操作符之外&#xff0c;其他的几个操作符可以作用于整数和浮点数。 2.对于/操作符如果两个操作数都为整数&#xff0c;执行整数除法。而只要有浮点数值型的就是浮点除法。 3.%操作符的两个操作数必须为 整数。返回的是整除之后的余数。 #include<std…

计算机毕业设计-基于Springboot的养老院管理系统-源码程序文档

项目源码&#xff0c;请关注❥点赞收藏并私信博主&#xff0c;谢谢~ 本系统开发采用技术为JSP、Bootstrap、Ajax、SSM、Java、Tomcat、Maven 此文章为本人亲自指导加编写&#xff0c;禁止任何人抄袭以及各类盈利性传播&#xff0c; 相关的代码部署论文ppt代码讲解答辩指导文件…

SpringBoot以及swagger的基本使用

1、SpringBoot是什么&#xff1f; 一种快速开发、启动Spring的框架、脚手架 遵循“约定优于配置”的思想&#xff0c;使得能够快速创建和配置Spring应用 2、SpringBoot的核心特性 自动配置&#xff0c;一些依赖、默认配置都预设好了&#xff0c;减少了配置量起步依赖&#x…

表格竖向展示

最近在做手机端web页面&#xff0c;页面中需要有个表格来显示数据&#xff0c;但是由于数据太多页面太窄&#xff0c;table展示横向滑动的话感觉很丑。所以让表格竖向显示了 具体页面如下: 实现代码&#xff1a;当然代码里面绑定的数据啊什么的你都可以修改为自己的内容&#…

【文献阅读】Social Bot Detection Based on Window Strategy

Abstract 机器人发帖的目的是在不同时期宣传不同的内容&#xff0c;其发帖经常会出现异常的兴趣变化、而人类发帖的目的是表达兴趣爱好和日常生活&#xff0c;其兴趣变化相对稳定。提出了一种基于窗口策略&#xff08;BotWindow Strategy&#xff09;的社交机器人检测模型基于…

【fastadmin】selectpickers多选提交后数据库只保存了一个选择值

问题描述 在 fastadmin 框架开发项目中&#xff0c;用到了selectpickers多选组件。 例如&#xff1a;选择了两位员工&#xff0c;但是提交后数据库只保存了一个选择值。 <div class"form-group"><label class"control-label col-xs-12 col-sm-2"…

Puppeteer动态代理实战:提升数据抓取效率

引言 Puppeteer是由Google Chrome团队开发的一个Node.js库&#xff0c;用于控制Chrome或Chromium浏览器。它提供了高级API&#xff0c;可以进行网页自动化操作&#xff0c;包括导航、屏幕截图、生成PDF、捕获网络活动等。在本文中&#xff0c;我们将重点介绍如何使用Puppeteer…

【深度学习】InternVL2-8B,图转文,docker部署

文章目录 基础fastapi服务请求fastapi接口 基础 https://huggingface.co/OpenGVLab/InternVL2-8B#%E7%AE%80%E4%BB%8B InternVL2-26B应该更好&#xff0c;但显存没那么大&#xff0c;只能跑InternVL2-8B了。 下载&#xff1a; cd /ssd/xiedong/InternVL2-26B git clone htt…

unity3d脚本使用start,updata,awake

最近学了一下unity&#xff0c;脚本编写用的c#&#xff0c;虽说没学过c#但是勉强根据教学还能写点代码。 在这里我来记录一下在我学习过程中感觉最重要的东西 消息函数&#xff1a; 在我们创建一个脚本文件的时候&#xff0c;我们首先可以看到两个默认给你写出来的函数。 这两…

RAG技术-为自然语言处理注入新动力

引言&#xff1a; 在自然语言处理&#xff08;NLP&#xff09;的领域中&#xff0c;RAG&#xff08;Retrieval-Augmented Generation&#xff09;技术以其独特的方式&#xff0c;正在改变我们与机器的交互方式。RAG技术结合了大语言模型的强大能力&#xff0c;使得机器在理解和…

MongoDB教程(九):java集成mongoDB

&#x1f49d;&#x1f49d;&#x1f49d;首先&#xff0c;欢迎各位来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里不仅可以有所收获&#xff0c;同时也能感受到一份轻松欢乐的氛围&#xff0c;祝你生活愉快&#xff01; 文章目录 引言一、环境准…

SAPUI5基础知识15 - 理解控件的本质

1. 背景 经过一系列的练习&#xff0c;通过不同的SAPUI5控件&#xff0c;我们完成了对应用程序界面的初步设计&#xff0c;在本篇博客中&#xff0c;让我们一起总结下SAPUI5控件的相关知识点&#xff0c;更深入地理解SAPUI5控件的本质。 通常而言&#xff0c;一个典型UI5应用…

RocketMQ单结点安装/Dashboard安装

目录 1.安装NameServer 2.安装Broker 3.使用自带工具测试数据发送 4.使用DashBoard进行查看 5.关闭相关设备 前置条件&#xff1a;两台虚拟机CentOS Linux release 7.5.1804(ps:当然也可以都部署在一台机器上) RocketMq属于天生集群。需要同时启动nameServer和Broker进行…

CMMI认证是什么?做CMMI认证的必要条件?CMMI认证的重要性

CMMI&#xff08;Capability Maturity Model Integration&#xff09;认证&#xff0c;作为企业追求卓越软件工程实践的标志&#xff0c;其实现过程既严谨又复杂。首先&#xff0c;我们需要深入理解CMMI的核心理念&#xff0c;即持续的过程改进和成熟度提升。 为了实现CMMI认证…

C++ 几何计算库

代码 #include <iostream> #include <list> #include <CGAL/Simple_cartesian.h> #include <CGAL/AABB_tree.h> #include <CGAL/AABB_traits.h> #include <CGAL/AABB_segment_primitive.h> #include <CGAL/Polygon_2.h>typedef CGAL…

线性代数|机器学习-P24加速梯度下降(动量法)

文章目录 1. 概述2. 引入3. 动量法梯度下降 1. 概述 我们之前学的最速梯度下降[线搜索方法] 公式如下&#xff1a; x k 1 x k − s k ∇ f ( x k ) \begin{equation} x_{k1}x_k-s_k\nabla f(x_k) \end{equation} xk1​xk​−sk​∇f(xk​)​​ 但对于这种方法来说&#xff…

助力数据记录:Connext的Historian功能启动指南

​ 在工业自动化领域&#xff0c;Connext以其全新的OPCUA数据采集解决方案为核心&#xff0c;不仅拥有超越传统的扩展功能&#xff0c;而且在多个行业中都得到了广泛的认证。Connext能够快速找出解决方案并迅速部署&#xff0c;整个过程不会影响到原有的生产。它的强大之处在于…