Qt
Internal/Contributor docs for the Qt SDK. Note: These are NOT official API docs; those are found at https://doc.qt.io/
Loading...
Searching...
No Matches
containers.qdoc
Go to the documentation of this file.
1
// Copyright (C) 2016 The Qt Company Ltd.
2
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GFDL-1.3-no-invariants-only
3
4
/*!
5
\page containers.html
6
\title Container Classes
7
\ingroup groups
8
\ingroup qt-basic-concepts
9
\keyword container class
10
\keyword container classes
11
12
\brief Qt's template-based container classes.
13
14
\section1 Introduction
15
16
The Qt library provides a set of general purpose template-based
17
container classes. These classes can be used to store items of a
18
specified type. For example, if you need a resizable array of
19
\l{QString}s, use QList<QString>.
20
21
These container classes are designed to be lighter, safer, and
22
easier to use than the STL containers. If you are unfamiliar with
23
the STL, or prefer to do things the "Qt way", you can use these
24
classes instead of the STL classes.
25
26
The container classes are \l{implicitly shared}, they are
27
\l{reentrant}, and they are optimized for speed, low memory
28
consumption, and minimal inline code expansion, resulting in
29
smaller executables. In addition, they are \l{thread-safe}
30
in situations where they are used as read-only containers
31
by all threads used to access them.
32
33
The containers provide iterators for traversal. \l{STL-style iterators}
34
are the most efficient ones and can be used together with Qt's and
35
STL's \l{generic algorithms}.
36
\l{Java-style Iterators} are provided for backwards compatibility.
37
38
\note Since Qt 5.14, range constructors are available for most of the
39
container classes. QMultiMap is a notable exception. Their use is
40
encouraged to replace of the various deprecated from/to methods of Qt 5.
41
For example:
42
43
\snippet code/doc_src_containers.cpp 25
44
45
\section1 The Container Classes
46
47
Qt provides the following sequential containers: QList,
48
QStack, and QQueue. For most
49
applications, QList is the best type to use. It provides very fast
50
appends. If you really need a linked-list, use std::list.
51
QStack and QQueue are convenience classes that provide LIFO and
52
FIFO semantics.
53
54
Qt also provides these associative containers: QMap,
55
QMultiMap, QHash, QMultiHash, and QSet. The "Multi" containers
56
conveniently support multiple values associated with a single
57
key. The "Hash" containers provide faster lookup by using a hash
58
function instead of a binary search on a sorted set.
59
60
As special cases, the QCache and QContiguousCache classes provide
61
efficient hash-lookup of objects in a limited cache storage.
62
63
\table
64
\header \li Class \li Summary
65
66
\row \li \l{QList}<T>
67
\li This is by far the most commonly used container class. It
68
stores a list of values of a given type (T) that can be accessed
69
by index. Internally, it stores an array of values of a
70
given type at adjacent
71
positions in memory. Inserting at the front or in the middle of
72
a list can be quite slow, because it can lead to large numbers
73
of items having to be moved by one position in memory.
74
75
\row \li \l{QVarLengthArray}<T, Prealloc>
76
\li This provides a low-level variable-length array. It can be used
77
instead of QList in places where speed is particularly important.
78
79
\row \li \l{QStack}<T>
80
\li This is a convenience subclass of QList that provides
81
"last in, first out" (LIFO) semantics. It adds the following
82
functions to those already present in QList:
83
\l{QStack::push()}{push()}, \l{QStack::pop()}{pop()},
84
and \l{QStack::top()}{top()}.
85
86
\row \li \l{QQueue}<T>
87
\li This is a convenience subclass of QList that provides
88
"first in, first out" (FIFO) semantics. It adds the following
89
functions to those already present in QList:
90
\l{QQueue::enqueue()}{enqueue()},
91
\l{QQueue::dequeue()}{dequeue()}, and \l{QQueue::head()}{head()}.
92
93
\row \li \l{QSet}<T>
94
\li This provides a single-valued mathematical set with fast
95
lookups.
96
97
\row \li \l{QMap}<Key, T>
98
\li This provides a dictionary (associative array) that maps keys
99
of type Key to values of type T. Normally each key is associated
100
with a single value. QMap stores its data in Key order; if order
101
doesn't matter QHash is a faster alternative.
102
103
\row \li \l{QMultiMap}<Key, T>
104
\li This provides a dictionary, like QMap, except it allows
105
inserting multiple equivalent keys.
106
107
\row \li \l{QHash}<Key, T>
108
\li This has almost the same API as QMap, but provides
109
significantly faster lookups. QHash stores its data in an
110
arbitrary order.
111
112
\row \li \l{QMultiHash}<Key, T>
113
\li This provides a hash-table-based dictionary, like QHash,
114
except it allows inserting multiple equivalent keys.
115
116
\endtable
117
118
Containers can be nested. For example, it is perfectly possible
119
to use a QMap<QString, QList<int>>, where the key type is
120
QString and the value type QList<int>.
121
122
The containers are defined in individual header files with the
123
same name as the container (e.g., \c <QList>). For
124
convenience, the containers are forward declared in \c
125
<QtContainerFwd>.
126
127
\target assignable data type
128
\target assignable data types
129
130
The values stored in the various containers can be of any
131
\e{assignable data type}. To qualify, a type must provide a
132
copy constructor, and an assignment operator. For some
133
operations a default constructor is also required. This
134
covers most data types you are likely to want to
135
store in a container, including basic types such as \c int and \c
136
double, pointer types, and Qt data types such as QString, QDate,
137
and QTime, but it doesn't cover QObject or any QObject subclass
138
(QWidget, QDialog, QTimer, etc.). If you attempt to instantiate a
139
QList<QWidget>, the compiler will complain that QWidget's copy
140
constructor and assignment operators are disabled. If you want to
141
store these kinds of objects in a container, store them as
142
pointers, for example as QList<QWidget *>.
143
144
Here's an example custom data type that meets the requirement of
145
an assignable data type:
146
147
\snippet code/doc_src_containers.cpp 0
148
149
If we don't provide a copy constructor or an assignment operator,
150
C++ provides a default implementation that performs a
151
member-by-member copy. In the example above, that would have been
152
sufficient. Also, if you don't provide any constructors, C++
153
provides a default constructor that initializes its member using
154
default constructors. Although it doesn't provide any
155
explicit constructors or assignment operator, the following data
156
type can be stored in a container:
157
158
\snippet streaming/main.cpp 0
159
160
Some containers have additional requirements for the data types
161
they can store. For example, the Key type of a QMap<Key, T> must
162
provide \c operator<(). Such special requirements are documented
163
in a class's detailed description. In some cases, specific
164
functions have special requirements; these are described on a
165
per-function basis. The compiler will always emit an error if a
166
requirement isn't met.
167
168
Qt's containers provide operator<<() and operator>>() so that they
169
can easily be read and written using a QDataStream. This means
170
that the data types stored in the container must also support
171
operator<<() and operator>>(). Providing such support is
172
straightforward; here's how we could do it for the Movie struct
173
above:
174
175
\snippet streaming/main.cpp 1
176
\codeline
177
\snippet streaming/main.cpp 2
178
179
\target default-constructed value
180
181
The documentation of certain container class functions refer to
182
\e{default-constructed values}; for example, QList
183
automatically initializes its items with default-constructed
184
values, and QMap::value() returns a default-constructed value if
185
the specified key isn't in the map. For most value types, this
186
simply means that a value is created using the default
187
constructor (e.g. an empty string for QString). But for primitive
188
types like \c{int} and \c{double}, as well as for pointer types,
189
the C++ language doesn't specify any initialization; in those
190
cases, Qt's containers automatically initialize the value to 0.
191
192
\section1 Iterating over Containers
193
194
\section2 Range-based for
195
196
Range-based \c for should preferably be used for containers:
197
198
\snippet code/doc_src_containers.cpp range_for
199
200
Note that when using a Qt container in a non-const context,
201
\l{implicit sharing} may perform an undesired detach of the container.
202
To prevent this, use \c std::as_const():
203
204
\snippet code/doc_src_containers.cpp range_for_as_const
205
206
For associative containers, this will loop over the values.
207
208
\section2 Index-based
209
210
For sequential containers that store their items contiguously in memory
211
(for example, QList), index-based iteration can be used:
212
213
\snippet code/doc_src_containers.cpp index
214
215
\section2 The Iterator Classes
216
217
Iterators provide a uniform means to access items in a container.
218
Qt's container classes provide two types of iterators: STL-style
219
iterators and Java-style iterators. Iterators of both types are
220
invalidated when the data in the container is modified or detached
221
from \l{Implicit Sharing}{implicitly shared copies} due to a call
222
to a non-const member function.
223
224
\section3 STL-Style Iterators
225
226
STL-style iterators have been available since the release of Qt
227
2.0. They are compatible with Qt's and STL's \l{generic
228
algorithms} and are optimized for speed.
229
230
For each container class, there are two STL-style iterator types:
231
one that provides read-only access and one that provides
232
read-write access. Read-only iterators should be used wherever
233
possible because they are faster than read-write iterators.
234
235
\table
236
\header \li Containers \li Read-only iterator
237
\li Read-write iterator
238
\row \li QList<T>, QStack<T>, QQueue<T> \li QList<T>::const_iterator
239
\li QList<T>::iterator
240
\row \li QSet<T> \li QSet<T>::const_iterator
241
\li QSet<T>::iterator
242
\row \li QMap<Key, T>, QMultiMap<Key, T> \li QMap<Key, T>::const_iterator
243
\li QMap<Key, T>::iterator
244
\row \li QHash<Key, T>, QMultiHash<Key, T> \li QHash<Key, T>::const_iterator
245
\li QHash<Key, T>::iterator
246
\endtable
247
248
The API of the STL iterators is modelled on pointers in an array.
249
For example, the \c ++ operator advances the iterator to the next
250
item, and the \c * operator returns the item that the iterator
251
points to. In fact, for QList and QStack, which store their
252
items at adjacent memory positions, the
253
\l{QList::iterator}{iterator} type is just a typedef for \c{T *},
254
and the \l{QList::iterator}{const_iterator} type is
255
just a typedef for \c{const T *}.
256
257
In this discussion, we will concentrate on QList and QMap. The
258
iterator types for QSet have exactly
259
the same interface as QList's iterators; similarly, the iterator
260
types for QHash have the same interface as QMap's iterators.
261
262
Here's a typical loop for iterating through all the elements of a
263
QList<QString> in order and converting them to lowercase:
264
265
\snippet code/doc_src_containers.cpp 10
266
267
STL-style iterators point directly at items. The \l{QList::begin()}{begin()}
268
function of a container returns an iterator that points to the first item in the
269
container. The \l{QList::end()}{end()} function of a container returns an iterator to the
270
imaginary item one position past the last item in the container.
271
\l {QList::end()}{end()} marks an invalid position; it must never be dereferenced.
272
It is typically used in a loop's break condition. If the list is
273
empty, \l{QList::begin}{begin()} equals \l{QList::end()}{end()}, so we never execute the loop.
274
275
The diagram below shows the valid iterator positions as red
276
arrows for a list containing four items:
277
278
\image stliterators1.png
279
280
Iterating backward with an STL-style iterator is done with reverse iterators:
281
282
\snippet code/doc_src_containers.cpp 11
283
284
In the code snippets so far, we used the unary \c * operator to
285
retrieve the item (of type QString) stored at a certain iterator
286
position, and we then called QString::toLower() on it.
287
288
For read-only access, you can use const_iterator, \l{QList::cbegin}{cbegin()},
289
and \l{QList::cend()}{cend()}. For example:
290
291
\snippet code/doc_src_containers.cpp 12
292
293
The following table summarizes the STL-style iterators' API:
294
295
\table
296
\header \li Expression \li Behavior
297
\row \li \c{*i} \li Returns the current item
298
\row \li \c{++i} \li Advances the iterator to the next item
299
\row \li \c{i += n} \li Advances the iterator by \c n items
300
\row \li \c{--i} \li Moves the iterator back by one item
301
\row \li \c{i -= n} \li Moves the iterator back by \c n items
302
\row \li \c{i - j} \li Returns the number of items between iterators \c i and \c j
303
\endtable
304
305
The \c{++} and \c{--} operators are available both as prefix
306
(\c{++i}, \c{--i}) and postfix (\c{i++}, \c{i--}) operators. The
307
prefix versions modify the iterators and return a reference to
308
the modified iterator; the postfix versions take a copy of the
309
iterator before they modify it, and return that copy. In
310
expressions where the return value is ignored, we recommend that
311
you use the prefix operators (\c{++i}, \c{--i}), as these are
312
slightly faster.
313
314
For non-const iterator types, the return value of the unary \c{*}
315
operator can be used on the left side of the assignment operator.
316
317
For QMap and QHash, the \c{*} operator returns the value
318
component of an item. If you want to retrieve the key, call key()
319
on the iterator. For symmetry, the iterator types also provide a
320
value() function to retrieve the value. For example, here's how
321
we would print all items in a QMap to the console:
322
323
\snippet code/doc_src_containers.cpp 13
324
325
Thanks to \l{implicit sharing}, it is very inexpensive for a
326
function to return a container per value. The Qt API contains
327
dozens of functions that return a QList or QStringList per value
328
(e.g., QSplitter::sizes()). If you want to iterate over these
329
using an STL iterator, you should always take a copy of the
330
container and iterate over the copy. For example:
331
332
\snippet code/doc_src_containers.cpp 14
333
334
This problem doesn't occur with functions that return a const or
335
non-const reference to a container.
336
337
\section4 Implicit sharing iterator problem
338
339
\l{Implicit sharing} has another consequence on STL-style
340
iterators: you should avoid copying a container while
341
iterators are active on that container. The iterators
342
point to an internal structure, and if you copy a container
343
you should be very careful with your iterators. E.g:
344
345
\snippet code/doc_src_containers.cpp 24
346
347
The above example only shows a problem with QList, but
348
the problem exists for all the implicitly shared Qt containers.
349
350
\section3 Java-Style Iterators
351
\l{java-style-iterators}{Java-Style iterators}
352
are modelled
353
on Java's iterator classes.
354
New code should prefer \l{STL-Style Iterators}.
355
356
\section1 Qt containers compared with std containers
357
358
\table
359
\header \li Qt container \li Closest std container
360
361
\row \li \l{QList}<T>
362
\li Similar to std::vector<T>
363
364
\l{QList} and \l{QVector} were unified in Qt 6. Both
365
use the datamodel from QVector. QVector is now an alias to QList.
366
367
This means that QList is not implemented as a linked list, so if
368
you need constant time insert, delete, append or prepend,
369
consider \c std::list<T>. See \l{QList} for details.
370
371
\row \li \l{QVarLengthArray}<T, Prealloc>
372
\li Resembles a mix of std::array<T> and std::vector<T>.
373
374
For performance reasons, QVarLengthArray lives on the stack unless
375
resized. Resizing it automatically causes it to use the heap instead.
376
377
\row \li \l{QStack}<T>
378
\li Similar to std::stack<T>, inherits from \l{QList}.
379
380
\row \li \l{QQueue}<T>
381
\li Similar to std::queue<T>, inherits from \l{QList}.
382
383
\row \li \l{QSet}<T>
384
\li Similar to std::unordered_set<T>. Internally, \l{QSet} is implemented with a
385
\l{QHash}.
386
387
\row \li \l{QMap}<Key, T>
388
\li Similar to std::map<Key, T>.
389
390
\row \li \l{QMultiMap}<Key, T>
391
\li Similar to std::multimap<Key, T>.
392
393
\row \li \l{QHash}<Key, T>
394
\li Most similar to std::unordered_map<Key, T>.
395
396
\row \li \l{QMultiHash}<Key, T>
397
\li Most similar to std::unordered_multimap<Key, T>.
398
399
\endtable
400
401
\section1 Qt containers and std algorithms
402
403
You can use Qt containers with functions from \c{#include <algorithm>}.
404
405
\snippet code/doc_src_containers.cpp 26
406
407
\section1 Other Container-Like Classes
408
409
Qt includes other template classes that resemble containers in
410
some respects. These classes don't provide iterators and cannot
411
be used with the \l foreach keyword.
412
413
\list
414
\li QCache<Key, T> provides a cache to store objects of a certain
415
type T associated with keys of type Key.
416
417
\li QContiguousCache<T> provides an efficient way of caching data
418
that is typically accessed in a contiguous way.
419
\endlist
420
421
Additional non-template types that compete with Qt's template
422
containers are QBitArray, QByteArray, QString, and QStringList.
423
424
\section1 Algorithmic Complexity
425
426
Algorithmic complexity is concerned about how fast (or slow) each
427
function is as the number of items in the container grow. For
428
example, inserting an item in the middle of a std::list is an
429
extremely fast operation, irrespective of the number of items
430
stored in the list. On the other hand, inserting an item
431
in the middle of a QList is potentially very expensive if the
432
QList contains many items, since half of the items must be
433
moved one position in memory.
434
435
To describe algorithmic complexity, we use the following
436
terminology, based on the "big Oh" notation:
437
438
\target constant time
439
\target logarithmic time
440
\target linear time
441
\target linear-logarithmic time
442
\target quadratic time
443
444
\list
445
\li \b{Constant time:} O(1). A function is said to run in constant
446
time if it requires the same amount of time no matter how many
447
items are present in the container. One example is
448
QList::push_back().
449
450
\li \b{Logarithmic time:} O(log \e n). A function that runs in
451
logarithmic time is a function whose running time is
452
proportional to the logarithm of the number of items in the
453
container. One example is the binary search algorithm.
454
455
\li \b{Linear time:} O(\e n). A function that runs in linear time
456
will execute in a time directly proportional to the number of
457
items stored in the container. One example is
458
QList::insert().
459
460
\li \b{Linear-logarithmic time:} O(\e{n} log \e n). A function
461
that runs in linear-logarithmic time is asymptotically slower
462
than a linear-time function, but faster than a quadratic-time
463
function.
464
465
\li \b{Quadratic time:} O(\e{n}\unicode{178}). A quadratic-time function
466
executes in a time that is proportional to the square of the
467
number of items stored in the container.
468
\endlist
469
470
The following table summarizes the algorithmic complexity of the sequential
471
container QList<T>:
472
473
\table
474
\header \li \li Index lookup \li Insertion \li Prepending \li Appending
475
\row \li QList<T> \li O(1) \li O(n) \li O(n) \li Amort. O(1)
476
\endtable
477
478
In the table, "Amort." stands for "amortized behavior". For
479
example, "Amort. O(1)" means that if you call the function
480
only once, you might get O(\e n) behavior, but if you call it
481
multiple times (e.g., \e n times), the average behavior will be
482
O(1).
483
484
The following table summarizes the algorithmic complexity of Qt's
485
associative containers and sets:
486
487
\table
488
\header \li{1,2} \li{2,1} Key lookup \li{2,1} Insertion
489
\header \li Average \li Worst case \li Average \li Worst case
490
\row \li QMap<Key, T> \li O(log \e n) \li O(log \e n) \li O(log \e n) \li O(log \e n)
491
\row \li QMultiMap<Key, T> \li O(log \e n) \li O(log \e n) \li O(log \e n) \li O(log \e n)
492
\row \li QHash<Key, T> \li Amort. O(1) \li O(\e n) \li Amort. O(1) \li O(\e n)
493
\row \li QSet<Key> \li Amort. O(1) \li O(\e n) \li Amort. O(1) \li O(\e n)
494
\endtable
495
496
With QList, QHash, and QSet, the performance of appending items
497
is amortized O(log \e n). It can be brought down to O(1) by
498
calling QList::reserve(), QHash::reserve(), or QSet::reserve()
499
with the expected number of items before you insert the items.
500
The next section discusses this topic in more depth.
501
502
\section1 Optimizations for Primitive and Relocatable Types
503
504
Qt containers can use optimized code paths if the stored
505
elements are relocatable or even primitive.
506
However, whether types are primitive or relocatable
507
cannot be detected in all cases.
508
You can declare your types to be primitive or relocatable
509
by using the Q_DECLARE_TYPEINFO macro with the Q_PRIMITIVE_TYPE
510
flag or the Q_RELOCATABLE_TYPE flag. See the documentation
511
of Q_DECLARE_TYPEINFO for further details and usage examples.
512
513
If you do not use Q_DECLARE_TYPEINFO,
514
Qt will use
515
\l {https://en.cppreference.com/w/cpp/types/is_trivial} {std::is_trivial_v<T>}
516
to identify primitive
517
types and it will require both
518
\l {https://en.cppreference.com/w/cpp/types/is_trivially_copyable} {std::is_trivially_copyable_v<T>}
519
and
520
\l {https://en.cppreference.com/w/cpp/types/is_destructible} {std::is_trivially_destructible_v<T>}
521
to identify relocatable types.
522
This is always a safe choice, albeit
523
of maybe suboptimal performance.
524
525
\section1 Growth Strategies
526
527
QList<T>, QString, and QByteArray store their items
528
contiguously in memory; QHash<Key, T> keeps a
529
hash table whose size is proportional to the number
530
of items in the hash. To avoid reallocating the data every single
531
time an item is added at the end of the container, these classes
532
typically allocate more memory than necessary.
533
534
Consider the following code, which builds a QString from another
535
QString:
536
537
\snippet code/doc_src_containers.cpp 23
538
539
We build the string \c out dynamically by appending one character
540
to it at a time. Let's assume that we append 15000 characters to
541
the QString string. Then the following 11 reallocations (out of a
542
possible 15000) occur when QString runs out of space: 8, 24, 56,
543
120, 248, 504, 1016, 2040, 4088, 8184, 16376.
544
At the end, the QString has 16376 Unicode
545
characters allocated, 15000 of which are occupied.
546
547
The values above may seem a bit strange, but there is a guiding
548
principle. It advances by doubling the size each time.
549
More precisely, it advances to the next power of two, minus
550
16 bytes. 16 bytes corresponds to eight characters, as QString
551
uses UTF-16 internally.
552
553
QByteArray uses the same algorithm as
554
QString, but 16 bytes correspond to 16 characters.
555
556
QList<T> also uses that algorithm, but 16 bytes correspond to
557
16/sizeof(T) elements.
558
559
QHash<Key, T> is a totally different case. QHash's internal hash
560
table grows by powers of two, and each time it grows, the items
561
are relocated in a new bucket, computed as qHash(\e key) %
562
QHash::capacity() (the number of buckets). This remark applies to
563
QSet<T> and QCache<Key, T> as well.
564
565
For most applications, the default growing algorithm provided by
566
Qt does the trick. If you need more control, QList<T>,
567
QHash<Key, T>, QSet<T>, QString, and QByteArray provide a trio of
568
functions that allow you to check and specify how much memory to
569
use to store the items:
570
571
\list
572
\li \l{QString::capacity()}{capacity()} returns the
573
number of items for which memory is allocated (for QHash and
574
QSet, the number of buckets in the hash table).
575
\li \l{QString::reserve()}{reserve}(\e size) explicitly
576
preallocates memory for \e size items.
577
\li \l{QString::squeeze()}{squeeze()} frees any memory
578
not required to store the items.
579
\endlist
580
581
If you know approximately how many items you will store in a
582
container, you can start by calling \l{QString::reserve()}{reserve()}, and when you are
583
done populating the container, you can call \l{QString::squeeze()}{squeeze()} to release
584
the extra preallocated memory.
585
*/
qtbase
src
corelib
doc
src
containers.qdoc
Generated on
for Qt by
1.14.0