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

C++ vector容器存储对象和存储指针的区别(vector对象、vector指针)(存储指针时推荐使用智能指针)

文章目录

      • **1. 内存管理**
        • - **存储对象**:
        • - **存储指针**:
      • **2. 生命周期控制**
        • - **存储对象**:
        • - **存储指针**:
      • **3. 性能差异**
        • - **存储对象**:
        • - **存储指针**:
      • **4. 使用场景**
        • - **选择存储对象的情况**:
        • - **选择存储指针的情况**:
      • **5. 注意事项**
        • - **存储指针的风险**:
        • - **推荐实践**:
          • - 使用智能指针(如 `std::unique_ptr`、`std::shared_ptr`)替代原始指针,自动管理内存:
      • **6. 总结对比表**
      • **7. 示例代码**
        • - **存储对象**:
        • - **存储指针**:

在 C++ 中, std::vector 容器存储对象(如 int、自定义类实例)和指针(如 int*MyClass*)时,存在显著差异,主要体现在 内存管理、生命周期控制、性能开销使用场景 上。以下是详细对比:


1. 内存管理

- 存储对象
  • vector 管理对象的完整生命周期。
  • vector 被销毁或调用 clear() 时,所有对象会自动调用析构函数,内存被释放。
  • 示例:
    std::vector<MyClass> vec;
    vec.push_back(MyClass()); // 存储对象
    vec.clear();              // 自动释放所有对象内存
    
- 存储指针
  • vector 仅管理指针本身的生命周期,不负责释放指针指向的对象
  • 需要手动遍历 vectordelete 每个指针指向的对象,否则会导致内存泄漏。
  • 示例:
    std::vector<MyClass*> vec;
    vec.push_back(new MyClass()); // 存储指针
    for (auto* ptr : vec) delete ptr; // 手动释放内存
    vec.clear();                    // 仅释放指针本身
    

2. 生命周期控制

- 存储对象
  • vector 的扩容(如 push_back 超过容量时)会复制或移动对象到新内存区域,原对象被析构。
  • 示例:
    std::vector<std::string> vec;
    vec.push_back("hello"); // 如果扩容,"hello" 会被复制到新内存
    
- 存储指针
  • vector 扩容时仅复制指针(地址),不会影响指针指向的对象
  • 指针指向的对象生命周期需独立管理,与 vector 无关。
  • 示例:
    std::vector<MyClass*> vec;
    MyClass* obj = new MyClass();
    vec.push_back(obj); // 存储指针
    // 即使 vec 扩容,obj 的内存地址不会改变
    

3. 性能差异

- 存储对象
  • 优点:直接存储对象,访问效率高(连续内存,缓存友好)。
  • 缺点:对象较大时,频繁扩容可能导致大量复制开销(需调用拷贝构造函数)。
  • 适合场景:对象较小或频繁访问,且不需要动态分配内存。
- 存储指针
  • 优点:避免对象复制开销(仅复制指针地址)。
  • 缺点:需要手动管理内存(new/delete),且指针访问可能因内存不连续导致缓存效率低。
  • 适合场景:对象较大或需要共享所有权(如多 vector 共同管理同一对象)。

4. 使用场景

- 选择存储对象的情况
  • 对象生命周期与 vector 绑定,无需共享。
  • 需要频繁访问或修改对象,且对象较小。
  • 示例:存储基本类型(intfloat)或小型结构体。
- 选择存储指针的情况
  • 对象需要动态分配(如通过 new 创建)。
  • 需要共享对象所有权(多个 vector 或其他容器指向同一对象)。
  • 对象较大,复制成本高。
  • 示例:存储多态对象(基类指针指向派生类实例)。

5. 注意事项

- 存储指针的风险
  1. 内存泄漏:忘记 delete 指针会导致内存泄漏。
  2. 悬空指针:若 vector 管理的指针指向的对象被提前释放,后续访问会导致未定义行为。
  3. 重复释放:如知识库 [1] 提到的 double free 错误(例如在类析构函数中调用 vector<T>().swap() 释放指针)。
- 推荐实践
- 使用智能指针(如 std::unique_ptrstd::shared_ptr)替代原始指针,自动管理内存:
std::vector<std::unique_ptr<MyClass>> vec;
vec.push_back(std::make_unique<MyClass>());
// 不需要手动 delete,vec 析构时自动释放

6. 总结对比表

特性存储对象存储指针
内存管理自动释放需手动 delete
生命周期绑定vector 绑定独立于 vector
扩容开销复制对象(调用拷贝/移动构造函数)仅复制指针地址
访问效率高(连续内存)低(间接访问)
适用场景对象小、生命周期短对象大、需要共享所有权或动态分配
风险无(除非自定义析构逻辑)内存泄漏、悬空指针、重复释放

7. 示例代码

- 存储对象
#include <vector>
struct MyClass {MyClass() { printf("Constructor\n"); }~MyClass() { printf("Destructor\n"); }
};
int main() {std::vector<MyClass> vec;vec.push_back(MyClass()); // 对象被复制到 vector 中vec.clear();              // 自动调用析构函数
}
- 存储指针
#include <vector>
struct MyClass {MyClass() { printf("Constructor\n"); }~MyClass() { printf("Destructor\n"); }
};
int main() {std::vector<MyClass*> vec;vec.push_back(new MyClass()); // 手动分配内存for (auto* ptr : vec) delete ptr; // 必须手动释放vec.clear();
}

通过以上对比,可以根据具体需求选择存储对象或指针。在现代 C++ 中,优先推荐使用智能指针来避免手动内存管理的风险。

相关文章:

C++ vector容器存储对象和存储指针的区别(vector对象、vector指针)(存储指针时推荐使用智能指针)

文章目录 **1. 内存管理**- **存储对象**&#xff1a;- **存储指针**&#xff1a; **2. 生命周期控制**- **存储对象**&#xff1a;- **存储指针**&#xff1a; **3. 性能差异**- **存储对象**&#xff1a;- **存储指针**&#xff1a; **4. 使用场景**- **选择存储对象的情况**…...

C#和C++在编译过程中的文件区分

1. .h是头文件&#xff08;Header File&#xff09; 用来 声明类、函数、常量等。 通常不包含实际实现&#xff0c;只是“定义接口” // 示例&#xff1a;math_utils.h#pragma once int add(int a, int b); //定义函数名2. .cpp是源文件&#xff08;Source File&…...

50天50个小项目 (Vue3 + Tailwindcss V4) ✨ | Dad Jokes(冷笑话卡片)

&#x1f4c5; 我们继续 50 个小项目挑战&#xff01;—— DadJokes 组件 仓库地址&#xff1a;https://github.com/SunACong/50-vue-projects 项目预览地址&#xff1a;https://50-vue-projects.vercel.app/ 豆包翻译确实可以&#xff0c;冷笑话应该属于各类语言比较难理解的…...

Spring AOP执行原理源码解析

对【com.example.demo.TestAspect#aopTest】连接点增加了五个通知 在调用【com.example.demo.A#testAop()】&#xff08;用户自定义&#xff09;方法时&#xff0c;Cglib拦截器对其进行了拦截 可以看到执行顺序分别是环绕前置&#xff0c;前置&#xff0c;环绕后置&#xff0c;…...

基于FPGA的超声波显示水位距离,通过蓝牙传输水位数据到手机,同时支持RAM存储水位数据,读取数据。

基于FPGA的超声波显示水位距离 前言一、整体框架二、代码架构1.超声波测距模块2.蓝牙数据发送模块3.数码管数据切换模块4.数码管驱动模块6.串口驱动7.顶层模块8.RAM ip核 仿真相关截图 前言 随着工业化进程的加速和环境保护意识的提升&#xff0c;对水资源管理和水位监测的需求…...

使用swoole作为MQTT客户端并接收实现即时消息推送

环境准备 首先需要安装swoole 可以使用pecl进行安装 &#xff0c;如 pecl install swool, 注意加上版本号 或者使用构建好的docker镜像&#xff0c;这里使用构建好的 zacksleo/php:7.1-alpine-fpm-swoole 镜像 使用 compose 安装依赖库 composer require jesusslim/mqttcl…...

在Windows下利用LoongArch-toolchain交叉编译Qt

文章目录 0.交叉编译的必要性1.下载交叉编译工具链1.1.直接在Windows下使用mingw&#xff08;不使用虚拟机&#xff09;编译&#xff08;还没成功&#xff0c;无法编译&#xff09;1.2.在虚拟机中的Ubuntu中进行交叉编译 2.下载qt源码3.编译Qt3.1.创建loongarch64的mkspec3.2.创…...

如何在 React 中监听 div 的滚动事件

在 React 中监听 div 的滚动事件&#xff08;scroll&#xff09;&#xff0c;可以通过为该 div 添加 onScroll 属性来实现。以下是一个基本的例子&#xff1a; ✅ 示例&#xff1a;监听 div 的滚动事件 import React, { useRef } from react;function ScrollComponent() {cons…...

AIRIOT无人机安防解决方案

随着无人机技术的飞速发展和广泛应用&#xff0c;其在安防领域的价值日益凸显&#xff0c;从关键设施巡检、大型活动安保到边境巡防、应急救援&#xff0c;无人机正成为立体化安防体系不可或缺的“空中哨兵”。然而&#xff0c;无人机安防应用蓬勃发展的同时&#xff0c;其自身…...

华为OD机考 - 水仙花数 Ⅰ(2025B卷 100分)

import java.util.*; public static Integer get(int count,int c){if(count<3||count>7){return -1;}//存储每位数的最高位……最低位int[] arr new int[count];List<Integer> res new ArrayList<>();for(int i(int) Math.pow(10,count-1);i<(int) Math…...

软考 系统架构设计师系列知识点之杂项集萃(81)

接前一篇文章&#xff1a;软考 系统架构设计师系列知识点之杂项集萃&#xff08;80&#xff09; 第145题 商业智能是企业对商业数据的搜集、管理和分析的系统过程&#xff0c;主要技术包括&#xff08;&#xff09;。 A. 数据仓库、联机分析和数据挖掘 B. 数据采集、数据清洗…...

php apache构建 Web 服务器

虚拟机配置流程winsever2016配置Apache、Mysql、php_windows server 2016配置web服务器-CSDN博客 PHP 和 Apache 通过 ​​模块化协作​​ 共同构建 Web 服务器&#xff0c;以下是它们的交互机制和工作流程&#xff1a; ​​一、核心组件分工​​ 组件角色​​Apache​​Web …...

Ntfs!ReadIndexBuffer函数分析之nt!CcGetVirtualAddress函数之nt!CcGetVacbMiss

第一部分&#xff1a; NtfsMapStream( IrpContext, Scb, LlBytesFromIndexBlocks( IndexBlock, Scb->ScbType.Index.IndexBlockByteShift ), Scb->ScbType.Index.BytesPerIndexBuffer, &am…...

Vue3 + TypeSrcipt 防抖、防止重复点击实例

需要实现防抖应用场景&#xff1a; 点击【查询】按钮&#xff0c;发送网络请求&#xff0c;等待并接收响应数据 原来点击【查询】的代码&#xff1a; <script setup lang"ts" name"ReagentTransactionsDrawer"> ...... // 查询&#xff0c;没有防…...

SQL进阶之旅 Day 14:数据透视与行列转换技巧

【SQL进阶之旅 Day 14】数据透视与行列转换技巧 开篇 欢迎来到“SQL进阶之旅”系列的第14天&#xff01;今天我们将探讨数据透视与行列转换技巧&#xff0c;这是数据分析和报表生成中的核心技能。无论你是数据库开发工程师、数据分析师还是后端开发人员&#xff0c;行转列或列…...

打通印染车间“神经末梢”:DeviceNet转Ethernet/IP连接机器人的高效方案

在印染行业自动化升级中&#xff0c;设备联网需求迫切。老旧印染设备多采用Devicenet协议&#xff0c;而新型工业机器人普遍支持Ethernet/IP协议&#xff0c;协议不兼容导致数据交互困难&#xff0c;设备协同效率低、生产监控滞后&#xff0c;成了行业数字化转型的阻碍。本文将…...

Ubuntu挂载本地镜像源(像CentOS 一样挂载本地镜像源)

1.挂载 ISO 镜像 sudo mount -o loop /ubuntu-22.04.5-desktop-amd64.iso /mnt/iso 2.备份现有的软件源配置文件&#xff1a; sudo cp /etc/apt/sources.list /etc/apt/sources.list.bak 3.编辑软件源配置文件 编辑 /etc/apt/sources.list sudo nano /etc/apt/sources.l…...

2025-06-02-IP 地址规划及案例分析

IP 地址规划及案例分析 参考资料 Plan for IP addressing - Cloud Adoption Frameworkwww.cnblogs.comimage-hosting/articles at master jonsam-ng/image-hosting 概述 在网络通信中&#xff0c;MAC 地址与 IP 地址分别位于 OSI 模型的数据链路层和网络层&#xff0c;二者协…...

AUTOSAR实战教程--开放式通用DoIP刷写工具OpenOTA开发计划

目录 软件概述 安装与运行 界面说明 3.1 功能区划分 3.2 状态显示 基本操作流程 4.1 DoIP连接配置 4.2 服务配置&#xff08;刷写流程&#xff09; 4.3 执行操作 4.4 保存配置 4.5 加载配置 功能详解 5.1 核心功能模块 诊断服务配置 通信设置 文件下载 工具功…...

Vue 学习路线图(从零到实战)

&#x1f3af; 学习目标&#xff1a;掌握 Vue 并能独立开发中大型项目 ✅ 适合人群&#xff1a;前端初学者、想快速上手做项目的开发者、中小型团队成员 &#x1f9ed; Vue 学习路线图&#xff08;从零到实战&#xff09; 第一阶段&#xff1a;基础语法 核心功能&#xff08;…...

AI赋能的浏览器自动化:Playwright MCP安装配置与实操案例

以下是对Playwright MCP的简单介绍&#xff1a; Playwright MCP 是一个基于 Playwright 的 MCP 工具&#xff0c;提供浏览器自动化功能不要求视觉模型支持&#xff0c;普通的文本大语言模型就可以通过结构化数据与网页交互支持多种浏览器操作&#xff0c;包括截图、点击、拖动…...

AI编程助手入门指南:GitHub Copilot、Cursor与Claude的安装与基础使用

&#x1f525; AI编程助手入门指南&#xff1a;GitHub Copilot、Cursor与Claude的安装与基础使用 你是否曾幻想过拥有一个24小时在线的编程搭档&#xff1f;它能理解你的思路、自动补全代码、解释复杂逻辑&#xff0c;甚至帮你调试错误&#xff1f;如今&#xff0c;这个幻想已成…...

Android 线性布局中常见的冲突属性总结

1. gravity vs layout_gravity 冲突原因&#xff1a;两者作用对象不同&#xff0c;混用会导致行为异常。 区别&#xff1a; android:gravity&#xff1a;父容器的属性&#xff0c;控制子元素在容器内的对齐方式。android:layout_gravity&#xff1a;子元素的属性&#xff0c;控…...

【技术笔记】MSYS2 指定 Python 版本安装方案

#工作记录 MSYS2 指定 Python 版本安装 一、前置条件 安装指定版本需要在干净的 MSYS2 环境中执行&#xff0c;为保证工具链的兼容性&#xff0c;若已安装 Python&#xff0c;需先卸载 Python 及与该版本深度绑定的工具链。具体操作如下&#xff1a; 卸载 Python&#xff1a…...

《校园生活平台从 0 到 1 的搭建》第一篇:创建项目与构建目录结构

在本系列第一篇中&#xff0c;我们将从项目初始化开始&#xff0c;搭建基本的目录结构&#xff0c;并完成四个主页面的创建与 TabBar 设置。 &#xff08;tip&#xff1a;你可能会觉得有点 ai 化&#xff0c;因为这个文案是我自己写了一遍文案之后让 ai 去优化输出的&#xff0…...

Boost ASIO 库深入学习(3)

Boost ASIO 库深入学习&#xff08;3&#xff09; UDP简单通信导论 在继续深入前&#xff0c;我们不妨也来点碎碎念&#xff0c;因为UDP通信协议的模型与TCP是不同的&#xff0c;这种差异正是理解“无连接通信”的关键所在。我们下面要构建的&#xff0c;是一个经典的UDP通信…...

【如何做好应用架构?】

一、应用架构定义 应用架构描述了各种用于支持业务架构并对数据架构所定义的各种数据进行出来的应用功能。这些应该功能指的是用来管理在数据架构中定义的数据&#xff0c;并对业务架构中定义的各项业务功能进行支持的能力。 其核心目标是确保应用系统高效、灵活、安全的支撑…...

1 Studying《蓝牙核心规范5.3》

目录 [Vol 0][Part B 蓝牙规范要求] 3 定义 3.1 蓝牙产品类型 4 核心配置 4.1 基本速率核心配置 4.2 增强型数据速率核心配置 4.4 低功耗核心配置 4.5 基本速率和低功耗结合的核心配置 4.6 主机控制器接口核心配置 [Vol 1][Part A 架构]1 概述 1.1 BR/EDR操作概述 …...

STM32+MPU6050传感器

#创作灵感## 在嵌入式系统开发中&#xff0c;STM32F103C8T6单片机与MPU6050传感器的组合因其高性能、低功耗以及丰富的功能而备受青睐。本文将简单介绍如何在Keil 5开发环境中实现STM32F103C8T6与MPU6050的连接和基本数据采集&#xff0c;带你快速入门智能硬件开发。 一、硬件…...

el-input限制输入数字,输入中文后数字校验失效

想要的效果&#xff1a;默认值为0&#xff0c;只能输入0-100的数字。 实现方式如下&#xff0c;使用 οnkeyup"this.valuethis.value.replace(/\D/g,‘’)"限制只能输入数字&#xff0c;输入数字没有问题&#xff0c;使用input实现数字不以0开头&#xff0c;也只能是…...