当前位置: 首页 > news >正文

类和对象(三)

目录

一. 构造函数初始化列表

二. 类型转换

三. static成员

四. 友元

五. 内部类

六. 匿名对象

七. 对象拷贝时的编译器优化


一. 构造函数初始化列表

1. 之前我们实现构造函数时,初始化成员变量主要使用函数体内赋值,构造函数初始化还有一种方式,就是初始化列表,初始化列表的使用方式是以一个冒号开始,接一个以逗号分隔的数据成员列表,每个"成员变量"后面跟一个放在括号中的初始值或表达式

2. 每个成员变量在初始化列表中只能出现一次,语法理解上初始化列表可以认为是每个成员变量定义初始化的地方

3. 引用成员变量,const成员变量,没有默认构造的类类型变量必须放在初始化列表位置进行初始化,否则会编译报错

#include<iostream>
using namespace std;class jubge
{
public:jubge(int tmp = 0):_tmp(tmp){cout << " 叮叮叮 " << endl;}int val(){return _tmp;}
private:int _tmp;
};
class Pc
{
public:Pc(int& x, int a = 1, int b = 21, int c = 11):_a(a), _b(b), _c(c), _s(33), _t(x), j(333)//, _b(b)每种成员变量只能在初始化列表出现一次{}void Printf(){cout << _a << " " << _b << " " << _c << " " << _t << " "<<j.val() << " " << _s << endl;}
private:int _a;int _b;int _c;int& _t;jubge j;const int _s;
};int main()
{int v = 10;int& x = v;Pc p(x);p.Printf();
}

而上文的引用必须初始化,const必须在初始化时赋值,类没有默认构造只能传参赋值

它们不能在函数体内赋值,引用必须初始化然后不能改变指向,const需要设置初始值后不能改变,类也需要参数初始化,没有其他函数使其改变

4. C++11支持在成员变量声明的位置给缺省值,这个缺省值主要是给没有显示在初始化列表初始化的成员使用的

5. 尽量使用初始化列表初始化,因为那些你不在初始化列表初始化的成员也会走初始化列表,如果这个成员在声明位置给了这个缺省值初始化列表会用这个缺省值初始化。如果你没有给缺省值,对于没有显示在初始化列表初始化的内置类型成员是否初始化取决于编译器,C++没规定。而对没显示在初始化列表的自定义类型成员会调用这个成员类型的默认构造函数,如果没有默认构造会编译错误

6. 初始化列表中按照成员变量在类中声明顺序进行初始化,跟成员在初始化列表出现的先后顺序无关。建议声明顺序和初始化列表顺序保持一致

#include<iostream>
using namespace std;class jubge
{
public:jubge(int tmp = 0):_tmp(tmp){cout << " 叮叮叮 " << endl;}int val(){return _tmp;}
private:int _tmp;
};
class Pc
{
public:Pc(int& x): _t(x){}void Printf(){cout << _a << " " << _b << " " << _c << " " << _t << " "<<j.val() << " " << _s << endl;}
private:int _a=1;int _b=21;int _c=11;int& _t;jubge j=333;const int _s=33;int* p = (int*)malloc(12);
};int main()
{int v = 10;int& x = v;Pc p(x);p.Printf();
}

我们发现我们没在初始化列表里定义,但是依然会根据初始化

我们可以思考一下以下代码指向结果为

#include<iostream>
using namespace std;
class Pc
{
public:Pc(int a):_a(a), _b(_a){}void Printf(){cout << _a << " " << _b << " "<< endl;}
private:int _b = 21;int _a = 1;
};int main()
{Pc p(10);p.Printf();
}

 由于初始化列表按照成员变量在类中的声明顺序进行初始化。跟成员在初始化列表中出现的顺序无关。所以会导致先初始化_b再初始化_a,用于初始化_b的_a是随机值,然后初始化_a为10

答案为 10  随机值

二. 类型转换

1. C++支持内置类型隐式转换为类类型对象,需要有相关内置类型为参数的构造函数

2. 构造函数前面加explicit 就不再支持隐式类型转换

#include<iostream>
using namespace std;
class Pc
{
public:Pc(int a):_a(a), _b(a){}Pc(int a,int b):_a(a), _b(b){}void Printf() const{cout << _a << " " << _b << " "<< endl;}
private:int _b = 21;int _a = 1;
};int main()
{Pc p = 10;p.Printf();//Pc& pp = 1;const Pc& pp = 1;//产生的临时对象具有常性pp.Printf();//C++11后才支持多参数转化Pc ppp = { 3,3 };ppp.Printf();
}

第一个赋值,1构造一个Pc的临时对象,再用这个临时对象拷贝构造p,编译器遇见连续构造+拷贝构造

如果将类改为以下代码

class Pc
{
public:explicit Pc(int a):_a(a), _b(a){}explicit Pc(int a,int b):_a(a), _b(b){}void Printf() const{cout << _a << " " << _b << " "<< endl;}
private:int _b = 21;int _a = 1;};

就不再支持隐式类型转换了

三. static成员

1. 用static修饰的成员变量,称之为静态成员变量,静态成员变量一定要在类外进行初始化

2. 静态成员变量为所有类对象所共享不属于某个具体的对象,不存在对象中,存放在静态区中。 

3. 用static修饰的成员函数,称之为静态成员函数,静态成员函数没有this指针

4. 静态成员函数可以访问其他静态成员但不能访问非静态的,因为没有this指针。

5. 非静态的成员函数,可以访问任意的静态成员变量和静态成员函数

6. 突破类域就可以访问静态成员,可以通过类名::静态成员对象.静态成员来访问来访问静态成员变量和静态成员函数

7. 静态成员变量不能在声明位置给缺省值初始化,因为缺省值是个构造函数初始化列表的,静态成员变量不属于某个对象,不走构造函数初始化列表

#include<iostream>
using namespace std;
class Pc
{
public:Pc():_a(1), _b(1){}static int GetVal(){//_b=0;//没有this指针无法访问return n;}void PrintfStatic(){cout << c << "  " << GetVal() << "  " << n << endl;}void Printf() const{cout << _a << " " << _b << " " << endl;}static int c;
private:int _b = 21;int _a = 1;static int n;};
int Pc:: c = 10;
int Pc::n = 111;
int main()
{Pc::c = 1;cout << Pc::c << endl;Pc p;cout << p.c << endl;p.PrintfStatic();
}

四. 友元

1. 友元提供了⼀种突破类访问限定符封装的⽅式,友元分为:友元函数和友元类,在函数声明或者类 声明的前⾯加friend并且把友元声明放到⼀个类的⾥⾯

2. 外部友元函数可访问类的私有和保护成员,友元函数仅仅是⼀种声明,他不是类的成员函数

3. 友元函数可以在类定义的任何地⽅声明,不受类访问限定符限制

4. ⼀个函数可以是多个类的友元函数

 如以下代码所示

#include<iostream>
using namespace std;class Pc;class AAA
{public:AAA():_t(10), _s(1){}
private:friend void Printf(const Pc& p, const AAA& a);//声明可以在任何地方int _t;int _s;
};class Pc
{friend void Printf(const Pc& p,const AAA& a);public:Pc():_a(1), _b(1){}
private:int _b = 21;int _a = 1;
};void Printf(const Pc& p, const AAA& a)
{cout << p._a << " " << p._b << " " << endl;cout << a._s << " " << a._t << " " << endl;
}
int main()
{Pc p;AAA a;Printf(p,a);
}

5. 友元类中的成员函数都可以是另⼀个类的友元函数,都可以访问另⼀个类中的私有和保护成员。

6. 友元类的关系是单向的不具有交换性,比如A类是B类的友元,但是B类不是A类的友元。

7. 友元类关系不能传递,如果A是B的友元,B是C的友元,但是A不是C的友元。

8. 有时提供了便利。但是友元会增加耦合度,破坏了封装,所以友元不宜多⽤

#include<iostream>
using namespace std;class Pc;
class BBB;
class AAA
{friend class Pc;
public:AAA():_t(10), _s(1){}//void PirntfPc(const Pc& p,const BBB& b)//{//	cout << p._b << " " << p._s << endl;//友元关系是单向的//	cout << b.n << endl;//友元的关系不可传递//}
private:int _t;int _s;
};
class BBB
{private:int n = 0;};class Pc
{friend class BBB;
public:Pc():_a(1), _b(1){}void PirntfAAA(const AAA& a){cout << a._t << " " << a._s << endl;}private:int _b = 21;int _a = 1;
};
int main()
{Pc p;AAA a;p.PirntfAAA(a);
}

五. 内部类

1. 如果一个类定义在另一个类的内部,这个类就叫做内部类。内部类是一个独立的类,跟定义在全局相比,他只是受外部类类域限制和访问限定符限制,所以外部类定义的对象中不包含内部类

2. 内部类默认为外部类的友元类

3. 内部类本质也是一种封装,当A类与B类紧密相关,A类的实现就是为了给B类使用的时候,可以考虑把A类设计为B的内部类,如果放到protected/private位置,那A类就是B类的专属内部类,其他地方都用不了

#include<iostream>
using namespace std;class Pc
{public:Pc():_a(1), _b(1){}void printfAAAPc(){AAA a;a.PirntfPc(*this);a.PrintfAAA();}class BBB{public:void PirntfPc(const Pc& p){cout << "BBB" << endl;cout << p._b << " " << p._a << endl;}};
private:class AAA{public:AAA():_t(10), _s(1){}void PrintfAAA(){cout << _t << " " << _s << endl;}void PirntfPc(const Pc& p){cout << "AAA" << endl;cout << p._b << " " << p._a << endl;}private:int _t;int _s;};int _b = 21;int _a = 1;
};
int main()
{Pc p;Pc::BBB b;b.PirntfPc(p);p.printfAAAPc();
}

六. 匿名对象

类型(实参)  定义出来的对象叫做匿名对象,相⽐之前我们定义的  类型对象名(实参)  定义出来的 叫有名对象

匿名对象⽣命周期只在当前⼀行,⼀般临时定义⼀个对象当前用⼀下即可,就可以定义匿名对象。

 以下代码可以验证

#include<iostream>
using namespace std;class Pc
{public:Pc():_a(1), _b(1),_val(111){cout << "构造了" << endl;}~Pc(){cout << "析构了" << endl;}int GetVal(){return _val;}
private:int _b ;int _a = 1;int _val = 111;
};
int main()
{Pc();cout << Pc().GetVal() << endl;return 0;
}

七. 对象拷贝时的编译器优化

现代编译器,会为了尽可能的提高程序效率,在不影响正确性的情况下会尽可能减少一些传参和传参过程中可以省略的拷贝

如何优化C++标准没有严格规定各个编译器会根据情况自行处理。当前主流的相对新的编译器对于连续一个表达式步骤中的连续拷贝会进行合并优化,有些更新更‘激进’的编译还会进行跨行跨表达式的合并优化

 如以下代码:

#include<iostream>
using namespace std;class AAA
{
public:AAA(int a){_a = a;cout << "构造" << endl;}AAA(const AAA& a){_a = a._a;cout << "拷贝构造" << endl;}void operator=(const AAA& a){_a = a._a;cout << "重载赋值" << endl;}void Printf(){cout << "printf->" << _a << endl;}~AAA(){cout << "析构" << endl;}
private:int _a;
};AAA f()
{AAA a(1);return a;
}int main()
{AAA aaa(1);aaa = f();aaa.Printf();cout << "*********" << endl;return 0;
}

 按理来说会 1. 将 aaa 执行构造函数 2. 进入 f 函数后 对a执行构造函数 ,3. 将a拷贝构造赋值给临时对象,4. 将a 析构 5. 执行函数重载=,6. 再将临时对象 析构 ,7. 再执行 aaa的Printf函数, 8. 执行cout  9. 将aaa 析构

而在vs2019中Debug下也是这么做的,但在vs2022的激进优化下变为了 1. 将aaa执行构造函数,2. 进入f 函数后对a 进行构造函数 3. 将a 赋值给 aaa,4. 将 a 析构 5.执行 aaa的Printf函数 6. 执行cout 7.  将aaa 析构

结果如下


这篇先写到这里,明天补第七条,喜欢可以点一下赞

(๑′ᴗ‵๑)I Lᵒᵛᵉᵧₒᵤ❤

相关文章:

类和对象(三)

目录 一. 构造函数初始化列表 二. 类型转换 三. static成员 四. 友元 五. 内部类 六. 匿名对象 七. 对象拷贝时的编译器优化 一. 构造函数初始化列表 1. 之前我们实现构造函数时&#xff0c;初始化成员变量主要使用函数体内赋值&#xff0c;构造函数初始化还有一种方式&…...

Android SurfaceFlinger——GraphicBuffer初始化(二十九)

在 SurfaceFlinger 中,GraphicBuffer 是一个关键的数据结构,用于封装和管理图形数据的内存缓冲区。它不仅在 SurfaceFlinger 内部使用,也被其他组件如 GPU 驱动、摄像头服务、视频解码器等广泛利用,以实现高效的数据交换和图形渲染。 一、概述 GraphicBuffer 对象封装了一…...

pytest:4种方法实现 - 重复执行用例 - 展示迭代次数

简介&#xff1a;在软件测试中&#xff0c;我们经常需要重复执行测试用例&#xff0c;以确保代码的稳定性和可靠性。在本文中&#xff0c;我们将介绍四种方法来实现重复执行测试用例&#xff0c;并显示当前迭代次数和剩余执行次数。这些方法将帮助你更好地追踪测试执行过程&…...

一文入门SpringSecurity 5

目录 提示 Apache Shiro和Spring Security 认证和授权 RBAC Demo 环境 Controller 引入Spring Security 初探Security原理 认证授权图示​编辑 图中涉及的类和接口 流程总结 提示 Spring Security源码的接口名和方法名都很长&#xff0c;看源码的时候要见名知意&am…...

IPython的HTML魔法:%%html_header命令全解析

IPython的HTML魔法&#xff1a;%%html_header命令全解析 在IPython和Jupyter Notebook中&#xff0c;%%html_header是一个魔术命令&#xff0c;它允许用户在Notebook的单元格中添加HTML头部&#xff08;head&#xff09;内容。这个功能特别有用&#xff0c;当你需要定制Notebo…...

将SQL中的占位符替换成参数

将SQL中的占位符替换成参数 描述 描述 此方法是将SQL中的${}或#{}替换为直接拼接到SQL中或直接替换为?的形式。具体详情看下面代码。 import java.util.*; import java.util.regex.Matcher; import java.util.regex.Pattern;/*** author HuYu* date 2023-09-21* since 1.0**…...

锁相环 vivado FPGA

原理 同步状态/跟踪状态&#xff1a;相位差在2kπ附近&#xff0c;频率差为0到达上述状态的过程称为捕获过程锁相环的捕获带&#xff1a;delta w的最大值&#xff0c;大于这个值的话就不能捕获鉴相器&#xff08;PD-phase discriminator&#xff09;&#xff1a;相乘加LPF&…...

英语科技写作 希拉里·格拉斯曼-蒂(英文版)pdf下载

下载链接&#xff1a; 链接1&#xff1a;https://pan.baidu.com 链接2&#xff1a;/s/1fxRUGnlJrKEzQVF6k1GmBA 提取码&#xff1a;b69t 由于是英文版&#xff0c;可能有些看着不太方便&#xff0c;可以在网页版使用以下软件中英文对照着看&#xff0c;看着更舒服&#xff0c;…...

《Dynamic Statistical Learning in Massive Datastreams》论文阅读笔记

论文地址: https://www3.stat.sinica.edu.tw/ss_newpaper/SS-2023-0195_na.pdf 论文题目翻译&#xff1a;《在大规模数据流中的动态统计学习》 核心观点&#xff1a; 动态跟踪和筛选框架&#xff08;DTS&#xff09;&#xff1a;论文提出了一个在线学习和模型更新的新框架&…...

【数据分享】2008-2022年我国省市县三级的逐日NO2数据(excel\shp格式)

空气质量数据是在我们日常研究中经常使用的数据&#xff01;之前我们给大家分享了2000-2022年的省市县三级的逐日PM2.5数据、2013-2022年的省市县三级的逐日CO数据和2013-2022年的省市县三级的逐日SO2数据&#xff08;均可查看之前的文章获悉详情&#xff09;&#xff01; 本次…...

JavaEE (1)

web开发概述 所谓web开发,指的是从网页中向后端程序发送请求,与后端程序进行 交互. 流程图如下 Web服务器是指驻留于因特网上某种类型计算机的程序. 可以向浏览器等Web客户端提供文档&#xff0c;也可以放置网站文件&#xff0c;让全世界浏览&#xff1b; 它是一个容器&…...

事务、函数和索引

什么是事务&#xff1f; 事务&#xff08;Transaction&#xff09;&#xff0c;就是将一组SQL语句放在同一批次内去执行&#xff0c;如果一个SQL语句出错&#xff0c;则该批次内 的所有SQL都将被取消执行。 特点 一个事务中如果有一个数据库操作失败&#xff0c;那么整个事务…...

Android APP 基于RecyclerView框架工程(知识体系积累)

说明&#xff1a;这个简单的基于RecyclerView的框架作用在于自己可以将平时积累的一些有效demo整合起来&#xff08;比如音视频编解码的、opengles的以及其他也去方向的、随着项目增多&#xff0c;工程量的增加&#xff0c;后期想高效的分析和查找并不容易&#xff09;&#xf…...

【iOS】GCD

参考文章&#xff1a;GCD函数和队列原理探索 之前写项目的时候&#xff0c;进行耗时的网络请求使用GCD处理过异步请求&#xff0c;但对一些概念都很模糊&#xff0c;这次就来系统学习一下GCD相关 相关概念 什么是GCD&#xff1f; Grand Center Dispatch简称GCD&#xff0c;是…...

C语言 | Leetcode C语言题解之第282题给表达式添加运算符

题目&#xff1a; 题解&#xff1a; #define MAX_COUNT 10000 // 解的个数足够大 #define NUM_COUNT 100 // 操作数的个数足够大 long long num[NUM_COUNT] {0};long long calc(char *a) { // 计算表达式a的值// 将数字和符号&#xff0c;入栈memset(num, 0, sizeof(num));in…...

如何使用 API list 极狐GitLab 容器镜像仓库中的 tag?

GitLab 是一个全球知名的一体化 DevOps 平台&#xff0c;很多人都通过私有化部署 GitLab 来进行源代码托管。极狐GitLab &#xff1a;https://gitlab.cn/install?channelcontent&utm_sourcecsdn 是 GitLab 在中国的发行版&#xff0c;专门为中国程序员服务。可以一键式部署…...

粒子群算法PSO优化BP神经网络(PSO-BP)回归预测——Python和MATLAB实现

下面是一个使用Python实现的粒子群算法&#xff08;PSO&#xff09;优化反向传播神经网络&#xff08;BP&#xff09;的示例代码。 以下是具体的代码实现&#xff1a; python import numpy as np from sklearn.datasets import make_regression from sklearn.model_selection…...

React-router路由配置及跳转

1、V6对比V5的修改内容 1、API: useNavigate 代替了useHistory 。 2、废弃了Route组件的exact属性。 3、组件 <Routes>代替了<Switch> 4、组件NavLink中移除了 activeStyle activeClassName 属性。 2、安装依赖react-router-dom npm install react-router-dom…...

vue3【实战】可编辑的脱敏信息

<script lang"ts" setup> import { ref, onMounted } from "vue"; let real_name ref("朝阳");let name ref("");onMounted(() > {name.value des_name(real_name.value); });function focusing() {name.value real_name…...

S71200 - 笔记

1 S71200 0 ProfiNet - 2 PLC编程 01.如何零基础快速上手S7-1200_哔哩哔哩_bilibili 西门子S7-1200PLC编程设计学习视频&#xff0c;从入门开始讲解_哔哩哔哩_bilibili...

使用docker在3台服务器上搭建基于redis 6.x的一主两从三台均是哨兵模式

一、环境及版本说明 如果服务器已经安装了docker,则忽略此步骤,如果没有安装,则可以按照一下方式安装: 1. 在线安装(有互联网环境): 请看我这篇文章 传送阵>> 点我查看 2. 离线安装(内网环境):请看我这篇文章 传送阵>> 点我查看 说明&#xff1a;假设每台服务器已…...

React第五十七节 Router中RouterProvider使用详解及注意事项

前言 在 React Router v6.4 中&#xff0c;RouterProvider 是一个核心组件&#xff0c;用于提供基于数据路由&#xff08;data routers&#xff09;的新型路由方案。 它替代了传统的 <BrowserRouter>&#xff0c;支持更强大的数据加载和操作功能&#xff08;如 loader 和…...

阿里云ACP云计算备考笔记 (5)——弹性伸缩

目录 第一章 概述 第二章 弹性伸缩简介 1、弹性伸缩 2、垂直伸缩 3、优势 4、应用场景 ① 无规律的业务量波动 ② 有规律的业务量波动 ③ 无明显业务量波动 ④ 混合型业务 ⑤ 消息通知 ⑥ 生命周期挂钩 ⑦ 自定义方式 ⑧ 滚的升级 5、使用限制 第三章 主要定义 …...

(二)TensorRT-LLM | 模型导出(v0.20.0rc3)

0. 概述 上一节 对安装和使用有个基本介绍。根据这个 issue 的描述&#xff0c;后续 TensorRT-LLM 团队可能更专注于更新和维护 pytorch backend。但 tensorrt backend 作为先前一直开发的工作&#xff0c;其中包含了大量可以学习的地方。本文主要看看它导出模型的部分&#x…...

TRS收益互换:跨境资本流动的金融创新工具与系统化解决方案

一、TRS收益互换的本质与业务逻辑 &#xff08;一&#xff09;概念解析 TRS&#xff08;Total Return Swap&#xff09;收益互换是一种金融衍生工具&#xff0c;指交易双方约定在未来一定期限内&#xff0c;基于特定资产或指数的表现进行现金流交换的协议。其核心特征包括&am…...

汇编常见指令

汇编常见指令 一、数据传送指令 指令功能示例说明MOV数据传送MOV EAX, 10将立即数 10 送入 EAXMOV [EBX], EAX将 EAX 值存入 EBX 指向的内存LEA加载有效地址LEA EAX, [EBX4]将 EBX4 的地址存入 EAX&#xff08;不访问内存&#xff09;XCHG交换数据XCHG EAX, EBX交换 EAX 和 EB…...

深入解析C++中的extern关键字:跨文件共享变量与函数的终极指南

&#x1f680; C extern 关键字深度解析&#xff1a;跨文件编程的终极指南 &#x1f4c5; 更新时间&#xff1a;2025年6月5日 &#x1f3f7;️ 标签&#xff1a;C | extern关键字 | 多文件编程 | 链接与声明 | 现代C 文章目录 前言&#x1f525;一、extern 是什么&#xff1f;&…...

selenium学习实战【Python爬虫】

selenium学习实战【Python爬虫】 文章目录 selenium学习实战【Python爬虫】一、声明二、学习目标三、安装依赖3.1 安装selenium库3.2 安装浏览器驱动3.2.1 查看Edge版本3.2.2 驱动安装 四、代码讲解4.1 配置浏览器4.2 加载更多4.3 寻找内容4.4 完整代码 五、报告文件爬取5.1 提…...

学校时钟系统,标准考场时钟系统,AI亮相2025高考,赛思时钟系统为教育公平筑起“精准防线”

2025年#高考 将在近日拉开帷幕&#xff0c;#AI 监考一度冲上热搜。当AI深度融入高考&#xff0c;#时间同步 不再是辅助功能&#xff0c;而是决定AI监考系统成败的“生命线”。 AI亮相2025高考&#xff0c;40种异常行为0.5秒精准识别 2025年高考即将拉开帷幕&#xff0c;江西、…...

HDFS分布式存储 zookeeper

hadoop介绍 狭义上hadoop是指apache的一款开源软件 用java语言实现开源框架&#xff0c;允许使用简单的变成模型跨计算机对大型集群进行分布式处理&#xff08;1.海量的数据存储 2.海量数据的计算&#xff09;Hadoop核心组件 hdfs&#xff08;分布式文件存储系统&#xff09;&a…...