Qt基础之四十一:记一次QVector的bad alloc问题追踪

本文详细探讨了QVector的特点,包括其动态数组的底层实现、内存分配策略,以及如何处理'bad alloc'问题。QVector在内存不足、内存碎片或非法参数时可能导致内存分配失败。在满容量后,QVector内存按2的n次方增长,但实际容量还需考虑QArrayData的数据成员占用。当添加元素超出2GB限制时,会出现内存分配失败的异常。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

目录

一.QVector的特点

二.QVector的内存分配策略

三.bad alloc问题追踪


一.QVector的特点

QVector 是 Qt 容器类库中的一个重要组成部分,它提供了类似于 C++ 标准库中的 std::vector 的功能,但同时拥有 Qt 独特的优势。QVector 是一个动态数组,支持添加、插入和删除元素。与其他容器相比,QVector 在随机访问元素时具有出色的性能,同时在尾部添加和删除元素时依然高效,不过在头部添加和删除元素的性能较差,此时需要移动大量元素以腾出空间或填补空缺。
1.底层实现:QVector 的底层实现基于动态数组。在内存中,QVector 的元素是连续存储的。这使得访问和遍历元素具有较高的效率。同时,QVector 支持 O(1) 复杂度的随机访问。在实际应用中,QVector 适用于对访问速度要求较高的场景。
2.内存分配:QVector 在内存管理上采用预分配策略。当向 QVector 中添加元素时,如果当前内存空间不足以容纳新元素,QVector 会重新分配一块更大的内存空间,并将原有元素复制到新的内存空间。为了减少内存分配和复制操作的开销,QVector 通常会预留一定的额外空间。这意味着 QVector 的实际容量(capacity)可能大于当前存储元素的数量(size)。
3.内存释放:当从 QVector 中删除元素时,QVector 通常不会立即释放内存。这是为了避免频繁的内存分配和释放操作。然而,在某些情况下,可能需要手动调整 QVector 的容量。此时,可以使用 squeeze() 函数来释放未使用的内存空间,或者使用 reserve() 函数来预留一定的内存空间。

我的代码报错terminate called after throwing an instance of 'std::bad_alloc' what(): std::bad_alloc,我写的代码如下:#include "mainwindow.h" #include "ui_mainwindow.h" #include <QtWidgets> #include <QResizeEvent> #include <QDebug> #include <QThread> MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent) , ui(new Ui::MainWindow) { ui->setupUi(this); replotTimer = new QTimer; ui->plot->setOpenGl(true); ui->plot->setNoAntialiasingOnDrag(true); mXDataTol = new QVector<double>; mYDataTol = new QVector<double>; mXDataTol->reserve(10000000); mYDataTol->reserve(10000000); CreateChart(); } MainWindow::~MainWindow() { delete ui; } void MainWindow::CreateChart() { ui->plot->addGraph(); ui->plot->graph(0)->setPen(QPen(QColor(100,149,237))); mData = ui->plot->graph(0)->data()->coreData(); ui->plot->xAxis2->setVisible(true); ui->plot->xAxis2->setTickLabels(false); ui->plot->yAxis2->setVisible(true); ui->plot->yAxis2->setTickLabels(false); connect(ui->plot->xAxis, SIGNAL(rangeChanged(QCPRange)), ui->plot->xAxis2, SLOT(setRange(QCPRange))); connect(ui->plot->yAxis, SIGNAL(rangeChanged(QCPRange)), ui->plot->yAxis2, SLOT(setRange(QCPRange))); ui->plot->setInteractions(QCP::iRangeDrag|QCP::iRangeZoom|QCP::iSelectPlottables); ui->plot->yAxis->setRange(-1.5,1.5); ui->plot->xAxis->setRange(0,10000000); ui->plot->yAxis->setLabel("数据"); ui->plot->xAxis->setLabel("个数"); for (int i=0;i<10000000;i++) { mXDataTol->append(i); if(i<5000000) mYDataTol->append(0); else mYDataTol->append(QRandomGenerator::global()->bounded(1.00)); } ui->plot->graph(0)->setData(*mXDataTol,*mYDataTol,true); }
06-09
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

草上爬

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值