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

C++ 面向对象

C++ 面向对象编程

一个类可以定义无数个对象,每一个对象都有自己的成员变量,但是他们共享一套成员方法。

构造函数的初始化列表和直接在构造函数中构造的区别

  • 初始化列表是用来初始化成员类的,用来调用成员的构造函数的

  • 一个是先调用默认构造后初始化,一个是调用构造函数初始化

  • 即:int a = 10int a; a = 10 的区别。对于普通类型区别不大。

初始化列表的默认初始化顺序:成员函数的定义顺序。

静态成员变量:类内声明,类外定义和初始化,不属于对象,属于类级别。
静态成员方法:可以用类名调用。(普通成员函数不能用类名调用的原因:需要隐式传入成员 this 指针,不传缺少参数无法编译。)
常成员方法:接收的 this 指针为 const 类型。

模版函数:建议定义在头文件

辨析:为什么模版函数,内联函数要定义在头文件中,普通函数不建议定义在头文件中?
原因:在函数编译链接过程中,函数/全局变量/静态变量/常量常量变量等会产生符号,且定义在 头文件中的普通函数将会展开,在链接时,连接器如果发现同名的函数符号,会直接报错(error: multiple definition of ‘yourFunction’)。而对于模版函数和内联函数会生成一个弱符号(weak sym),连接器会从多个头文件展开后的源文件中选取一个链接,不会因出现同名弱符号而报错。且由于模版本身是不编译的,所以在一个文件中定义的模版一般无法在另一个文件中使用,因为模版不编译而不产生符号,导致使用的文件产生的 “UND” 找不到对应的函数定义。(有声明未定义的函数会别标志为“UND”并且在链接过程中在其他 .o 文件中寻找定义。)

new 和 delete 重载的应用 -> 对象池

#include <iostream>using namespace std;template <class T>
class Queue {
public:explicit Queue(const T& data = T()): _front(new QueueItem(data)) { _rear = _front; }~Queue() {if (empty()) {delete _front;return;}// 略}void push(const T& data) {QueueItem* temp = new QueueItem(data);_rear->_next = temp;_rear = temp;}void pop() {if (empty()) {perror("Queue is empty"); exit(1);}auto temp = _front->_next;delete _front;_front = temp;if (_front->_next == nullptr) {_rear = _front;}}[[nodiscard]] T& front() const {if (empty()) {perror("Queue is empty"); exit(1);}return _front->_next->_data;}[[nodiscard]] bool empty() const {return _front->_next == nullptr;}private:struct QueueItem {QueueItem(T data = T()): _next(nullptr), _data(data) {}void* operator new(size_t size) {if (_itemPool == nullptr) {_itemPool = (QueueItem *)new char[POOL_SIZE * sizeof(QueueItem)];auto p = _itemPool;for (; p < _itemPool + POOL_SIZE - 1; p++) {p->_next = p + 1;}p->_next = nullptr;}auto p = _itemPool;_itemPool = _itemPool->_next;return p;}void operator delete(void* p) {auto *ptr = (QueueItem *)p;ptr->_next = _itemPool;_itemPool = ptr;}QueueItem* _next;T _data;static QueueItem *_itemPool;static const int POOL_SIZE = 10000;};QueueItem *_front;  // 指向头节点QueueItem *_rear;   // 指向尾节点};template <class T>Queue<T>::QueueItem *Queue<T>::QueueItem::_itemPool = nullptr;int main() {Queue<int> que;for (int i = 0; i < 100; i++) {que.push(1);que.push(2);que.push(3);que.push(4);que.push(5);}for (int i = 0; i < 20; i++) {cout << que.front() << endl;que.pop();}return 0;
}

可通过重载 new 和 delete 运算符实现顺序链表的连接池。

http://www.dtcms.com/a/281925.html

相关文章:

  • 滚珠导轨在封装设备如何体现高精度运行?
  • 创建linux端口映射连接小网
  • 基于CentOS的分布式GitLab+Jenkins+Docker架构:企业级CI/CD流水线实战全记录
  • 如何选择适合的云手机配置?解决资源不足带来的性能瓶颈
  • Clip微调系列:《coOp: learning to prompt for vision-language models》
  • 蓝光三维扫描技术:手机闪光灯模块全尺寸3D检测的高效解决方案
  • Clip微调系列:《CLIP-Adapter: Better Vision-Language Models with FeatureAdapters》
  • pytorch | minist手写数据集
  • 防止应用调试分析IP被扫描加固实战教程
  • react19+nextjs+antd切换主题颜色
  • 【python学习】windows使用conda管理python虚拟环境
  • RNN循环神经网络
  • 面试问题:
  • STM32硬件I2C的注意事项
  • CCK-8 实验详解及 Graphpad 作图指南
  • 阿里云 RabbitMQ 可观测性最佳实践
  • 恶补DSP:1.F28335的时钟系统
  • Swarm Network 选择 Walrus 实现可验证 AI
  • 网络安全(初级)(Python实现sql自动化布尔盲注)
  • JWT基础详解
  • 【云原生网络】Istio基础篇
  • 使用 CrewAI 进行股票分析:自动化投资决策的新途径
  • Capture One24下载与保姆级安装教程!
  • 解决“Windows 无法启动服务”问题指南
  • 数据库询问RAG框架Vanna的总体架构
  • 线上崩溃复盘
  • Unity Android Logcat插件 输出日志中文乱码解决
  • FPGA基础 -- Verilog 访问寄存器数组的指定位示例
  • 第六章 OBProxy 路由与使用运维
  • [Linux入门] Linux 账号和权限管理入门:从基础到实践