C++基础三(构造函数,形参默认值,函数重载,单例模式,析构函数,内联函数,拷贝构造函数)
C++有六个默认函数,分别是:
1、默认构造函数;
2、默认拷贝构造函数;
3、默认析构函数;
4、赋值运算符;
5、取址运算符;
6、取址运算符const;
构造函数
构造函数(初始化类成员变量):
1、属于类的成员函数之一
2、构造函数没有返回类型
3、构造函数的函数名必须与类名相同
4、构造函数不允许手动调用(不能通过类对象调用)
5、构造函数在类对象创建时会被自动调用
6、如果没有显示声明构造函数,系统会生成一个默认的构造函数
7、如果显示声明了构造函数,那么默认构造函数将不会在被创建
8、构造函数不被设置为静态成员函数
9、构造函数的函数首部之前或函数首部之后不能用const修饰
10、new一个类指针也会触发构造函数的调用, malloc不会
11、构造函数通过函数重载的方式也可以拥有多个
12、构造函数也可以使用形参默认值
13、构造函数可以是private的
#include <string.h>
#include <stdlib.h>
#include <iostream>
using namespace std;#if 0 // 构造函数class Node
{public://Node();Node(int n = 0, char* ptr = NULL, float score = 1.3f);// static Node(); // error// Node() const; // error// const Node(); // errorprivate:int m_num;char* m_ptr;float m_fscore;
};// 类成员变量初始化方式2(常用这种方式)
// 通常普通类成员变量都需要在构造函数上进行初始化。
//Node::Node()
// : m_num(111),
// m_ptr(NULL),
// m_fscore(0.0)
//{
// // 类成员变量初始化方式1:
// //m_num = 10;
//
// cout << "construct: " << m_num << endl;
//}Node::Node(int n, char* ptr, float score): m_num(n),m_ptr(NULL),m_fscore(score)
{m_ptr = new char[100];strcpy(m_ptr, ptr);cout << m_num << " : " << m_ptr << " : " << m_fscore << endl;
}int main(int argc,char *argv[])
{// 触发构造函数调用Node n;// n.Node(); // errorNode n1(1, "st", 1.1);// 触发构造函数调用//Node* n1 = new Node;// 不会触发构造函数调用//Node* n2 = (Node*)malloc(sizeof(Node));return 0;
}
#endif
形参默认值
形参默认值:
1、给函数的形参一个默认的值。
2、如果我们在调用函数时指定实参的值,则形参的默认值不生效。
3、如果调用函数时没有给实参值,形参默认值才会生效。
4、形参默认值只需要在函数声明时写出来即可。
如果有函数声明的前提下, 在函数定义时写形参默认值会报错。
5、形参默认值只能出现在形参列表的最右边(后边)。
#if 0 // 形参默认值class Node
{public:// 形参默认值void fun(int x, int y, int num = 12, int z = 1);void display();private:int m_num;
};void Node::fun(int x, int y, int num, int z)
{m_num = num;
}void Node::display()
{cout << m_num << endl;
}int main()
{Node n;n.fun(1, 2, 12345);n.display();n.fun(1, 2);n.display();return 0;
}#endif
函数重载
函数重载:
1、在C++允许出现同名函数
(1) 形参个数不同
(2) 形参类型不同
(3) 形参顺序不同: 前提类型不同
(4) 函数的返回类型不影响
2、函数重载时,调用函数是在编译期间就确定了具体调用对象,
因为我们将函数重载称之为静态联编
3、重载函数调用时,不要出现二义性问题
#if 0 // 函数重载
#include <stdio.h>void fun(int x, float y)
{cout << "fun(int x, float y)" << endl;
}void fun(int x, int y)
{cout << "void fun(int x, int y)" << endl;
}void fun(int x, int y, int z = 0)
{cout << "void fun(int x, int y, int z = 0)" << endl;
}void fun()
{cout << "fun()" << endl;
}void fun(float x, int y)
{cout << "fun(float x, int y)" << endl;
}int main()
{ fun(1.1f, 12);fun(12, 1.3f);fun(1, 2, 3);fun(1, 2); // error: 出现了二义性return 0;
}#endif
单例模式
#if 1 // 设计模式:单例模式class Node
{public:static Node* getInstance();private:Node();private:static Node* m_ptr;
};Node* Node::m_ptr = NULL;Node::Node()
{cout << "construst" << endl;
}Node* Node::getInstance()
{if (m_ptr == NULL){m_ptr = new Node;}return m_ptr;
}int main()
{//Node n; // error: ‘Node::Node()’ is private//Node* n1 = new Node; // error: ‘Node::Node()’ is private//Node* n2 = (Node*)malloc(sizeof(Node)); // 可以Node* n = Node::getInstance();cout << n << endl; Node* n1 = Node::getInstance();cout << n1 << endl; Node* n2 = Node::getInstance();cout << n2 << endl; Node* n3 = Node::getInstance();cout << n3 << endl; Node* n4 = Node::getInstance();cout << n4 << endl; return 0;
}
析构函数
析构函数:
1、类对象或类指针销毁时析构函数会被自动调用
2、如果没有显示声明析构函数,则系统会生成一个默认的析构函数
3、如果显示声明了析构函数,则不会在生成默认的析构函数
4、析构函数没有返回类型,没有形参列表,且函数名与类名相同
5、析构函数只有一个,不能重载
6、析构函数不能设置为私有的
7、析构函数可以手动调用,但不允许
8、如果类对象的作用域相同,那么在销毁时析构函数的执行顺序与构造相反。
#include <iostream>
using namespace std;class Node
{public:Node();Node(int n);~Node();//~Node(int num = 0); // error: destructors may not have parametersvoid fun();inline void fun1(){cout << "fun1" << endl;}private:int* m_ptr;
};Node::Node(): m_ptr(NULL)
{if (m_ptr == NULL){m_ptr = new int[100];}cout << "construct" << endl;
}Node::Node(int n): m_ptr(NULL)
{if (m_ptr == NULL){m_ptr = new int[100];}*m_ptr = n;cout << *m_ptr << "construct" << endl;
}void Node::fun()
{Node n(8);cout << "fun" << endl;
}Node::~Node()
{cout << *m_ptr << "destruct" << endl;if (m_ptr != NULL){delete[] m_ptr;m_ptr = NULL;}
}int main(int argc,char *argv[])
{Node n0(0), n2(2);n0.fun();Node n3(3), n4(4), n5(5);//Node* n1 = new Node;//delete n1;// n0.~Node(); // 可以但不允许cout << "aaa" << endl;return 0;
}
内联函数
内联函数:
1、在普通类成员函数声明之前添加 inline 关键字
2、内联函数通常函数定义和函数声明都写在类声明中
3、内联函数的函数体中不能出现复杂结构
4、函数函数体中的代码一般3行左右
5、通常被频繁调用的函数才需要设置为内联函数
6、内联函数是进行函数逻辑的替换,其不进行真正的函数调用,减少系统开销
7、系统会自动判断当前函数是否是内联函数,也会自动判别是否需要使用内联。
内联函数与c的宏函数有什么不同:
1、宏函数是在编译阶段处理,是纯字符串替换; 内联函数是运行阶段处理,是函数逻辑替换; 2、宏函数无法调试; 内 联函数可以调试;
3、宏函数没有返回值; 内联函数有返回值;
4、宏函数不能访问类成员; 内联函数可以访问类成员;
拷贝构造函数
拷贝构造函数:
1、拷贝构造函数属于构造函数的重载
2、拷贝构造函数,要求形参中有一个当前类的类对象的引用
3、如果没有显示声明拷贝构造函数,系统会生成一个默认的拷贝构造函数
4、如果显示声明了拷贝构造函数,那么默认拷贝构造函数将不会在被创建
5、当使用一个老对象去构造一个新对象时会调用拷贝构造函数
(1) 用一个老对象构造一个新对象,或者新对象赋给老对象。
(2) 当函数的形参是一个非引用的类对象时,实参到形参传递时。
(3) 函数返回一个非引用的类对象时。
6、如果没有显示声明和定义拷贝构造函数时,类中的非指针类成员的值是可以被拷贝的。
默认拷贝构造函数负责执行了值的拷贝。
7、如果显示声明和定义拷贝构造函数后,类中的所有成员变量都需要我们手动进行拷贝。
8、拷贝分为浅拷贝和深拷贝,默认拷贝构造函数只能进行浅拷贝。
9、浅拷贝只进行值的拷贝,深拷贝还会执行内存拷贝。
#include <iostream>
using namespace std;class Node
{public:Node();Node(const Node& n);~Node();private:int m_num;int* m_ptr;
};Node::Node(): m_num(12),m_ptr(NULL)
{if (m_ptr == NULL){m_ptr = new int[100];}*m_ptr = 123;cout << "construct" << m_num << " -- " << *m_ptr << endl;
}Node::Node(const Node& n): m_num(0)
{if (m_ptr == NULL){m_ptr = new int[100];}*m_ptr = *(n.m_ptr);cout << "copy construct" << m_num << " -- " << *m_ptr << endl;
}Node::~Node()
{cout << "destruct: " << m_num << " -- " << *m_ptr << " : " << m_ptr << endl;if (m_ptr != NULL){delete[] m_ptr;m_ptr = NULL;}
}int main(int argc,char *argv[])
{Node n1;Node n2(n1);return 0;
}
#include <iostream>
using namespace std;class Node
{public:Node();Node(const Node& n);~Node();Node fun(Node n);private:int m_num;
};Node::Node(): m_num(12)
{cout << "construct: " << m_num << endl;
}Node::Node(const Node& n): m_num(0)
{cout << "copy construct: " << m_num << endl;
}Node Node::fun(Node n)
{n.m_num = 111;cout << "fun" << endl;return n;
}Node::~Node()
{cout << "destruct: " << m_num << endl;
}int main(int argc,char *argv[])
{Node n1;//Node n2(n1);//Node n3 = n1;//Node n3 = n1.fun(n1);// 如果有返回,但我们没有接收的时候,系统会自动生成一个临时(隐藏)对象// 当前的临时(隐藏)对象的生存周期是从return开始,当前函数调用语句执行完毕结束。n1.fun(n1);cout << "---------" << endl;return 0;
}
相关文章:

C++基础三(构造函数,形参默认值,函数重载,单例模式,析构函数,内联函数,拷贝构造函数)
C有六个默认函数,分别是: 1、默认构造函数; 2、默认拷贝构造函数; 3、默认析构函数; 4、赋值运算符; 5、取址运算符; 6、取址运算符const; 构造函数 构造函数(初始化类成员变量): 1、属于类的成员函数之一 …...

Flutter Color 大调整,需适配迁移,颜色不再是 0-255,而是 0-1.0,支持更大色域
在之前的 3.10 里, Flutter 的 Impeller 在 iOS 上支持了 P3 广色域图像渲染,但是当时也仅仅是当具有广色域图像或渐变时,Impeller 才会在 iOS 上显示 P3 的广色域的颜色,而如果你使用的是 Color API,会发现使用的还是…...

如何使用VBA识别Excel中的“单元格中的图片”(2/2)
Excel 365升级了新功能,支持两种不同的插入图片方式: 放置在单元格中(Place in cell),新功能,此操作插入的图片下文中简称为单元格中的图片。放置在单元格上(Place over cell)&…...
2024系统架构师---下午题目常考概念
1.管道-过滤器的概念:管道-过滤器风格具备高内聚、低耦合、支持软件重用、扩展性好、支持并发等优点,但它有编写复杂、不适合处理交互应用等缺点。 2.隐式调用的概念:隐式调用基于事件触发的思想,具备支持软件重用,改…...

【Linux】从零开始认识五种IO模型 --- 理解五种IO模型,开始使用非阻塞IO
恐惧让你沦为囚犯, 希望让你重获自由。 --- 《肖申克的救赎》--- 五种IO模型与阻塞IO 1 前言2 五种IO模型3 非阻塞IO 1 前言 通过网络通信的学习,我们能够理解网络通信的本质是进程间通信,而进程间通信的本质就是IO。 IO就是input与outp…...
Spring Boot 集成阿里云直播点播
在当今数字化时代,视频直播和点播服务已经成为许多应用的核心功能。阿里云提供了强大的直播和点播服务,能够满足各种规模的应用需求。而 Spring Boot 作为一种流行的 Java 开发框架,能够快速构建高效的应用程序。本文将详细介绍如何在 Spring…...
舍伍德业务安全架构(Sherwood Applied Business Security Architecture, SABSA)
舍伍德业务安全架构(Sherwood Applied Business Security Architecture, SABSA)是一个企业级的安全架构框架,它提供了一个全面的方法来设计和实现信息安全策略。SABSA模型将业务需求与安全控制相结合,确保企业的信息安全措施能够支…...
论可以对抗ai编程的软件开发平台(直接把软件需求描述变成软件的抗ai开发平台)的设计
论可以对抗ai编程的软件开发平台(直接把软件需求描述变成软件的抗ai开发平台)的设计 大家知道,传统的数学密码,都可以被量子计算机破解,但是这些年发展出很多数学密码,量子计算机也破解不了,叫…...

饿了么数据库表设计
有商家表、商品表、商品规格表、购物车表,不难分析出表是不够全面的。 (1)首先分析需要补充的表 1.对于购物车而言肯定有对应的用户,因此要添加一个用户表。 2.商品规格是冷,热,半分糖、全糖,对于冷热和半分糖是可以分…...
Flink处理乱序的数据的最佳实践
目录 网络延迟和分布式系统 事件时间与处理时间的差异 事件时间和水位线(Watermark) 时间窗口(TimeWindow) 滚动窗口(Tumbling Window) 滑动窗口(Sliding Window) 会话窗口(Session Window) 自定义Watermark生成策略 设置允许延迟和侧输出 设置允许的最大延迟时间 使…...

Android OpenGL ES详解——模板Stencil
目录 一、概念 1、模板测试 2、模板缓冲 二、模板测试如何使用 1、开启和关闭模板测试 2、开启/禁止模板缓冲区写入 3、模板测试策略函数 4、更新模板缓冲 5、模板测试应用——物体轮廓 三、模板缓冲如何使用 1、创建模板缓冲 2、使用模板缓冲 3、模板缓冲应用——…...

vscode在cmake config中不知道怎么选一个工具包?select a kit
vscode在cmake config中不知道怎么选一个工具包,或者发现一直在用VS的工具包想换成自己的工具包。select a kit vscode在cmake config中不知道怎么选一个工具包,或者发现一直在用VS的工具包想换成自己的工具包。select a kit 1.在VSCode中 按ctrlshift…...

无人机之无线电监测设备技术篇
一、技术原理 无人机的无线电监测设备主要通过捕捉和分析无人机发出的无线电信号来实现对无人机的监测和定位。这些信号包括无人机的上行遥控信号、下行数据图传信号等。设备采用多种技术手段,如频谱分析、信号解调、定位算法等,对接收到的信号进行处理和…...
【系统架构设计师】预测试卷一:案例分析
更多内容请见: 备考系统架构设计师-专栏介绍和目录 文章目录 试题一(共25分)【问题 1】(12分)【问题 2】(13分)试题二(共 25分)【问题 1】(12分)【问题 2】(7分)【问题 3】(6分)试题三(共25分)【问题 1】(9分)【问题 2】(16分)试题四(共25分)【问题 1】…...

一篇文章教会你I2C通信(软件I2C和硬件I2C)以读取MPU6050为例,附STM32代码示例
目录 一、I2C通信介绍: (1)基本概念: (2)特点: (3)工作原理: 二、I2C通信原理: (1)I2C 物理层: &…...
Python实现SPFA算法
目录 Python实现SPFA算法引言一、SPFA算法的理论基础1.1 最短路径问题1.2 SPFA算法的基本原理1.3 SPFA算法的复杂度 二、SPFA算法的Python实现2.1 基本实现2.2 案例一:使用SPFA算法进行城市交通最短路径计算2.2.1 实现代码 2.3 案例二:负权重边的处理2.3…...

MYSQL安装(ubuntu系统)
rpm -qa 查询安装软件包 ps axj 查询服务 卸载mysql(万不得已) ps axj | grep mysql 查看是否存在mysql服务 systemctl stop mysqld 关闭该服务 rpm -qa | grep mysql 查安装mysql安装包 rmp -qa | grep mysql | xargs (yum apt) -y remove进行批量…...

Cpp二叉搜索树的讲解与实现(21)
文章目录 前言一、二叉搜索树的概念定义特点 二、二叉树的实现基本框架查找插入删除当只有0 ~ 1个孩子的时候当有2个孩子的时候 三、二叉树的应用K模型KV模型 四、二叉树的性能分析总结 前言 这是全新的一个篇章呢,二叉搜索树是我们接下来学习set、map的前提 迈过它…...

微服务设计模式 — 补偿事务模式(Compensating Transaction Pattern)
微服务设计模式 — 补偿事务模式(Compensating Transaction Pattern) 定义 在云计算和分布式系统中,管理跨多个微服务或组件的事务一致性是一项极具挑战性的任务,补偿事务模式Compensating Transaction Pattern)是一种…...

20 实战:形状编码、运动补偿和纹理编码的实现(基于python)
在当今多媒体时代,视频处理与编码已经成为各个领域中不可或缺的一部分。无论是视频编辑、流媒体传输,还是计算机视觉应用,视频编码技术都扮演着关键角色。本文将详细解析一个基于Python的图形用户界面(GUI)视频编码器。通过对代码的逐行讲解、功能分析以及参数调节方法的探…...

深度学习在微纳光子学中的应用
深度学习在微纳光子学中的主要应用方向 深度学习与微纳光子学的结合主要集中在以下几个方向: 逆向设计 通过神经网络快速预测微纳结构的光学响应,替代传统耗时的数值模拟方法。例如设计超表面、光子晶体等结构。 特征提取与优化 从复杂的光学数据中自…...

eNSP-Cloud(实现本地电脑与eNSP内设备之间通信)
说明: 想象一下,你正在用eNSP搭建一个虚拟的网络世界,里面有虚拟的路由器、交换机、电脑(PC)等等。这些设备都在你的电脑里面“运行”,它们之间可以互相通信,就像一个封闭的小王国。 但是&#…...
三维GIS开发cesium智慧地铁教程(5)Cesium相机控制
一、环境搭建 <script src"../cesium1.99/Build/Cesium/Cesium.js"></script> <link rel"stylesheet" href"../cesium1.99/Build/Cesium/Widgets/widgets.css"> 关键配置点: 路径验证:确保相对路径.…...
day52 ResNet18 CBAM
在深度学习的旅程中,我们不断探索如何提升模型的性能。今天,我将分享我在 ResNet18 模型中插入 CBAM(Convolutional Block Attention Module)模块,并采用分阶段微调策略的实践过程。通过这个过程,我不仅提升…...
Go 语言接口详解
Go 语言接口详解 核心概念 接口定义 在 Go 语言中,接口是一种抽象类型,它定义了一组方法的集合: // 定义接口 type Shape interface {Area() float64Perimeter() float64 } 接口实现 Go 接口的实现是隐式的: // 矩形结构体…...
Linux简单的操作
ls ls 查看当前目录 ll 查看详细内容 ls -a 查看所有的内容 ls --help 查看方法文档 pwd pwd 查看当前路径 cd cd 转路径 cd .. 转上一级路径 cd 名 转换路径 …...

el-switch文字内置
el-switch文字内置 效果 vue <div style"color:#ffffff;font-size:14px;float:left;margin-bottom:5px;margin-right:5px;">自动加载</div> <el-switch v-model"value" active-color"#3E99FB" inactive-color"#DCDFE6"…...

【项目实战】通过多模态+LangGraph实现PPT生成助手
PPT自动生成系统 基于LangGraph的PPT自动生成系统,可以将Markdown文档自动转换为PPT演示文稿。 功能特点 Markdown解析:自动解析Markdown文档结构PPT模板分析:分析PPT模板的布局和风格智能布局决策:匹配内容与合适的PPT布局自动…...

高等数学(下)题型笔记(八)空间解析几何与向量代数
目录 0 前言 1 向量的点乘 1.1 基本公式 1.2 例题 2 向量的叉乘 2.1 基础知识 2.2 例题 3 空间平面方程 3.1 基础知识 3.2 例题 4 空间直线方程 4.1 基础知识 4.2 例题 5 旋转曲面及其方程 5.1 基础知识 5.2 例题 6 空间曲面的法线与切平面 6.1 基础知识 6.2…...
反射获取方法和属性
Java反射获取方法 在Java中,反射(Reflection)是一种强大的机制,允许程序在运行时访问和操作类的内部属性和方法。通过反射,可以动态地创建对象、调用方法、改变属性值,这在很多Java框架中如Spring和Hiberna…...