我刚注意到QList
没有resize
方法,而QVector
例如有{{1}}方法。为什么是这样?是否有相同的功能?
答案 0 :(得分:10)
嗯,这是更通用的答案,但我希望您会看到,通过比较QList
和QVector
为什么不需要手动扩展容器。
QList使用内部缓冲区来保存指向元素的指针(或者,如果元素小于指针大小,或者元素是shared classes - 元素本身之一),真实数据将保留在堆。
在此期间,删除数据不会减少内部缓冲区(空白区域将通过左移或右移元素填充,在开始和结束时留出空间以便以后插入)。
追加项目,如QVector
将在数组末尾创建额外的新空间,因为与QVector
不同,实际数据不会存储在内部缓冲区中,您可以创建很多单个指令中的空间,无论项目的大小是多少(与QVector
不同) - 因为您只是将指针添加到索引缓冲区中。
例如,如果您使用的是32位系统(每个指针4个字节)并且您在QList
中存储了50个项目,并且每个项目的大小为1MB,那么QVector
缓冲区将需要调整大小到50MB,QList
的内部缓冲区只需要分配200B的内存。这是您需要在resize()
中调用QVector
的地方,但在QList
中没有必要,因为分配小块内存不会有问题,因为分配50MB的内存。
但是,这需要付出代价,这意味着您有时希望QVector
代替QList
:对于存储在QList
中的单项,您需要一个额外的分配在堆上 - 保留项的实际数据(内部缓冲区中的指针指向的数据)。如果要添加大于指针的10000个项目(因为,如果它可以适合指针,它将直接存储在内部缓冲区中),则需要10000个系统调用来为堆上的10000个项目分配数据。但是,如果您使用的是QVector
,并且调用resize
,则可以在单个alloc调用中调整所有项目 - 因此,如果您需要很多内容,请不要使用QList
插入或追加,更喜欢QVector
。当然,如果您使用QList
来存储共享类,则无需额外分配,这使QList
更适合。
因此,对大多数情况更喜欢QList
:
QLinkedList
QVector
数据更快。请勿在以下情况中使用它,并且更喜欢QVector
:
最后,请注意:QList
(和QVector
)具有reserve(int alloc)
函数,如果QList
为alloc
,则会导致QList
的内部缓冲区增长大于内部缓冲区的当前大小。但是,这不会影响size()
的外部大小({{1}}将始终返回列表中包含的确切元素数。)
答案 1 :(得分:7)
我认为原因是因为QList
不要求元素类型具有默认构造函数。
因此,QList
没有创建只复制它们的对象的操作。
但是如果你真的需要调整QList
的大小(无论出于何种原因),这里有一个能够做到这一点的函数。请注意,它只是一个便利函数,并没有考虑到性能。
template<class T>
void resizeList(QList<T> & list, int newSize) {
int diff = newSize - list.size();
T t;
if (diff > 0) {
list.reserve(diff);
while (diff--) list.append(t);
} else if (diff < 0) list.erase(list.end() + diff, list.end());
}
答案 2 :(得分:0)
小岛的答案很好,但它会多次添加同一个对象。这是一个实用程序函数,它将为智能指针列表添加不同的对象。
template<class T>
void resizeSmartList(QList<QSharedPointer<T> > & list, int newSize) {
int diff = newSize - list.size();
if (diff > 0) {
list.reserve(diff);
while (diff>0){
QSharedPointer<T> t = QSharedPointer<T>(new T);
list.append(t);
diff--;
}
}else if (diff < 0) list.erase(list.end() + diff, list.end());
}
如果没有智能指针使用,以下内容会在列表中添加不同的对象。
template<class T>
void resizeList(QList<T> & list, int newSize) {
int diff = newSize - list.size();
if (diff > 0) {
list.reserve(diff);
while (diff>0){
T t = new T;
list.append(t);
diff--;
}
}else if (diff < 0) list.erase(list.end() + diff, list.end());
}
还要记住你的对象必须有默认构造函数(在头文件中用arg =“someValue”声明构造函数),否则它将失败。
答案 3 :(得分:-3)
只需使用
之类的东西QList<Smth> myList;
// ... some operations on the list here
myList << QVector<Smth>(desiredNewSize - myList.size()).toList();
基本上,到处都有这些to
/ from
Vector
/ List
/ Set()
方法,这使得在必要时调整Qt容器的大小变得微不足道一种有点手工,但微不足道和有效(我相信)的方式。
另一个(1或2线)解决方案是:
myList.reserve(newListSize); // note, how we have to reserve manually
std::fill_n(std::back_inserter(myList), desiredNewSize - myList.size(), Smth());
- 对于面向STL的人来说:)
有关有效QList::resize()
可能有多复杂的背景知识,请参阅: