C++面试八股文:智能指针
一、了解哪些智能指针?
回答:智能指针是用于管理动态分配的内存,行为类似于指针,但又具有自动管理内存的能力,所以称为智能指针。
首先说一下 auto_ptr和unique_ptr,它们都是独占式指针,同一时间只能有一个指针拥有所有权。
auto_ptr是c++98引入,unique_ptr是c++11引入,替代了auto_ptr。
auto_ptr在设计上存在一些问题:
1、所有权转移的隐式语义(容易引发意外行为)
问题本质:auto_ptr 的拷贝构造函数和赋值运算符会转移资源的所有权,导致原指针变为 nullptr。这种隐式所有权转移在代码中难以察觉,容易引发逻辑错误。
对比 unique_ptr:C++11 的 unique_ptr 通过禁用拷贝构造函数(仅允许显式移动语义)明确所有权转移的意图,避免隐式错误。
2、与 STL 容器不兼容
问题本质:STL 容器(如 vector、list)要求元素是可拷贝且行为可预测的。而 auto_ptr 在拷贝时转移所有权,会导致容器内部元素意外失效。
对比 unique_ptr:unique_ptr 不可拷贝,但支持移动语义,可与 C++11 后的容器安全配合。
3、无法正确处理数组
因为它内部使用delete而不是delete[],所以如果用auto_ptr来管理数组会导致内存泄漏。
shared_ptr和weak_ptr:
shared_ptr使用了引用计数(use count)技术,当复制个shared_ptr对象时,被管理的资源并没有被复制,而是增加了引用计数。当析构一个shared_ptr对象时,也不会直接释放被管理的的资源,而是将引用计数减一。当引用计数为0时,才会真正的释放资源。shared_ptr可以方便的共享资源而不必创建多个资源。
weak_ptr则是一种弱引用,指向shared_ptr所管理的对象;
weak_ptr可以解决shared_ptr所持有的资源循环引用问题。weak_ptr在指向shared_ptr时,并不会增加shared_ptr的引用计数。
weak_ptr常用于实现观察者模式、缓存机制等场景,在这些场景中,我们需要观察一个对象,但又不想影响其生命周期。
shared_ptr VS unique_ptr
- 尽可能用unique_ptr
- shared_ptr应该仅在绝对需要共享时使用
- 避免使用任何其他指针,如weakptr、raw指针等。
从技术上讲,所有unique_ptr都可以被shared_ptr替换而不会出现编译错误,但我们仍然使用unique_ptr,原因如下:
- unique_ptr具有与C++原始指针“几乎”相同的效率,包括大小分配、引用/取消引用(假设没有自定义构造函数)。
- unique_ptr明确了对象的所有权和生存时间。
- shared_ptr的大小是原始指针的两倍,堆栈更大。
- 最重要的是,shared_ptr需要维护引用计数,并且该操作必须是原子的(即线程安全的),这与unique_ptr相比增加了恒定的开销。
因此,我们不应该仅仅因为方便而使用shared_ptr:对于shared_pcr的每次使用,我们都需要明确解释为什么这应该是一个共享指针,以及需要共享这个对象的模块/函数是什么。
避免使用所有其他指针,如weak_ptr或raw指针,因为在大多数情况下,它们可以被unique_ptr/shared_ptr替换。然而,显然也有例外,例如,所有QT应用程序都建议使用原始指针来管理UI组件。
常见面试问题
智能指针有哪些种类,它们各自的特点和适用场景是什么?
unique_ptr(独占指针)
特点:
独占所有权,同一时间只能有一个 unique_ptr 指向资源。
禁止拷贝,支持移动语义(通过 std::move 转移所有权)。
轻量级,无额外内存开销。
shared_ptr(共享指针)
特点:
共享所有权,通过引用计数管理资源生命周期。
支持拷贝和赋值,引用计数为 0 时自动释放资源。
控制块包含引用计数和删除器,有一定内存开销。
适用场景:
多个对象需要共享同一资源(如缓存、观察者模式)。
需要跨作用域共享资源。
weak_ptr(弱指针)
特点:
不增加引用计数,用于观测 shared_ptr 管理的资源。
需通过 lock() 转换为 shared_ptr 才能访问资源。
适用场景:
解决 shared_ptr 循环引用问题。
缓存或观察者模式中避免持有资源所有权。
智能指针是如何实现自动内存管理的,背后的原理是什么?
核心机制:基于 RAII(Resource Acquisition Is Initialization) 设计模式。
资源获取即初始化:在构造函数中分配资源,在析构函数中释放资源。
{std::unique_ptr<int> p(new int(42)); // 构造时分配内存// 使用 p
} // 离开作用域时,p 析构并自动调用 delete 释放内存
shared_ptr 是如何实现引用计数的,引用计数在多线程环境下有什么问题,如何解决?
引用计数的修改需保证原子性,否则多线程竞争导致计数错误。
使用 原子操作(std::atomic) 管理引用计数。
注意:shared_ptr 的引用计数本身是线程安全的,但指向的资源需额外同步。
unique_ptr 为什么不支持拷贝和赋值操作,它是如何实现资源的独占式管理的?
禁止拷贝和赋值:
通过删除拷贝构造函数和拷贝赋值运算符实现:
unique_ptr(const unique_ptr&) = delete;
unique_ptr& operator=(const unique_ptr&) = delete;
支持移动语义:
允许通过移动构造函数和移动赋值运算符转移所有权:
std::unique_ptr<int> p1(new int(42));
std::unique_ptr<int> p2 = std::move(p1); // p1 变为 nullptr
weak_ptr 的作用是什么,它与 shared_ptr 之间是如何配合工作来解决循环引用问题的?
观测 shared_ptr 管理的资源,不增加引用计数。
通过 lock() 安全访问资源:
std::weak_ptr<int> wp = sp; // sp 是 shared_ptr
if (auto spt = wp.lock()) { // 转换为 shared_ptr// 使用 spt
}
解决循环引用:
示例:
class B;
class A {std::shared_ptr<B> b_ptr;
};
class B {std::shared_ptr<A> a_ptr; // 循环引用,引用计数无法归零
};
改进:
class B {std::weak_ptr<A> a_ptr; // 使用 weak_ptr 打破循环
};
智能指针在实际项目中有哪些应用场景,举例说明。
智能指针和原始指针相比,有哪些优势和劣势?
优点:
- 能够自动管理内存,避免内存泄漏、悬空指针、野指针等问题,提高代码的安全性。
- 能够提升代码的整洁性,不用编写繁琐的内存分配和释放代码,使代码逻辑更加清晰,可读性更好,可维护性更强。
缺点:
性能开销和灵活性降低。
在使用智能指针时,需要注意哪些问题,如何避免常见的错误?
(1) 避免混合使用 new 和智能指针构造函数:
(2) 慎用 get():
避免通过 get() 获取的原始指针手动释放资源。
(3) 优先使用 make_shared/make_unique:
提升性能并减少内存碎片:
auto p = std::make_shared<int>(42); // 优于 shared_ptr<int>(new int(42))
(4) 尽量不要使用裸指针初始化智能指针?
因为可能存在同一个裸指针初始了多个智能指针,在智能指针析构时会造成资源的多次释放。
为什么优先使用 make_shared/make_unique
(1) make_shared和make_unique是C++11和C++14引入的工厂函数。它们的优势在于,当创建对象时,能够将对象和控制块的内存分配合并为一次。而直接使用new的话,需要先分配对象内存,再分配控制块,这样会有两次内存分配,增加开销和碎片。
std::shared_ptr<Widget> sp(new Widget); // 两次内存分配:对象 + 控制块
auto sp = std::make_shared<Widget>(); // 一次内存分配:对象和控制块连续存储
合并对象和控制块的内存分配,减少内存碎片和开销。
make_shared 将对象和控制块存储在连续内存中,提高 CPU 缓存命中率,减少访问延迟。
(2) 能保证异常安全。如果在构造智能指针时,参数表达式抛出异常,使用new可能会导致内存泄漏,而make_shared/make_unique因为直接在参数中构造,不会有这个问题。例如,如果有一个函数调用作为参数,可能在new之后,构造shared_ptr之前抛出异常,导致内存泄漏。而make函数会避免这种情况。
示例:潜在的内存泄漏
void process(std::shared_ptr<Widget> sp, int priority);// 危险!若 getPriority() 抛出异常,new Widget 的内存会泄漏!
process(std::shared_ptr<Widget>(new Widget), getPriority());
使用 make_shared 解决
process(std::make_shared<Widget>(), getPriority()); // 无泄漏风险
原因:make_shared 直接在函数参数中构造智能指针,若后续操作抛出异常,已构造的智能指针会自动释放资源。
(3) 代码简洁性
避免重复类型书写:
make_shared/make_unique 利用 auto 关键字,简化代码:
auto sp = std::make_shared<std::vector<std::string>>(); // 清晰简洁
// vs
std::shared_ptr<std::vector<std::string>> sp(new std::vector<std::string>);
适用场景
默认选择:优先使用 make_shared 和 make_unique,除非有特殊需求。
不适用场景:
需要自定义删除器(如 delete[] 或文件句柄释放)。
需要分离对象和控制块的生命周期(如大量 weak_ptr 长期存在时,make_shared 可能延迟内存释放)。
相关文章:
C++面试八股文:智能指针
一、了解哪些智能指针? 回答:智能指针是用于管理动态分配的内存,行为类似于指针,但又具有自动管理内存的能力,所以称为智能指针。 首先说一下 auto_ptr和unique_ptr,它们都是独占式指针,同一时…...
nohup命令使用说明
文章目录 如何在后台运行程序呢?如何正常运行代码重定向呢?nohup: ignoring input 如何在后台运行程序呢? 使用nohup命令即可, nohup python dataset/ReferESpatialDataset.py >>dataset_20250417.log 2>&1 &n…...
MYSQL “Too Many Connections“ 错误解决
1.查询当前连接数 show status like "Threads_connected"; 2.查询数据库最大连接数 show variables like "max_connections" 3.查询所有活动连接 show processlist; 4.根据查询结果观察是否有长时间未被释放的连接 参数解释 : 字段说明id连接的唯一…...
Linux `init 6` 相关命令的完整使用指南
Linux init 6 相关命令的完整使用指南—目录 一、init 系统简介二、init 6 的含义与作用三、不同 Init 系统下的 init 6 行为1. SysVinit(如 CentOS 6、Debian 7)2. systemd(如 CentOS 7、Ubuntu 16.04)3. Upstart(如 …...
【外研在线-注册/登录安全分析报告】
前言 由于网站注册入口容易被黑客攻击,存在如下安全问题: 暴力破解密码,造成用户信息泄露短信盗刷的安全问题,影响业务及导致用户投诉带来经济损失,尤其是后付费客户,风险巨大,造成亏损无底洞…...
【NLP 63、大模型应用 —— Agent】
人与人最大的差距就是勇气和执行力,也是唯一的差距 —— 25.4.16 一、Agent 相关工作 二、Agent 特点 核心特征: 1.专有场景(针对某个垂直领域) 2.保留记忆(以一个特定顺序做一些特定任务,记忆当前任务的前…...
React 打包
路由懒加载 原本的加载方式 #使用lazy()函数声明的路由页面 使用Suspense组件进行加载 使用CDN优化...
2025.4.14-2025.4.20学习周报
目录 摘要Abstract1. 文献阅读1.1 模型架构1.2 实验分析1.3 代码实践 总结 摘要 在本周阅读的论文中,作者提出了一种名为MGSFformer的空气质量预测模型。模型通过残差去冗余模块可以有效解耦多粒度数据间的信息重叠;时空注意力模块采用并行建模策略&…...
Spring 微服务解决了单体架构的哪些痛点?
1. 部署困难 (Deployment Difficulty & Risk) 单体痛点: 整体部署: 对单体应用的任何微小修改(哪怕只是一行代码),都需要重新构建、测试和部署整个庞大的应用程序。部署频率低: 由于部署过程复杂且风险高,发布周期通常很长&a…...
【1】云原生,kubernetes 与 Docker 的关系
Kubernetes?K8s? Kubernetes经常被写作K8s。其中的数字8替代了K和s中的8个字母——这一点倒是方便了发推,也方便了像我这样懒惰的人。 什么是云原生? 云原生: 它是一种构建和运行应用程序的方法,它包含&am…...
Kubernetes控制平面组件:APIServer 限流机制详解
云原生学习路线导航页(持续更新中) kubernetes学习系列快捷链接 Kubernetes架构原则和对象设计(一)Kubernetes架构原则和对象设计(二)Kubernetes架构原则和对象设计(三)Kubernetes控…...
springboot全局异常捕获处理
一、需求 实际项目中,经常抛出各种异常,不能直接抛出异常给前端,这样用户体验相当不好,用户看不懂你的Exception,对于一些sql异常,直接抛到页面上也不安全。所以有没有好的办法解决这些问题呢,当然有了&am…...
Flask(1): 在windows系统上部署项目1
1 前言 学习python也有段时间了,最近一个小项目要部署,正好把过程写下来。 在程序的结构上我选择了w/s模式,相比于c/s模式,无需考虑客户端的升级;框架我选择了flask,就是冲着轻量级去的,就是插件…...
【文献阅读】EndoNet A Deep Architecture for Recognition Tasks on Laparoscopic Videos
关于数据集的整理 Cholec80 胆囊切除手术视频数据集介绍 https://zhuanlan.zhihu.com/p/700024359 数据集信息 Cholec80 数据集 是一个针对内窥镜引导 下的胆囊切除手术视频流程识别数据集。数据集提供了每段视频中总共7种手术动作及总共7种手术工具的标注,标…...
基于springboot的个人财务管理系统的设计与实现
博主介绍:java高级开发,从事互联网行业六年,熟悉各种主流语言,精通java、python、php、爬虫、web开发,已经做了六年的毕业设计程序开发,开发过上千套毕业设计程序,没有什么华丽的语言࿰…...
Linux系统编程---孤儿进程与僵尸进程
1、前言 在上一篇博客文章已经对Linux系统编程内容进行了较为详细的梳理,本文将在上一篇的基础上,继续梳理Linux系统编程中关于孤儿进程和僵尸进程的知识脉络。如有疑问的博客朋友可以通过下面的博文链接进行参考学习。 Linux系统编程---多进程-CSDN博客…...
简单使用MCP
简单使用MCP 1 简介 模型上下文协议(Model Context Protocol,MCP)是由Anthropic(产品是Claude)推出的开放协议,它规范了应用程序如何向LLM提供上下文。MCP可帮助你在LLM之上构建代理和复杂的工作流。 从…...
Semaphore的核心机制
在 Java 中,Semaphore 通过 许可计数器 和 同步队列 的机制实现并发线程数的限制。以下是其核心实现原理和步骤的详细分析: 一、核心机制 许可计数器(Permits) • 初始化时指定的许可数(如 new Semaphore(3)࿰…...
计算机视觉与深度学习 | RNN原理,公式,代码,应用
RNN(循环神经网络)详解 一、原理 RNN(Recurrent Neural Network)是一种处理序列数据的神经网络,其核心思想是通过循环连接(隐藏状态)捕捉序列中的时序信息。每个时间步的隐藏状态 ( h_t ) 不仅依赖当前输入 ( x_t ),还依赖前一时间步的隐藏状态 ( h_{t-1} ),从而实现…...
Keil MDK 编译问题:last line of file ends without a newline
问题与处理策略 问题描述 ..\..\User\main.c(38): warning: #1-D: last line of file ends without a newline} ..\..\User\main.c: 1 warning, 0 errors问题原因 这是文件末尾缺少换行符警告 处理策略 在文件(main.c)的最后一行按回车键添加一个空…...
MySQL:9.表的内连和外连
9.表的内连和外连 表的连接分为内连和外连 9.1 内连接 内连接实际上就是利用where子句对两种表形成的笛卡儿积进行筛选,之前查询都是内连 接,也是在开发过程中使用的最多的连接查询。 语法: select 字段 from 表1 inner join 表2 on 连接…...
C++栈操作集合
数组 #include <bits/stdc.h> using namespace std;class sss{ private:int a[1000];int curr -1; public:void push(int);void pop();int top();bool empyt();int size(); };int main() {sss n;while(true){int a;cout<<"1.添加\n2.删除-\n3.显示栈顶\n4.储…...
在阿里云和树莓派上编写一个守护进程程序
目录 一、阿里云邮件守护进程 1. 安装必要库 2. 创建邮件发送脚本 mail_daemon.py 3. 设置后台运行 二、树莓派串口守护进程 1. 启用树莓派串口 2. 安装依赖库 3. 创建串口输出脚本 serial_daemon.py 4. 设置开机自启 5. 使用串口助手接收 一、阿里云邮件守护进程 1.…...
每日一题——最小测试用例集覆盖问题
最小测试用例集覆盖问题(C语言实现) 问题描述 假设我们有一系列测试用例,每个测试用例会覆盖若干个代码模块。 我们使用一个二维数组来表示这些测试用例的覆盖情况: 如果某个测试用例 i 能覆盖代码模块 j,则数组中…...
LangChain 单智能体模式示例【纯代码】
# LangChain 单智能体模式示例import os from typing import Anyfrom langchain.agents import AgentType, initialize_agent, Tool from langchain_openai import ChatOpenAI from langchain.tools import BaseTool from langchain_experimental.tools.python.tool import Pyt…...
基于前端技术的QR码API开发实战:从原理到部署
前言 QR码(Quick Response Code)是一种二维码,于1994年开发。它能快速存储和识别数据,包含黑白方块图案,常用于扫描获取信息。QR码具有高容错性和快速读取的优点,广泛应用于广告、支付、物流等领域。通过扫…...
RenderStage::drawInner
文章目录 RenderStage::drawInnerOSG渲染后台关系图OSG的渲染流程RenderBin::draw(renderInfo,previous)RenderBin::drawImplementationRenderLeaf::renderosg::State::apply(const StateSet*)Drawable::draw(RenderInfo& renderInfo)Drawable::drawInner(RenderInfo& …...
C++初阶-类和对象(中)
目录 1.类的默认成员函数 2.构造函数(难度较高) 编辑 编辑 编辑 3.析构函数 4.拷贝构造函数 5.赋值运算符重载 5.1运算符重载 5.2赋值运算符重载 6.取地址运算符重载 6.1const成员函数 6.2取地址运算符重载 7.总结 1.类的默认成员函数…...
智谱开源新一代GLM模型,全面布局AI智能体生态
2024年4月15日,智谱在中关村论坛上正式发布了全球首个集深度研究与实际操作能力于一体的AI智能体——AutoGLM沉思。这一革命性技术的发布标志着智谱在AGI(通用人工智能)领域的又一次重要突破。智谱的最新模型不仅推动了AI智能体技术的升级&am…...
Vue3中provide和inject的用法示例
在 Vue3 中,provide 和 inject 用于实现跨层级组件通信。以下是一个简单的示例: 1. 父组件 (祖先组件) - 提供数据 javascript 复制 // ParentComponent.vue import { provide, ref, reactive } from vue;export default {setup() {// 提供静态数据p…...
