【c++随笔16】reserve之后,使用std::copy会崩溃?

news2024/11/26 12:46:13

【c++随笔16】reserve之后,使用std::copy会崩溃?

  • 一、reserve之后,使用std::copy会崩溃?
  • 二、函数std::reserve、std::resize、std::copy
    • 1、std::resize:
    • 2、std::reserve:
    • 3、std::copy:
  • 三、崩溃原因分析
    • 方案1、你可以使用 std::back_inserter 插入迭代器来向 destination 中插入元素,代码如下:
    • 方案2、resize将容器大小调整同时,新增的元素将会被默认初始化为0,这样std::copy中的destination.begin()就可用访问到元素了

原创作者:郑同学的笔记
原创地址:https://zhengjunxue.blog.csdn.net/article/details/134272501

一、reserve之后,使用std::copy会崩溃?

先看代码

#include <iostream>
#include <vector>
#include <algorithm>

int main() {
    std::vector<int> source = { 1, 2, 3, 4, 5 };
    std::vector<int> destination;

    // 在目标vector中预留足够的空间,避免不必要的重新分配
    destination.reserve(source.size());

    // 使用std::copy将source中的元素复制到destination中
    std::copy(source.begin(), source.end(), destination.begin());

    // 打印复制后的destination内容
    for (int num : destination) {
        std::cout << num << " ";
    }
    std::cout << std::endl;

    return 0;
}

运行,崩溃截图如下
在这里插入图片描述

我们先简单说下std::reserve、std::resize、std::copy三个函数的用法,然后再说崩溃的原因。

二、函数std::reserve、std::resize、std::copy

1、std::resize:

  • std::resize 用于改变容器的大小,并可以指定新元素的初值。
  • 如果新的大小大于当前大小,则在末尾添加新元素;如果新的大小小于当前大小,则删除多余的元素。
  • 使用 resize 会改变容器内的元素个数,并可能导致元素值的改变。

示例代码如下:

std::vector<int> vec = {1, 2, 3};
vec.resize(5); // 将容器大小调整为 5,新增的元素将会被默认初始化为0
vec.resize(3); // 将容器大小调整为 3,多余的元素将会被删除

2、std::reserve:

  • std::reserve 用于预留容器的空间,但不会改变容器的大小。
  • 当你知道容器将要存储大量元素时,可以使用 reserve 来提前分配足够的内存,避免容器多次扩张操作。
  • 使用 reserve 不会改变容器中的元素个数,只是预分配了足够的空间。

示例代码如下:

std::vector<int> vec;
vec.reserve(100); // 预留至少能容纳100个元素的空间

3、std::copy:

template <class InputIterator, class OutputIterator>
OutputIterator copy(InputIterator first, InputIterator last, OutputIterator result);

std::copy 用于将一个范围内的元素复制到另一个范围,它可以用于将一个容器中的元素复制到另一个容器中

注意事项

  • 确保输出范围有足够的空间来容纳被复制的元素,否则可能会导致未定义的行为。
  • 当处理容器时,确保输出容器有足够的容量,或者使用插入迭代器(如 std::back_inserter)来确保动态分配足够的空间。
int source[] = {1, 2, 3, 4, 5};
int destination[5];

// 使用 std::copy 将 source 数组的内容复制到 destination 数组
std::copy(std::begin(source), std::end(source), std::begin(destination));

三、崩溃原因分析

  • 在这个代码中,当你使用 std::copy 将 source 中的元素复制到 destination 中时,会导致程序崩溃的原因是因为 destination 容器的空间虽然预留了,但是其中并没有实际的元素,所以无法直接通过 destination.begin() 来访问 destination 的首个元素。

  • 在使用 std::copy 进行复制时,目标容器必须有足够的空间来容纳被复制的元素,并且使用 std::back_inserter 或者确保目标容器大小与源容器相同的方式来进行插入操作。如果目标容器没有足够的空间或者没有正确的插入迭代器,就会导致未定义的行为,可能导致程序崩溃。

要修复这个问题,有两个方法你可以使用

方案1、你可以使用 std::back_inserter 插入迭代器来向 destination 中插入元素,代码如下:

#include <iostream>
#include <vector>
#include <algorithm>

int main() {
    std::vector<int> source = { 1, 2, 3, 4, 5 };
    std::vector<int> destination;

    // 在目标vector中预留足够的空间,避免不必要的重新分配
    destination.reserve(source.size());

    // 使用std::copy将source中的元素复制到destination中
    std::copy(source.begin(), source.end(), std::back_inserter(destination));

    // 打印复制后的destination内容
    for (int num : destination) {
        std::cout << num << " ";
    }
    std::cout << std::endl;

    return 0;
}

方案2、resize将容器大小调整同时,新增的元素将会被默认初始化为0,这样std::copy中的destination.begin()就可用访问到元素了

#include <iostream>
#include <vector>
#include <algorithm>

int main() {
    std::vector<int> source = { 1, 2, 3, 4, 5 };
    std::vector<int> destination;

    // 在目标vector中预留足够的空间,避免不必要的重新分配
    destination.resize(source.size());

    // 使用std::copy将source中的元素复制到destination中
    std::copy(source.begin(), source.end(), destination.begin());

    // 打印复制后的destination内容
    for (int num : destination) {
        std::cout << num << " ";
    }
    std::cout << std::endl;

    return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1301058.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【概率方法】重要性采样

从一个极简分布出发 假设我们有一个关于随机变量 X X X 的函数 f ( X ) f(X) f(X)&#xff0c;满足如下分布 p ( X ) p(X) p(X)0.90.1 f ( X ) f(X) f(X)0.10.9 如果我们要对 f ( X ) f(X) f(X) 的期望 E p [ f ( X ) ] \mathbb{E}_p[f(X)] Ep​[f(X)] 进行估计&#xff0…

极速学习SSM之SpringMVC笔记

文章目录 一、SpringMVC简介1、什么是MVC2、什么是SpringMVC3、SpringMVC的特点 二、HelloWorld1、开发环境2、创建maven工程a>添加web模块b>打包方式&#xff1a;warc>引入依赖 3、配置web.xmla>默认配置方式b>扩展配置方式 4、创建请求控制器5、创建springMVC…

ansible中的角色

1.理解roles在企业中的定位及写法 查看创建目录结构 ansible - galaxy list 指定新的位置 vim ansible.cfg roles_path ~/.ansible/roles 建立项目 cd roles/ ansible-galaxy init vsftpd tree vsftpd/ 编辑任务执行&#xff08;顺序&#xff09;文件 vim vsftpd/tas…

selenium库的使用

来都来了给我点个赞收藏一下再走呗&#x1f339;&#x1f339;&#x1f339;&#x1f339;&#x1f339; 目录 一、下载需要用到的python库selenium 二、selenium的基本使用 1.在python代码引入库 2.打开浏览器 3.元素定位 1&#xff09;通过id定位 2&#xff09;通过标…

Go开发运维:Go服务发布到K8S集群

目录 一、实验 1.Go服务发布到k8s集群 二、问题 1.如何从Harbor拉取镜像 一、实验 1.Go服务发布到k8s集群 &#xff08;1&#xff09;linux机器安装go(基于CentOS 7系统) yum install go -y &#xff08;2&#xff09;查看版本 go version &#xff08;3&#xff09;创…

UE引擎 LandscapeGrass 实现机制分析(UE5.2)

前言 随着电脑和手机硬件性能越来越高&#xff0c;游戏越来越追求大世界&#xff0c;而大世界非常核心的一环是植被&#xff0c;目前UE5引擎提供给植被生成的主流两种方式为 手刷植被和LandscapeGrass(WeightMap程序化植被)。当然UE5.3推出新一代PCGFramework 节点程序化生成框…

办公word-从不是第一页添加页码

总结 实际需要注意的是&#xff0c;分隔符、分节符和分页符并不是一个含义 分隔符包含其他两个&#xff1b;分页符&#xff1a;是增加一页&#xff1b;分节符&#xff1a;指将文档分为几部分。 从不是第一页插入页码1步骤 1&#xff0c;插入默认页码 自己可以测试时通过**…

一文掌握Ascend C孪生调试

1 What&#xff0c;什么是孪生调试 Ascend C提供孪生调试方法&#xff0c;即CPU域模拟NPU域的行为&#xff0c;相同的算子代码可以在CPU域调试精度&#xff0c;NPU域调试性能。孪生调试的整体方案如下&#xff1a;开发者通过调用Ascend C类库编写Ascend C算子kernel侧源码&am…

784. 字母大小写全排列

字母大小写全排列 描述 : 给定一个字符串 s &#xff0c;通过将字符串 s 中的每个字母转变大小写&#xff0c;我们可以获得一个新的字符串。 返回 所有可能得到的字符串集合 。以 任意顺序 返回输出。 题目 : LeetCode 784.字母大小写全排列 : 784. 字母大小写全排列 分析…

JavaScript常用技巧专题二

文章目录 一、前言二、生成随机字符串三、转义HTML特殊字符四、单词首字母大写五、将字符串转换为小驼峰六、删除数组中的重复值七、移除数组中的假值八、获取两个数字之间的随机数九、将数字截断到固定的小数点十、日期10.1、计算两个日期之间天数10.2、从日期中获取是一年中的…

fd信息查看

一、/proc/pid/fdinfo和/proc/pid/fd的含义 二、实例 ref&#xff1a; linux下的/proc/pid/fdinfo和/proc/pid/fd_proc/417/fdinfo/0-CSDN博客 linux select read阻塞_图解Linux的IO模型和相关技术-CSDN博客

实现加盐加密方法以及java nio中基于MappedByteBuffer操作大文件

自己实现 传统MD5可通过彩虹表暴力破解&#xff0c; 加盐加密算法是一种常用的密码保护方法&#xff0c;它将一个随机字符串&#xff08;盐&#xff09;添加到原始密码中&#xff0c;然后再进行加密处理。 1. 每次调用方法产生一个唯一盐值&#xff08;UUID &#xff09;密码…

生产问题: 利用线程Thread预加载数据缓存,其它类全局变量获取缓存偶发加载不到

生产问题: 利用线程Thread预加载数据缓存偶发加载不到 先上代码 public class ThreadTest {//本地缓存Map<String, Object> map new HashMap<String, Object>();class ThreadA implements Runnable{Overridepublic void run() {System.out.println("Thread…

【崩坏:星穹铁道】卡芙卡狂喜!1.5版本新遗器值不值得刷?

《崩坏&#xff1a;星穹铁道》在1.5版本上线了4套新的遗器套装&#xff0c;照例还是两套外圈遗器两套内圈遗器&#xff0c;分别可以通过幽冥之径侵蚀隧洞副本和模拟宇宙第八世界刷取。 那么本期闲游盒小盒子就简单解析一下这4套新遗器&#xff0c;它们的属性适合哪些角色&#…

2021版吴恩达深度学习课程Deeplearning.ai 05序列模型 12.5

学习内容 05.序列模型 1.1 为什么用序列模型 1.序列模型常见的应用 1.2 注释 notation 1.*T_x(i)表示训练样本x(i)的序列长度&#xff0c;T_y(i)表示target(i)的序列长度2.训练集表示单词的方式*构建字典的方式*在训练集中查找出现频率最高的单词*网络搜集常用字典3.如果遇…

微服务——服务保护Sentinel

雪崩问题 在单体项目里面&#xff0c;如果某一个模块出问题会导致整个项目都有问题。 在微服务项目里面&#xff0c;单独一个服务出问题理论上是不会影响别的服务的。 但是如果有别的业务需要调用这一个模块的话还是会有问题。 问题产生原因和解决思路 最初那只是一个小小…

GEE影像升尺度(10m->250m)

GEE影像升尺度&#xff08;10m->250m&#xff09; 代码 var ext /* color: #d63000 *//* shown: false *//* displayProperties: [{"type": "rectangle"}] */ee.Geometry.Polygon([[[108.74625980473367, 28.562445155322063],[108.74625980473367, …

云计算在计算机领域的应用与发展

云计算在计算机领域的应用与发展 一、引言 随着科技的不断发展&#xff0c;计算机领域已经成为当今社会最为活跃和创新的领域之一。云计算作为一种新兴的计算模式&#xff0c;已经在计算机领域中得到了广泛的应用&#xff0c;并且正在不断地推动着计算机领域的发展。本文将探…

令牌桶算法理解学习(限流算法)

令牌桶算法是网络流量整形&#xff08;Traffic Shaping&#xff09;和速率限制&#xff08;Rate Limiting&#xff09;中最常使用的一种算法。典型情况下&#xff0c;令牌桶算法用来控制发送到网络上的数据的数目&#xff0c;并允许突发数据的发送。 用简单的话语来说就是限制…

基于MyBatis二级缓存深入装饰器模式

视频地址 学习文档 文章目录 一、示意代码二、装饰器三、经典案例—MyBatis二级缓存1、Cache 标准定义2、PerpetualCache 基础实现3、增强实现3-1、ScheduledCache3-2、LruCache 先来说说我对装饰器理解&#xff1a;当你有一个基础功能的代码&#xff0c;但你想在不改变原来代…