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
qhash.cpp
Go to the documentation of this file.
1// Copyright (C) 2020 The Qt Company Ltd.
2// Copyright (C) 2021 Intel Corporation.
3// Copyright (C) 2012 Giuseppe D'Angelo <dangelog@gmail.com>.
4// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
5// Qt-Security score:significant reason:default
6
7// for rand_s, _CRT_RAND_S must be #defined before #including stdlib.h.
8// put it at the beginning so some indirect inclusion doesn't break it
9#ifndef _CRT_RAND_S
10#define _CRT_RAND_S
11#endif
12#include <stdlib.h>
13#include <stdint.h>
14
15#include "qhash.h"
16
17#ifdef truncate
18#undef truncate
19#endif
20
21#include <qbitarray.h>
22#include <qstring.h>
23#include <qglobal.h>
24#include <qbytearray.h>
25#include <qdatetime.h>
26#include <qbasicatomic.h>
27#include <qendian.h>
28#include <private/qrandom_p.h>
29#include <private/qsimd_p.h>
30
31#ifndef QT_BOOTSTRAPPED
32#include <qcoreapplication.h>
33#include <qrandom.h>
34#include <private/qlocale_tools_p.h>
35#endif // QT_BOOTSTRAPPED
36
37// Implementation of SipHash algorithm
38#include "../../3rdparty/siphash/siphash.cpp"
39
40#include <array>
41#include <limits.h>
42
43#if defined(QT_NO_DEBUG) && !defined(NDEBUG)
44# define NDEBUG
45#endif
46#include <assert.h>
47
48#ifdef Q_CC_GNU
49# define Q_DECL_HOT_FUNCTION __attribute__((hot))
50#else
51# define Q_DECL_HOT_FUNCTION
52#endif
53
55
56void qt_from_latin1(char16_t *dst, const char *str, size_t size) noexcept; // qstring.cpp
57
58// We assume that pointers and size_t have the same size. If that assumption should fail
59// on a platform the code selecting the different methods below needs to be fixed.
60static_assert(sizeof(size_t) == QT_POINTER_SIZE, "size_t and pointers have different size.");
61
62namespace {
63struct HashSeedStorage
64{
65 static constexpr int SeedCount = 2;
66 QBasicAtomicInteger<quintptr> seeds[SeedCount] = { Q_BASIC_ATOMIC_INITIALIZER(0), Q_BASIC_ATOMIC_INITIALIZER(0) };
67
68#if !QT_SUPPORTS_INIT_PRIORITY || defined(QT_BOOTSTRAPPED)
69 constexpr HashSeedStorage() = default;
70#else
71 HashSeedStorage() { initialize(0); }
72#endif
73
74 enum State {
75 OverriddenByEnvironment = -1,
76 JustInitialized,
77 AlreadyInitialized
78 };
79 struct StateResult {
80 quintptr requestedSeed;
81 State state;
82 };
83
84 StateResult state(int which = -1);
85 Q_DECL_HOT_FUNCTION QHashSeed currentSeed(int which)
86 {
87 return { state(which).requestedSeed };
88 }
89
90 void resetSeed()
91 {
92#ifndef QT_BOOTSTRAPPED
93 if (state().state < AlreadyInitialized)
94 return;
95
96 // update the public seed
97 QRandomGenerator *generator = QRandomGenerator::system();
98 seeds[0].storeRelaxed(sizeof(size_t) > sizeof(quint32)
99 ? generator->generate64() : generator->generate());
100#endif
101 }
102
103 void clearSeed()
104 {
105 state();
106 seeds[0].storeRelaxed(0); // always write (smaller code)
107 }
108
109private:
110 Q_NEVER_INLINE Q_DECL_COLD_FUNCTION StateResult initialize(int which) noexcept;
111};
112
113[[maybe_unused]] HashSeedStorage::StateResult HashSeedStorage::initialize(int which) noexcept
114{
115 StateResult result = { 0, OverriddenByEnvironment };
116#ifdef QT_BOOTSTRAPPED
117 Q_UNUSED(which);
118 Q_UNREACHABLE_RETURN(result);
119#else
120 // can't use qEnvironmentVariableIntValue (reentrancy)
121 const char *seedstr = getenv("QT_HASH_SEED");
122 if (seedstr) {
123 auto r = qstrntoll(seedstr, strlen(seedstr), 10);
124 if (r.used > 0 && size_t(r.used) == strlen(seedstr)) {
125 if (r.result) {
126 // can't use qWarning here (reentrancy)
127 fprintf(stderr, "QT_HASH_SEED: forced seed value is not 0; ignored.\n");
128 }
129
130 // we don't have to store to the seed, since it's pre-initialized by
131 // the compiler to zero
132 return result;
133 }
134 }
135
136 // update the full seed
137 auto x = qt_initial_random_value();
138 for (int i = 0; i < SeedCount; ++i) {
139 seeds[i].storeRelaxed(x.data[i]);
140 if (which == i)
141 result.requestedSeed = x.data[i];
142 }
143 result.state = JustInitialized;
144 return result;
145#endif
146}
147
148inline HashSeedStorage::StateResult HashSeedStorage::state(int which)
149{
150 constexpr quintptr BadSeed = quintptr(Q_UINT64_C(0x5555'5555'5555'5555));
151 StateResult result = { BadSeed, AlreadyInitialized };
152
153#if defined(QT_BOOTSTRAPPED)
154 result = { 0, OverriddenByEnvironment };
155#elif !QT_SUPPORTS_INIT_PRIORITY
156 // dynamic initialization
157 static auto once = [&]() {
158 result = initialize(which);
159 return true;
160 }();
161 Q_UNUSED(once);
162#endif
163
164 if (result.state == AlreadyInitialized && which >= 0)
165 return { seeds[which].loadRelaxed(), AlreadyInitialized };
166 return result;
167}
168} // unnamed namespace
169
170/*
171 The QHash seed itself.
172*/
173#ifdef Q_DECL_INIT_PRIORITY
174Q_DECL_INIT_PRIORITY(05)
175#else
176Q_CONSTINIT
177#endif
178static HashSeedStorage qt_qhash_seed;
179
180/*
181 * Hashing for memory segments is based on the public domain MurmurHash2 by
182 * Austin Appleby. See http://murmurhash.googlepages.com/
183 */
184#if QT_POINTER_SIZE == 4
185Q_NEVER_INLINE Q_DECL_HOT_FUNCTION
186static inline uint murmurhash(const void *key, uint len, uint seed) noexcept
187{
188 // 'm' and 'r' are mixing constants generated offline.
189 // They're not really 'magic', they just happen to work well.
190
191 const unsigned int m = 0x5bd1e995;
192 const int r = 24;
193
194 // Initialize the hash to a 'random' value
195
196 unsigned int h = seed ^ len;
197
198 // Mix 4 bytes at a time into the hash
199
200 const unsigned char *data = reinterpret_cast<const unsigned char *>(key);
201 const unsigned char *end = data + (len & ~3);
202
203 while (data != end) {
204 size_t k;
205 memcpy(&k, data, sizeof(uint));
206
207 k *= m;
208 k ^= k >> r;
209 k *= m;
210
211 h *= m;
212 h ^= k;
213
214 data += 4;
215 }
216
217 // Handle the last few bytes of the input array
218 len &= 3;
219 if (len) {
220 unsigned int k = 0;
221 end += len;
222
223 while (data != end) {
224 k <<= 8;
225 k |= *data;
226 ++data;
227 }
228 h ^= k;
229 h *= m;
230 }
231
232 // Do a few final mixes of the hash to ensure the last few
233 // bytes are well-incorporated.
234
235 h ^= h >> 13;
236 h *= m;
237 h ^= h >> 15;
238
239 return h;
240}
241
242#else
243Q_NEVER_INLINE Q_DECL_HOT_FUNCTION
244static inline uint64_t murmurhash(const void *key, uint64_t len, uint64_t seed) noexcept
245{
246 const uint64_t m = 0xc6a4a7935bd1e995ULL;
247 const int r = 47;
248
249 uint64_t h = seed ^ (len * m);
250
251 const unsigned char *data = reinterpret_cast<const unsigned char *>(key);
252 const unsigned char *end = data + (len & ~7ul);
253
254 while (data != end) {
255 uint64_t k;
256 memcpy(&k, data, sizeof(uint64_t));
257
258 k *= m;
259 k ^= k >> r;
260 k *= m;
261
262 h ^= k;
263 h *= m;
264
265 data += 8;
266 }
267
268 len &= 7;
269 if (len) {
270 // handle the last few bytes of input
271 size_t k = 0;
272 end += len;
273
274 while (data != end) {
275 k <<= 8;
276 k |= *data;
277 ++data;
278 }
279 h ^= k;
280 h *= m;
281 }
282
283 h ^= h >> r;
284 h *= m;
285 h ^= h >> r;
286
287 return h;
288}
289
290#endif
291
293 None = 0,
295};
296
297template <ZeroExtension = None> static size_t
298qHashBits_fallback(const uchar *p, size_t size, size_t seed, size_t seed2) noexcept;
299template <> size_t qHashBits_fallback<None>(const uchar *p, size_t size, size_t seed, size_t seed2) noexcept
300{
301 if (size <= QT_POINTER_SIZE)
302 return murmurhash(p, size, seed);
303
304 return siphash(reinterpret_cast<const uchar *>(p), size, seed, seed2);
305}
306
307template <> size_t qHashBits_fallback<ByteToWord>(const uchar *data, size_t size, size_t seed, size_t seed2) noexcept
308{
309 auto quick_from_latin1 = [](char16_t *dest, const uchar *data, size_t size) {
310 // Quick, "inlined" version for very short blocks
311 std::copy_n(data, size, dest);
312 };
313 if (size <= QT_POINTER_SIZE / 2) {
314 std::array<char16_t, QT_POINTER_SIZE / 2> buf;
315 quick_from_latin1(buf.data(), data, size);
316 return murmurhash(buf.data(), size * 2, seed);
317 }
318
319 constexpr size_t TailSizeMask = sizeof(void *) / 2 - 1;
320 std::array<char16_t, 256> buf;
321 SipHash<> siphash(size * 2, seed, seed2);
322 ptrdiff_t offset = 0;
323 for ( ; offset + buf.size() < size; offset += buf.size()) {
324 qt_from_latin1(buf.data(), reinterpret_cast<const char *>(data) + offset, buf.size());
325 siphash.addBlock(reinterpret_cast<uint8_t *>(buf.data()), sizeof(buf));
326 }
327 if (size_t n = size - offset; n > TailSizeMask) {
328 n &= ~TailSizeMask;
329 qt_from_latin1(buf.data(), reinterpret_cast<const char *>(data) + offset, n);
330 siphash.addBlock(reinterpret_cast<uint8_t *>(buf.data()), n * 2);
331 offset += n;
332 }
333
334 quick_from_latin1(buf.data(), data + offset, size - offset);
335 return siphash.finalize(reinterpret_cast<uint8_t *>(buf.data()), (size - offset) * 2);
336}
337
338#if defined(__SANITIZE_ADDRESS__) || defined(__SANITIZE_THREAD__) // GCC
339# define QHASH_AES_SANITIZER_BUILD
340#elif __has_feature(address_sanitizer) || __has_feature(thread_sanitizer) // Clang
341# define QHASH_AES_SANITIZER_BUILD
342#endif
343
344// When built with a sanitizer, aeshash() is rightfully reported to have a
345// heap-buffer-overflow issue. However, we consider it to be safe in this
346// specific case and overcome the problem by correctly discarding the
347// out-of-range bits. To allow building the code with sanitizer,
348// QHASH_AES_SANITIZER_BUILD is used to disable aeshash() usage.
349#if QT_COMPILER_SUPPORTS_HERE(AES) && QT_COMPILER_SUPPORTS_HERE(SSE4_2) &&
350 !defined(QHASH_AES_SANITIZER_BUILD)
351# define AESHASH
352# define QT_FUNCTION_TARGET_STRING_AES_AVX2 "avx2,aes"
353# define QT_FUNCTION_TARGET_STRING_AES_AVX512
354 QT_FUNCTION_TARGET_STRING_ARCH_SKYLAKE_AVX512 ","
355 QT_FUNCTION_TARGET_STRING_AES
356# define QT_FUNCTION_TARGET_STRING_VAES_AVX512
357 QT_FUNCTION_TARGET_STRING_ARCH_SKYLAKE_AVX512 ","
358 QT_FUNCTION_TARGET_STRING_VAES
359# undef QHASH_AES_SANITIZER_BUILD
360# if QT_POINTER_SIZE == 8
361# define mm_set1_epz _mm_set1_epi64x
362# define mm_cvtsz_si128 _mm_cvtsi64_si128
363# define mm_cvtsi128_sz _mm_cvtsi128_si64
364# define mm256_set1_epz _mm256_set1_epi64x
365# else
366# define mm_set1_epz _mm_set1_epi32
367# define mm_cvtsz_si128 _mm_cvtsi32_si128
368# define mm_cvtsi128_sz _mm_cvtsi128_si32
369# define mm256_set1_epz _mm256_set1_epi32
370# endif
371
372namespace {
373 // This is inspired by the algorithm in the Go language. See:
374 // https://github.com/golang/go/blob/01b6cf09fc9f272d9db3d30b4c93982f4911d120/src/runtime/asm_amd64.s#L1105
375 // https://github.com/golang/go/blob/01b6cf09fc9f272d9db3d30b4c93982f4911d120/src/runtime/asm_386.s#L908
376 //
377 // Even though we're using the AESENC instruction from the CPU, this code
378 // is not encryption and this routine makes no claim to be
379 // cryptographically secure. We're simply using the instruction that performs
380 // the scrambling round (step 3 in [1]) because it's just very good at
381 // spreading the bits around.
382 //
383 // Note on Latin-1 hashing (ZX == ByteToWord): for simplicity of the
384 // algorithm, we pass sizes equivalent to the UTF-16 content (ZX == None).
385 // That means we must multiply by 2 on entry, divide by 2 on pointer
386 // advancing, and load half as much data from memory (though we produce
387 // exactly as much data in registers). The compilers appear to optimize
388 // this out.
389 //
390 // [1] https://en.wikipedia.org/wiki/Advanced_Encryption_Standard#High-level_description_of_the_algorithm
391
392 template <ZeroExtension ZX, typename T> static const T *advance(const T *ptr, ptrdiff_t n)
393 {
394 if constexpr (ZX == None)
395 return ptr + n;
396
397 // see note above on ZX == ByteToWord hashing
398 auto p = reinterpret_cast<const uchar *>(ptr);
399 n *= sizeof(T);
400 return reinterpret_cast<const T *>(p + n/2);
401 }
402
403 template <ZeroExtension> static __m128i loadu128(const void *ptr);
404 template <> Q_ALWAYS_INLINE QT_FUNCTION_TARGET(AES) __m128i loadu128<None>(const void *ptr)
405 {
406 return _mm_loadu_si128(reinterpret_cast<const __m128i *>(ptr));
407 }
408 template <> Q_ALWAYS_INLINE QT_FUNCTION_TARGET(AES) __m128i loadu128<ByteToWord>(const void *ptr)
409 {
410 // use a MOVQ followed by PMOVZXBW
411 // the compiler usually combines them as a single, loading PMOVZXBW
412 __m128i data = _mm_loadl_epi64(static_cast<const __m128i *>(ptr));
413 return _mm_cvtepu8_epi16(data);
414 }
415
416 // hash 16 bytes, running 3 scramble rounds of AES on itself (like label "final1")
417 Q_ALWAYS_INLINE static void QT_FUNCTION_TARGET(AES) QT_VECTORCALL
418 hash16bytes(__m128i &state0, __m128i data)
419 {
420 state0 = _mm_xor_si128(state0, data);
421 state0 = _mm_aesenc_si128(state0, state0);
422 state0 = _mm_aesenc_si128(state0, state0);
423 state0 = _mm_aesenc_si128(state0, state0);
424 }
425
426 // hash twice 16 bytes, running 2 scramble rounds of AES on itself
427 template <ZeroExtension ZX>
428 static void QT_FUNCTION_TARGET(AES) QT_VECTORCALL
429 hash2x16bytes(__m128i &state0, __m128i &state1, const __m128i *src0, const __m128i *src1)
430 {
431 __m128i data0 = loadu128<ZX>(src0);
432 __m128i data1 = loadu128<ZX>(src1);
433 state0 = _mm_xor_si128(data0, state0);
434 state1 = _mm_xor_si128(data1, state1);
435 state0 = _mm_aesenc_si128(state0, state0);
436 state1 = _mm_aesenc_si128(state1, state1);
437 state0 = _mm_aesenc_si128(state0, state0);
438 state1 = _mm_aesenc_si128(state1, state1);
439 }
440
441 struct AESHashSeed
442 {
443 __m128i state0;
444 __m128i mseed2;
445 AESHashSeed(size_t seed, size_t seed2) QT_FUNCTION_TARGET(AES);
446 __m128i state1() const QT_FUNCTION_TARGET(AES);
447 __m256i state0_256() const QT_FUNCTION_TARGET(AES_AVX2)
448 { return _mm256_set_m128i(state1(), state0); }
449 };
450} // unnamed namespace
451
452Q_ALWAYS_INLINE AESHashSeed::AESHashSeed(size_t seed, size_t seed2)
453{
454 __m128i mseed = mm_cvtsz_si128(seed);
455 mseed2 = mm_set1_epz(seed2);
456
457 // mseed (epi16) = [ seed, seed >> 16, seed >> 32, seed >> 48, len, 0, 0, 0 ]
458 mseed = _mm_insert_epi16(mseed, short(seed), 4);
459 // mseed (epi16) = [ seed, seed >> 16, seed >> 32, seed >> 48, len, len, len, len ]
460 mseed = _mm_shufflehi_epi16(mseed, 0);
461
462 // merge with the process-global seed
463 __m128i key = _mm_xor_si128(mseed, mseed2);
464
465 // scramble the key
466 __m128i state0 = _mm_aesenc_si128(key, key);
467 this->state0 = state0;
468}
469
470Q_ALWAYS_INLINE __m128i AESHashSeed::state1() const
471{
472 {
473 // unlike the Go code, we don't have more per-process seed
474 __m128i state1 = _mm_aesenc_si128(state0, mseed2);
475 return state1;
476 }
477}
478
479template <ZeroExtension ZX>
480static size_t QT_FUNCTION_TARGET(AES) QT_VECTORCALL
481aeshash128_16to32(__m128i state0, __m128i state1, const __m128i *src, const __m128i *srcend)
482{
483 {
484 const __m128i *src2 = advance<ZX>(srcend, -1);
485 if (advance<ZX>(src, 1) < srcend) {
486 // epilogue: between 16 and 31 bytes
487 hash2x16bytes<ZX>(state0, state1, src, src2);
488 } else if (src != srcend) {
489 // epilogue: between 1 and 16 bytes, overlap with the end
490 __m128i data = loadu128<ZX>(src2);
491 hash16bytes(state0, data);
492 }
493
494 // combine results:
495 state0 = _mm_xor_si128(state0, state1);
496 }
497
498 return mm_cvtsi128_sz(state0);
499}
500
501// load all 16 bytes and mask off the bytes past the end of the source
502static const qint8 maskarray[] = {
503 -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
504 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
505};
506
507// load 16 bytes ending at the data end, then shuffle them to the beginning
508static const qint8 shufflecontrol[] = {
509 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15,
510 -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1
511};
512
513template <ZeroExtension ZX>
514static size_t QT_FUNCTION_TARGET(AES) QT_VECTORCALL
515aeshash128_lt16(__m128i state0, const __m128i *src, const __m128i *srcend, size_t len)
516{
517 if (len) {
518 // We're going to load 16 bytes and mask zero the part we don't care
519 // (the hash of a short string is different from the hash of a longer
520 // including NULLs at the end because the length is in the key)
521 // WARNING: this may produce valgrind warnings, but it's safe
522
523 constexpr quintptr CachelineSize = 64;
524 __m128i data;
525
526 if ((quintptr(src) & (CachelineSize / 2)) == 0) {
527 // lower half of the cacheline:
528 __m128i mask = _mm_loadu_si128(reinterpret_cast<const __m128i *>(maskarray + 15 - len));
529 data = loadu128<ZX>(src);
530 data = _mm_and_si128(data, mask);
531 } else {
532 // upper half of the cacheline:
533 __m128i control = _mm_loadu_si128(reinterpret_cast<const __m128i *>(shufflecontrol + 15 - len));
534 data = loadu128<ZX>(advance<ZX>(srcend, -1));
535 data = _mm_shuffle_epi8(data, control);
536 }
537
538 hash16bytes(state0, data);
539 }
540 return mm_cvtsi128_sz(state0);
541}
542
543template <ZeroExtension ZX>
544static size_t QT_FUNCTION_TARGET(AES) QT_VECTORCALL
545aeshash128_ge32(__m128i state0, __m128i state1, const __m128i *src, const __m128i *srcend)
546{
547 // main loop: scramble two 16-byte blocks
548 for ( ; advance<ZX>(src, 2) < srcend; src = advance<ZX>(src, 2))
549 hash2x16bytes<ZX>(state0, state1, src, advance<ZX>(src, 1));
550
551 return aeshash128_16to32<ZX>(state0, state1, src, srcend);
552}
553
554# if QT_COMPILER_SUPPORTS_HERE(VAES)
555template <ZeroExtension> static __m256i loadu256(const void *ptr);
556template <> Q_ALWAYS_INLINE QT_FUNCTION_TARGET(VAES) __m256i loadu256<None>(const void *ptr)
557{
558 return _mm256_loadu_si256(reinterpret_cast<const __m256i *>(ptr));
559}
560template <> Q_ALWAYS_INLINE QT_FUNCTION_TARGET(VAES) __m256i loadu256<ByteToWord>(const void *ptr)
561{
562 // VPMOVZXBW xmm, ymm
563 __m128i data = _mm_loadu_si128(reinterpret_cast<const __m128i *>(ptr));
564 return _mm256_cvtepu8_epi16(data);
565}
566
567template <ZeroExtension ZX>
568static size_t QT_FUNCTION_TARGET(VAES_AVX512) QT_VECTORCALL
569aeshash256_lt32_avx256(__m256i state0, const uchar *p, size_t len)
570{
571 __m128i state0_128 = _mm256_castsi256_si128(state0);
572 if (len) {
573 __m256i data;
574 if constexpr (ZX == None) {
575 __mmask32 mask = _bzhi_u32(-1, unsigned(len));
576 data = _mm256_maskz_loadu_epi8(mask, p);
577 } else {
578 __mmask16 mask = _bzhi_u32(-1, unsigned(len) / 2);
579 __m128i data0 = _mm_maskz_loadu_epi8(mask, p);
580 data = _mm256_cvtepu8_epi16(data0);
581 }
582 __m128i data0 = _mm256_castsi256_si128(data);
583 if (len >= sizeof(__m128i)) {
584 state0 = _mm256_xor_si256(state0, data);
585 state0 = _mm256_aesenc_epi128(state0, state0);
586 state0 = _mm256_aesenc_epi128(state0, state0);
587 // we're XOR'ing the two halves so we skip the third AESENC
588 // state0 = _mm256_aesenc_epi128(state0, state0);
589
590 // XOR the two halves and extract
591 __m128i low = _mm256_extracti128_si256(state0, 0);
592 __m128i high = _mm256_extracti128_si256(state0, 1);
593 state0_128 = _mm_xor_si128(low, high);
594 } else {
595 hash16bytes(state0_128, data0);
596 }
597 }
598 return mm_cvtsi128_sz(state0_128);
599}
600
601template <ZeroExtension ZX>
602static size_t QT_FUNCTION_TARGET(VAES) QT_VECTORCALL
603aeshash256_ge32(__m256i state0, const __m128i *s, const __m128i *end, size_t len)
604{
605 static const auto hash32bytes = [](__m256i &state0, __m256i data) QT_FUNCTION_TARGET(VAES) {
606 state0 = _mm256_xor_si256(state0, data);
607 state0 = _mm256_aesenc_epi128(state0, state0);
608 state0 = _mm256_aesenc_epi128(state0, state0);
609 state0 = _mm256_aesenc_epi128(state0, state0);
610 };
611
612 // hash twice 32 bytes, running 2 scramble rounds of AES on itself
613 const auto hash2x32bytes = [](__m256i &state0, __m256i &state1, const void *src0,
614 const void *src1) QT_FUNCTION_TARGET(VAES) {
615 __m256i data0 = loadu256<ZX>(src0);
616 __m256i data1 = loadu256<ZX>(src1);
617 state0 = _mm256_xor_si256(data0, state0);
618 state1 = _mm256_xor_si256(data1, state1);
619 state0 = _mm256_aesenc_epi128(state0, state0);
620 state1 = _mm256_aesenc_epi128(state1, state1);
621 state0 = _mm256_aesenc_epi128(state0, state0);
622 state1 = _mm256_aesenc_epi128(state1, state1);
623 };
624
625 const __m256i *src = reinterpret_cast<const __m256i *>(s);
626 const __m256i *srcend = reinterpret_cast<const __m256i *>(end);
627
628 __m256i state1 = _mm256_aesenc_epi128(state0, mm256_set1_epz(len));
629
630 // main loop: scramble two 32-byte blocks
631 for ( ; advance<ZX>(src, 2) < srcend; src = advance<ZX>(src, 2))
632 hash2x32bytes(state0, state1, src, advance<ZX>(src, 1));
633
634 const __m256i *src2 = advance<ZX>(srcend, -1);
635 if (advance<ZX>(src, 1) < srcend) {
636 // epilogue: between 32 and 31 bytes
637 hash2x32bytes(state0, state1, src, src2);
638 } else if (src != srcend) {
639 // epilogue: between 1 and 32 bytes, overlap with the end
640 __m256i data = loadu256<ZX>(src2);
641 hash32bytes(state0, data);
642 }
643
644 // combine results:
645 state0 = _mm256_xor_si256(state0, state1);
646
647 // XOR the two halves and extract
648 __m128i low = _mm256_extracti128_si256(state0, 0);
649 __m128i high = _mm256_extracti128_si256(state0, 1);
650 return mm_cvtsi128_sz(_mm_xor_si128(low, high));
651}
652
653template <ZeroExtension ZX>
654static size_t QT_FUNCTION_TARGET(VAES)
655aeshash256(const uchar *p, size_t len, size_t seed, size_t seed2) noexcept
656{
657 AESHashSeed state(seed, seed2);
658 auto src = reinterpret_cast<const __m128i *>(p);
659 const auto srcend = reinterpret_cast<const __m128i *>(advance<ZX>(p, len));
660
661 if (len < sizeof(__m128i))
662 return aeshash128_lt16<ZX>(state.state0, src, srcend, len);
663
664 if (len <= sizeof(__m256i))
665 return aeshash128_16to32<ZX>(state.state0, state.state1(), src, srcend);
666
667 return aeshash256_ge32<ZX>(state.state0_256(), src, srcend, len);
668}
669
670template <ZeroExtension ZX>
671static size_t QT_FUNCTION_TARGET(VAES_AVX512)
672aeshash256_avx256(const uchar *p, size_t len, size_t seed, size_t seed2) noexcept
673{
674 AESHashSeed state(seed, seed2);
675 auto src = reinterpret_cast<const __m128i *>(p);
676 const auto srcend = reinterpret_cast<const __m128i *>(advance<ZX>(p, len));
677
678 if (len <= sizeof(__m256i))
679 return aeshash256_lt32_avx256<ZX>(state.state0_256(), p, len);
680
681 return aeshash256_ge32<ZX>(state.state0_256(), src, srcend, len);
682}
683# endif // VAES
684
685template <ZeroExtension ZX>
686static size_t QT_FUNCTION_TARGET(AES)
687aeshash128(const uchar *p, size_t len, size_t seed, size_t seed2) noexcept
688{
689 AESHashSeed state(seed, seed2);
690 auto src = reinterpret_cast<const __m128i *>(p);
691 const auto srcend = reinterpret_cast<const __m128i *>(advance<ZX>(p, len));
692
693 if (len < sizeof(__m128i))
694 return aeshash128_lt16<ZX>(state.state0, src, srcend, len);
695
696 if (len <= sizeof(__m256i))
697 return aeshash128_16to32<ZX>(state.state0, state.state1(), src, srcend);
698
699 return aeshash128_ge32<ZX>(state.state0, state.state1(), src, srcend);
700}
701
702template <ZeroExtension ZX = None>
703static size_t aeshash(const uchar *p, size_t len, size_t seed, size_t seed2) noexcept
704{
705 if constexpr (ZX == ByteToWord)
706 len *= 2; // see note above on ZX == ByteToWord hashing
707
708# if QT_COMPILER_SUPPORTS_HERE(VAES)
709 if (qCpuHasFeature(VAES)) {
710 if (qCpuHasFeature(AVX512VL))
711 return aeshash256_avx256<ZX>(p, len, seed, seed2);
712 return aeshash256<ZX>(p, len, seed, seed2);
713 }
714# endif
715 return aeshash128<ZX>(p, len, seed, seed2);
716}
717#endif // x86 AESNI
718
719#if defined(Q_PROCESSOR_ARM) && QT_COMPILER_SUPPORTS_HERE(CRYPTO) && !defined(QHASH_AES_SANITIZER_BUILD) && !defined(QT_BOOTSTRAPPED)
720QT_FUNCTION_TARGET(AES)
721static size_t aeshash(const uchar *p, size_t len, size_t seed, size_t seed2) noexcept
722{
723 uint8x16_t key;
724# if QT_POINTER_SIZE == 8
725 uint64x2_t vseed = vcombine_u64(vcreate_u64(seed), vcreate_u64(seed2));
726 key = vreinterpretq_u8_u64(vseed);
727# else
728
729 uint32x2_t vseed = vmov_n_u32(seed);
730 vseed = vset_lane_u32(seed2, vseed, 1);
731 key = vreinterpretq_u8_u32(vcombine_u32(vseed, vseed));
732# endif
733
734 // Compared to x86 AES, ARM splits each round into two instructions
735 // and includes the pre-xor instead of the post-xor.
736 const auto hash16bytes = [](uint8x16_t &state0, uint8x16_t data) QT_FUNCTION_TARGET(AES) {
737 auto state1 = state0;
738 state0 = vaeseq_u8(state0, data);
739 state0 = vaesmcq_u8(state0);
740 auto state2 = state0;
741 state0 = vaeseq_u8(state0, state1);
742 state0 = vaesmcq_u8(state0);
743 auto state3 = state0;
744 state0 = vaeseq_u8(state0, state2);
745 state0 = vaesmcq_u8(state0);
746 state0 = veorq_u8(state0, state3);
747 };
748
749 uint8x16_t state0 = key;
750
751 if (len < 8)
752 goto lt8;
753 if (len < 16)
754 goto lt16;
755 if (len < 32)
756 goto lt32;
757
758 // rounds of 32 bytes
759 {
760 // Make state1 = ~state0:
761 uint8x16_t state1 = veorq_u8(state0, vdupq_n_u8(255));
762
763 // do simplified rounds of 32 bytes: unlike the Go code, we only
764 // scramble twice and we keep 256 bits of state
765 const auto *e = p + len - 31;
766 while (p < e) {
767 uint8x16_t data0 = vld1q_u8(p);
768 uint8x16_t data1 = vld1q_u8(p + 16);
769 auto oldstate0 = state0;
770 auto oldstate1 = state1;
771 state0 = vaeseq_u8(state0, data0);
772 state1 = vaeseq_u8(state1, data1);
773 state0 = vaesmcq_u8(state0);
774 state1 = vaesmcq_u8(state1);
775 auto laststate0 = state0;
776 auto laststate1 = state1;
777 state0 = vaeseq_u8(state0, oldstate0);
778 state1 = vaeseq_u8(state1, oldstate1);
779 state0 = vaesmcq_u8(state0);
780 state1 = vaesmcq_u8(state1);
781 state0 = veorq_u8(state0, laststate0);
782 state1 = veorq_u8(state1, laststate1);
783 p += 32;
784 }
785 state0 = veorq_u8(state0, state1);
786 }
787 len &= 0x1f;
788
789 // do we still have 16 or more bytes?
790 if (len & 0x10) {
791lt32:
792 uint8x16_t data = vld1q_u8(p);
793 hash16bytes(state0, data);
794 p += 16;
795 }
796 len &= 0xf;
797
798 if (len & 0x08) {
799lt16:
800 uint8x8_t data8 = vld1_u8(p);
801 uint8x16_t data = vcombine_u8(data8, vdup_n_u8(0));
802 hash16bytes(state0, data);
803 p += 8;
804 }
805 len &= 0x7;
806
807lt8:
808 if (len) {
809 // load the last chunk of data
810 // We're going to load 8 bytes and mask zero the part we don't care
811 // (the hash of a short string is different from the hash of a longer
812 // including NULLs at the end because the length is in the key)
813 // WARNING: this may produce valgrind warnings, but it's safe
814
815 uint8x8_t data8;
816
817 if (Q_LIKELY(quintptr(p + 8) & 0xff8)) {
818 // same page, we definitely can't fault:
819 // load all 8 bytes and mask off the bytes past the end of the source
820 static const qint8 maskarray[] = {
821 -1, -1, -1, -1, -1, -1, -1,
822 0, 0, 0, 0, 0, 0, 0,
823 };
824 uint8x8_t mask = vld1_u8(reinterpret_cast<const quint8 *>(maskarray) + 7 - len);
825 data8 = vld1_u8(p);
826 data8 = vand_u8(data8, mask);
827 } else {
828 // too close to the end of the page, it could fault:
829 // load 8 bytes ending at the data end, then shuffle them to the beginning
830 static const qint8 shufflecontrol[] = {
831 1, 2, 3, 4, 5, 6, 7,
832 -1, -1, -1, -1, -1, -1, -1,
833 };
834 uint8x8_t control = vld1_u8(reinterpret_cast<const quint8 *>(shufflecontrol) + 7 - len);
835 data8 = vld1_u8(p - 8 + len);
836 data8 = vtbl1_u8(data8, control);
837 }
838 uint8x16_t data = vcombine_u8(data8, vdup_n_u8(0));
839 hash16bytes(state0, data);
840 }
841
842 // extract state0
843# if QT_POINTER_SIZE == 8
844 return vgetq_lane_u64(vreinterpretq_u64_u8(state0), 0);
845# else
846 return vgetq_lane_u32(vreinterpretq_u32_u8(state0), 0);
847# endif
848}
849#endif
850
851size_t qHashBits(const void *p, size_t size, size_t seed) noexcept
852{
853#ifdef QT_BOOTSTRAPPED
854 // the seed is always 0 in bootstrapped mode (no seed generation code),
855 // so help the compiler do dead code elimination
856 seed = 0;
857#endif
858 // mix in the length as a secondary seed. For seed == 0, seed2 must be
859 // size, to match what we used to do prior to Qt 6.2.
860 size_t seed2 = size;
861 if (seed)
862 seed2 = qt_qhash_seed.currentSeed(1);
863
864 auto data = reinterpret_cast<const uchar *>(p);
865#ifdef AESHASH
866 if (seed && qCpuHasFeature(AES) && qCpuHasFeature(SSE4_2))
867 return aeshash(data, size, seed, seed2);
868#elif defined(Q_PROCESSOR_ARM) && QT_COMPILER_SUPPORTS_HERE(CRYPTO) && !defined(QHASH_AES_SANITIZER_BUILD) && !defined(QT_BOOTSTRAPPED)
869 if (seed && qCpuHasFeature(AES))
870 return aeshash(data, size, seed, seed2);
871#endif
872
873 return qHashBits_fallback<>(data, size, seed, seed2);
874}
875
876size_t qHash(QByteArrayView key, size_t seed) noexcept
877{
878 return qHashBits(key.constData(), size_t(key.size()), seed);
879}
880
881size_t qHash(QStringView key, size_t seed) noexcept
882{
883 return qHashBits(key.data(), key.size()*sizeof(QChar), seed);
884}
885
886#ifndef QT_BOOTSTRAPPED
887size_t qHash(const QBitArray &bitArray, size_t seed) noexcept
888{
889 qsizetype m = bitArray.d.size() - 1;
890 size_t result = qHashBits(reinterpret_cast<const uchar *>(bitArray.d.constData()), size_t(qMax(0, m)), seed);
891
892 // deal with the last 0 to 7 bits manually, because we can't trust that
893 // the padding is initialized to 0 in bitArray.d
894 qsizetype n = bitArray.size();
895 if (n & 0x7)
896 result = ((result << 4) + bitArray.d.at(m)) & ((1 << n) - 1);
897 return result;
898}
899#endif
900
901size_t qHash(QLatin1StringView key, size_t seed) noexcept
902{
903#ifdef QT_BOOTSTRAPPED
904 // the seed is always 0 in bootstrapped mode (no seed generation code),
905 // so help the compiler do dead code elimination
906 seed = 0;
907#endif
908
909 auto data = reinterpret_cast<const uchar *>(key.data());
910 size_t size = key.size();
911
912 // Mix in the length as a secondary seed.
913 // Multiplied by 2 to match the byte size of the equiavlent UTF-16 string.
914 size_t seed2 = size * 2;
915 if (seed)
916 seed2 = qt_qhash_seed.currentSeed(1);
917
918#if defined(AESHASH)
919 if (seed && qCpuHasFeature(AES) && qCpuHasFeature(SSE4_2))
920 return aeshash<ByteToWord>(data, size, seed, seed2);
921#endif
922 return qHashBits_fallback<ByteToWord>(data, size, seed, seed2);
923}
924
925/*!
926 \class QHashSeed
927 \inmodule QtCore
928 \since 6.2
929
930 The QHashSeed class is used to convey the QHash seed. This is used
931 internally by QHash and provides three static member functions to allow
932 users to obtain the hash and to reset it.
933
934 QHash and the qHash() functions implement what is called as "salted hash".
935 The intent is that different applications and different instances of the
936 same application will produce different hashing values for the same input,
937 thus causing the ordering of elements in QHash to be unpredictable by
938 external observers. This improves the applications' resilience against
939 attacks that attempt to force hashing tables into degenerate mode.
940
941 Most applications will not need to deal directly with the hash seed, as
942 QHash will do so when needed. However, applications may wish to use this
943 for their own purposes in the same way as QHash does: as an
944 application-global random value (but see \l QRandomGenerator too). Note
945 that the global hash seed may change during the application's lifetime, if
946 the resetRandomGlobalSeed() function is called. Users of the global hash
947 need to store the value they are using and not rely on getting it again.
948
949 This class also implements functionality to set the hash seed to a
950 deterministic value, which the qHash() functions will take to mean that
951 they should use a fixed hashing function on their data too. This
952 functionality is only meant to be used in debugging applications. This
953 behavior can also be controlled by setting the \c QT_HASH_SEED environment
954 variable to the value zero (any other value is ignored).
955
956 \sa QHash, QRandomGenerator
957*/
958
959/*!
960 \fn QHashSeed::QHashSeed(size_t data)
961
962 Constructs a new QHashSeed object using \a data as the seed.
963 */
964
965/*!
966 \fn QHashSeed::operator size_t() const
967
968 Converts the returned hash seed into a \c size_t.
969 */
970
971/*!
972 \threadsafe
973
974 Returns the current global QHash seed. The value returned by this function
975 will be zero if setDeterministicGlobalSeed() has been called or if the
976 \c{QT_HASH_SEED} environment variable is set to zero.
977 */
978QHashSeed QHashSeed::globalSeed() noexcept
979{
980 return qt_qhash_seed.currentSeed(0);
981}
982
983/*!
984 \threadsafe
985
986 Forces the Qt hash seed to a deterministic value (zero) and asks the
987 qHash() functions to use a pre-determined hashing function. This mode is
988 only useful for debugging and should not be used in production code.
989
990 Regular operation can be restored by calling resetRandomGlobalSeed().
991 */
992void QHashSeed::setDeterministicGlobalSeed()
993{
994 qt_qhash_seed.clearSeed();
995}
996
997/*!
998 \threadsafe
999
1000 Reseeds the Qt hashing seed to a new, random value. Calling this function
1001 is not necessary, but long-running applications may want to do so after a
1002 long period of time in which information about its hash may have been
1003 exposed to potential attackers.
1004
1005 If the environment variable \c QT_HASH_SEED is set to zero, calling this
1006 function will result in a no-op.
1007
1008 Qt never calls this function during the execution of the application, but
1009 unless the \c QT_HASH_SEED variable is set to 0, the hash seed returned by
1010 globalSeed() will be a random value as if this function had been called.
1011 */
1012void QHashSeed::resetRandomGlobalSeed()
1013{
1014 qt_qhash_seed.resetSeed();
1015}
1016
1017#if QT_DEPRECATED_SINCE(6,6)
1018/*! \relates QHash
1019 \since 5.6
1020 \deprecated [6.6] Use QHashSeed::globalSeed() instead.
1021
1022 Returns the current global QHash seed.
1023
1024 The seed is set in any newly created QHash. See \l{qHash} about how this seed
1025 is being used by QHash.
1026
1027 \sa QHashSeed, QHashSeed::globalSeed()
1028 */
1029int qGlobalQHashSeed()
1030{
1031 return int(QHashSeed::globalSeed() & INT_MAX);
1032}
1033
1034/*! \relates QHash
1035 \since 5.6
1036 \deprecated [6.6] Use QHashSeed instead.
1037
1038 Sets the global QHash seed to \a newSeed.
1039
1040 Manually setting the global QHash seed value should be done only for testing
1041 and debugging purposes, when deterministic and reproducible behavior on a QHash
1042 is needed. We discourage to do it in production code as it can make your
1043 application susceptible to \l{algorithmic complexity attacks}.
1044
1045 From Qt 5.10 and onwards, the only allowed values are 0 and -1. Passing the
1046 value -1 will reinitialize the global QHash seed to a random value, while
1047 the value of 0 is used to request a stable algorithm for C++ primitive
1048 types types (like \c int) and string types (QString, QByteArray).
1049
1050 The seed is set in any newly created QHash. See \l{qHash} about how this seed
1051 is being used by QHash.
1052
1053 If the environment variable \c QT_HASH_SEED is set, calling this function will
1054 result in a no-op.
1055
1056 \sa QHashSeed::globalSeed(), QHashSeed
1057 */
1058void qSetGlobalQHashSeed(int newSeed)
1059{
1060 if (Q_LIKELY(newSeed == 0 || newSeed == -1)) {
1061 if (newSeed == 0)
1062 QHashSeed::setDeterministicGlobalSeed();
1063 else
1064 QHashSeed::resetRandomGlobalSeed();
1065 } else {
1066 // can't use qWarning here (reentrancy)
1067 fprintf(stderr, "qSetGlobalQHashSeed: forced seed value is not 0; ignoring call\n");
1068 }
1069}
1070#endif // QT_DEPRECATED_SINCE(6,6)
1071
1072/*!
1073 \internal
1074
1075 Private copy of the implementation of the Qt 4 qHash algorithm for strings,
1076 (that is, QChar-based arrays, so all QString-like classes),
1077 to be used wherever the result is somehow stored or reused across multiple
1078 Qt versions. The public qHash implementation can change at any time,
1079 therefore one must not rely on the fact that it will always give the same
1080 results.
1081
1082 The qt_hash functions must *never* change their results.
1083
1084 This function can hash discontiguous memory by invoking it on each chunk,
1085 passing the previous's result in the next call's \a chained argument.
1086*/
1087uint qt_hash(QStringView key, uint chained) noexcept
1088{
1089 uint h = chained;
1090
1091 for (auto c: key) {
1092 h = (h << 4) + c.unicode();
1093 h ^= (h & 0xf0000000) >> 23;
1094 }
1095 h &= 0x0fffffff;
1096 return h;
1097}
1098
1099/*!
1100 \fn template <typename T1, typename T2> size_t qHash(const std::pair<T1, T2> &key, size_t seed = 0)
1101 \since 5.7
1102 \qhashbuiltinTS{T1}{T2}
1103*/
1104
1105/*!
1106 \fn template <typename... T> size_t qHashMulti(size_t seed, const T &...args)
1107 \relates QHash
1108 \since 6.0
1109
1110 Returns the hash value for the \a{args}, using \a seed to seed
1111 the calculation, by successively applying qHash() to each
1112 element and combining the hash values into a single one.
1113
1114 Note that the order of the arguments is significant. If order does
1115 not matter, use qHashMultiCommutative() instead. If you are hashing raw
1116 memory, use qHashBits(); if you are hashing a range, use qHashRange().
1117
1118 This function is provided as a convenience to implement qHash() for
1119 your own custom types. For example, here's how you could implement
1120 a qHash() overload for a class \c{Employee}:
1121
1122 \snippet code/src_corelib_tools_qhash.cpp 13
1123
1124 \sa qHashMultiCommutative, qHashRange
1125*/
1126
1127/*!
1128 \fn template <typename... T> size_t qHashMultiCommutative(size_t seed, const T &...args)
1129 \relates QHash
1130 \since 6.0
1131
1132 Returns the hash value for the \a{args}, using \a seed to seed
1133 the calculation, by successively applying qHash() to each
1134 element and combining the hash values into a single one.
1135
1136 The order of the arguments is insignificant. If order does
1137 matter, use qHashMulti() instead, as it may produce better quality
1138 hashing. If you are hashing raw memory, use qHashBits(); if you are
1139 hashing a range, use qHashRange().
1140
1141 This function is provided as a convenience to implement qHash() for
1142 your own custom types.
1143
1144 \sa qHashMulti, qHashRange
1145*/
1146
1147/*! \fn template <typename InputIterator> size_t qHashRange(InputIterator first, InputIterator last, size_t seed = 0)
1148 \relates QHash
1149 \since 5.5
1150
1151 Returns the hash value for the range [\a{first},\a{last}), using \a seed
1152 to seed the calculation, by successively applying qHash() to each
1153 element and combining the hash values into a single one.
1154
1155 The return value of this function depends on the order of elements
1156 in the range. That means that
1157
1158 \snippet code/src_corelib_tools_qhash.cpp 30
1159
1160 and
1161 \snippet code/src_corelib_tools_qhash.cpp 31
1162
1163 hash to \b{different} values. If order does not matter, for example for hash
1164 tables, use qHashRangeCommutative() instead. If you are hashing raw
1165 memory, use qHashBits().
1166
1167 Use this function only to implement qHash() for your own custom
1168 types. For example, here's how you could implement a qHash() overload for
1169 std::vector<int>:
1170
1171 \snippet code/src_corelib_tools_qhash.cpp qhashrange
1172
1173 It bears repeating that the implementation of qHashRange() - like
1174 the qHash() overloads offered by Qt - may change at any time. You
1175 \b{must not} rely on the fact that qHashRange() will give the same
1176 results (for the same inputs) across different Qt versions, even
1177 if qHash() for the element type would.
1178
1179 \sa qHashBits(), qHashRangeCommutative()
1180*/
1181
1182/*! \fn template <typename InputIterator> size_t qHashRangeCommutative(InputIterator first, InputIterator last, size_t seed = 0)
1183 \relates QHash
1184 \since 5.5
1185
1186 Returns the hash value for the range [\a{first},\a{last}), using \a seed
1187 to seed the calculation, by successively applying qHash() to each
1188 element and combining the hash values into a single one.
1189
1190 The return value of this function does not depend on the order of
1191 elements in the range. That means that
1192
1193 \snippet code/src_corelib_tools_qhash.cpp 30
1194
1195 and
1196 \snippet code/src_corelib_tools_qhash.cpp 31
1197
1198 hash to the \b{same} values. If order matters, for example, for vectors
1199 and arrays, use qHashRange() instead. If you are hashing raw
1200 memory, use qHashBits().
1201
1202 Use this function only to implement qHash() for your own custom
1203 types. For example, here's how you could implement a qHash() overload for
1204 std::unordered_set<int>:
1205
1206 \snippet code/src_corelib_tools_qhash.cpp qhashrangecommutative
1207
1208 It bears repeating that the implementation of
1209 qHashRangeCommutative() - like the qHash() overloads offered by Qt
1210 - may change at any time. You \b{must not} rely on the fact that
1211 qHashRangeCommutative() will give the same results (for the same
1212 inputs) across different Qt versions, even if qHash() for the
1213 element type would.
1214
1215 \sa qHashBits(), qHashRange()
1216*/
1217
1218/*! \fn size_t qHashBits(const void *p, size_t len, size_t seed = 0)
1219 \relates QHash
1220 \since 5.4
1221
1222 Returns the hash value for the memory block of size \a len pointed
1223 to by \a p, using \a seed to seed the calculation.
1224
1225 Use this function only to implement qHash() for your own custom
1226 types. For example, here's how you could implement a qHash() overload for
1227 std::vector<int>:
1228
1229 \snippet code/src_corelib_tools_qhash.cpp qhashbits
1230
1231 This takes advantage of the fact that std::vector lays out its data
1232 contiguously. If that is not the case, or the contained type has
1233 padding, you should use qHashRange() instead.
1234
1235 It bears repeating that the implementation of qHashBits() - like
1236 the qHash() overloads offered by Qt - may change at any time. You
1237 \b{must not} rely on the fact that qHashBits() will give the same
1238 results (for the same inputs) across different Qt versions.
1239
1240 \sa qHashRange(), qHashRangeCommutative()
1241*/
1242
1243/*!
1244 \fn template <typename T, std::enable_if_t<std::is_same_v<T, bool>, bool> = true> size_t qHash(T key, size_t seed)
1245 \since 6.9
1246
1247 \qhashbuiltin
1248
1249 \note This is qHash(bool), constrained to accept only arguments of type bool,
1250 not arguments of types that merely convert to bool.
1251
1252 \note In Qt versions prior to 6.9, this overload was unintendedly provided by
1253 an undocumented 1-to-2-arg qHash adapter template function, with identical behavior.
1254*/
1255
1256/*! \fn size_t qHash(char key, size_t seed = 0)
1257 \since 5.0
1258 \qhashbuiltin
1259*/
1260
1261/*! \fn size_t qHash(uchar key, size_t seed = 0)
1262 \since 5.0
1263 \qhashbuiltin
1264*/
1265
1266/*! \fn size_t qHash(signed char key, size_t seed = 0)
1267 \since 5.0
1268 \qhashbuiltin
1269*/
1270
1271/*! \fn size_t qHash(ushort key, size_t seed = 0)
1272 \since 5.0
1273 \qhashbuiltin
1274*/
1275
1276/*! \fn size_t qHash(short key, size_t seed = 0)
1277 \since 5.0
1278 \qhashbuiltin
1279*/
1280
1281/*! \fn size_t qHash(uint key, size_t seed = 0)
1282 \since 5.0
1283 \qhashbuiltin
1284*/
1285
1286/*! \fn size_t qHash(int key, size_t seed = 0)
1287 \since 5.0
1288 \qhashbuiltin
1289*/
1290
1291/*! \fn size_t qHash(ulong key, size_t seed = 0)
1292 \since 5.0
1293 \qhashbuiltin
1294*/
1295
1296/*! \fn size_t qHash(long key, size_t seed = 0)
1297 \since 5.0
1298 \qhashbuiltin
1299*/
1300
1301/*! \fn size_t qHash(quint64 key, size_t seed = 0)
1302 \since 5.0
1303 \qhashbuiltin
1304*/
1305
1306/*! \fn size_t qHash(qint64 key, size_t seed = 0)
1307 \since 5.0
1308 \qhashbuiltin
1309*/
1310
1311/*! \fn size_t qHash(quint128 key, size_t seed = 0)
1312 \since 6.8
1313 \qhashbuiltin
1314
1315 \note This function is only available on platforms that support a native
1316 128-bit integer type.
1317*/
1318
1319/*! \fn size_t qHash(qint128 key, size_t seed = 0)
1320 \since 6.8
1321 \qhashbuiltin
1322
1323 \note This function is only available on platforms that support a native
1324 128-bit integer type.
1325 */
1326
1327/*! \fn size_t qHash(char8_t key, size_t seed = 0)
1328 \since 6.0
1329 \qhashbuiltin
1330*/
1331
1332/*! \fn size_t qHash(char16_t key, size_t seed = 0)
1333 \since 6.0
1334 \qhashbuiltin
1335*/
1336
1337/*! \fn size_t qHash(char32_t key, size_t seed = 0)
1338 \since 6.0
1339 \qhashbuiltin
1340*/
1341
1342/*! \fn size_t qHash(wchar_t key, size_t seed = 0)
1343 \since 6.0
1344 \qhashbuiltin
1345*/
1346
1347/*! \fn size_t qHash(float key, size_t seed = 0) noexcept
1348 \since 5.3
1349 \qhashbuiltin
1350*/
1351
1352/*!
1353 \since 5.3
1354 \qhashbuiltin
1355*/
1356size_t qHash(double key, size_t seed) noexcept
1357{
1358 // ensure -0 gets mapped to 0
1359 key += 0.0;
1360 if constexpr (sizeof(double) == sizeof(size_t)) {
1361 size_t k;
1362 memcpy(&k, &key, sizeof(double));
1363 return QHashPrivate::hash(k, seed);
1364 } else {
1365 return murmurhash(&key, sizeof(key), seed);
1366 }
1367}
1368
1369/*!
1370 \since 5.3
1371 \qhashbuiltin
1372*/
1373size_t qHash(long double key, size_t seed) noexcept
1374{
1375 // detect the actual size of long double's payload, not the space it
1376 // occupies in memory
1377 using Limits = std::numeric_limits<long double>;
1378 constexpr size_t SignSize = Limits::is_signed;
1379 constexpr quint64 ExponentRange = Limits::max_exponent - Limits::min_exponent;
1380 constexpr size_t ExponentSize = 64 - qCountLeadingZeroBits(ExponentRange);
1381 constexpr size_t Size = (Limits::digits + SignSize + ExponentSize) / 8;
1382
1383 if constexpr (sizeof(long double) == sizeof(double) || !Limits::is_iec559) {
1384 return qHash(double(key));
1385 } else {
1386#if defined(Q_PROCESSOR_X86) && defined(Q_CC_GNU_ONLY) && !defined(__LONG_DOUBLE_128__)
1387 // Check our calculation was right. long double is either:
1388 // 8 (matches the block above)
1389 // 10 (standard x87's IEEE 754 extended precision)
1390 // 16 (-mlong-double-128; Clang doesn't define __LONG_DOUBLE_128__)
1391 static_assert(Size == 10);
1392#endif
1393 alignas(long double) quint8 buffer[sizeof(long double)];
1394
1395 // ensure -0 gets mapped to 0
1396 key += static_cast<long double>(0.0);
1397 qToUnaligned(key, buffer);
1398
1399 // Work around: https://issuetracker.google.com/issues/400937647
1400 // An over-eager Bionic diagnostic in NDK 30 Clang LLVM 21.0.0:
1401 // "error: 'memset' will set 0 bytes; maybe the arguments got flipped? [-Werror,-Wuser-defined-warnings]""
1402 // Note! Trying to work around the problem with an if-constexpr does not work.
1403 size_t paddingSize = sizeof(long double) - Size;
1404 if (paddingSize > 0) {
1405 if constexpr (QSysInfo::ByteOrder == QSysInfo::BigEndian)
1406 (memset)(buffer, 0, paddingSize);
1407 else
1408 (memset)(buffer + Size, 0, paddingSize);
1409 }
1410 return murmurhash(buffer, sizeof(long double), seed);
1411 }
1412}
1413
1414/*!
1415 \fn template <typename Enum, std::enable_if_t<std::is_enum_v<Enum>, bool> = true> size_t qHash(Enum key, size_t seed)
1416 \since 6.5
1417 \qhashbuiltin
1418
1419 \note Prior to Qt 6.5, unscoped enums relied on the integer overloads of this
1420 function due to implicit conversion to their underlying integer types.
1421 For scoped enums, you had to implement an overload yourself. This is still the
1422 backwards-compatible fix to remain compatible with older Qt versions.
1423*/
1424
1425/*! \fn size_t qHash(const QChar key, size_t seed = 0)
1426 \since 5.0
1427 \qhashold{QHash}
1428*/
1429
1430/*! \fn size_t qHash(const QByteArray &key, size_t seed = 0)
1431 \since 5.0
1432 \qhashold{QHash}
1433*/
1434
1435/*! \fn size_t qHash(QByteArrayView key, size_t seed = 0)
1436 \since 6.0
1437 \qhashold{QHash}
1438*/
1439
1440/*! \fn size_t qHash(const QBitArray &key, size_t seed = 0)
1441 \since 5.0
1442 \qhashold{QHash}
1443*/
1444
1445/*! \fn size_t qHash(const QString &key, size_t seed = 0)
1446 \since 5.0
1447 \qhashold{QHash}
1448*/
1449
1450/*! \fn size_t qHash(QLatin1StringView key, size_t seed = 0)
1451 \since 5.0
1452 \qhashold{QHash}
1453*/
1454
1455/*! \fn template <class T> size_t qHash(const T *key, size_t seed = 0)
1456 \since 5.0
1457 \qhashbuiltin
1458*/
1459
1460/*! \fn size_t qHash(std::nullptr_t key, size_t seed = 0)
1461 \since 6.0
1462 \qhashbuiltin
1463*/
1464
1465/*! \fn template<typename T> bool qHashEquals(const T &a, const T &b)
1466 \relates QHash
1467 \since 6.0
1468 \internal
1469
1470 This method is being used by QHash to compare two keys. Returns true if the
1471 keys \a a and \a b are considered equal for hashing purposes.
1472
1473 The default implementation returns the result of (a == b). It can be reimplemented
1474 for a certain type if the equality operator is not suitable for hashing purposes.
1475 This is for example the case if the equality operator uses qFuzzyCompare to compare
1476 floating point values.
1477*/
1478
1479
1480/*!
1481 \class QHash
1482 \inmodule QtCore
1483 \brief The QHash class is a template class that provides a hash-table-based dictionary.
1484 \compares equality
1485
1486 \ingroup tools
1487 \ingroup shared
1488
1489 \reentrant
1490
1491 QHash<Key, T> is one of Qt's generic \l{container classes}, where
1492 \a Key is the type used for lookup keys and \a T is the mapped value
1493 type. It stores (key, value) pairs and provides very fast lookup of
1494 the value associated with a key.
1495
1496 QHash provides very similar functionality to QMap. The
1497 differences are:
1498
1499 \list
1500 \li QHash provides faster lookups than QMap. (See \l{Algorithmic
1501 Complexity} for details.)
1502 \li When iterating over a QMap, the items are always sorted by
1503 key. With QHash, the items are arbitrarily ordered.
1504 \li The key type of a QMap must provide operator<(). The key
1505 type of a QHash must provide operator==() and a global
1506 hash function called qHash() (see \l{qHash}).
1507 \endlist
1508
1509 Here's an example QHash with QString keys and \c int values:
1510 \snippet code/src_corelib_tools_qhash.cpp 0
1511
1512 To insert a (key, value) pair into the hash, you can use operator[]():
1513
1514 \snippet code/src_corelib_tools_qhash.cpp 1
1515
1516 This inserts the following three (key, value) pairs into the
1517 QHash: ("one", 1), ("three", 3), and ("seven", 7). Another way to
1518 insert items into the hash is to use insert():
1519
1520 \snippet code/src_corelib_tools_qhash.cpp 2
1521
1522 To look up a value, use operator[]() or value():
1523
1524 \snippet code/src_corelib_tools_qhash.cpp 3
1525
1526 If there is no item with the specified key in the hash, these
1527 functions return a \l{default-constructed value}.
1528
1529 If you want to check whether the hash contains a particular key,
1530 use contains():
1531
1532 \snippet code/src_corelib_tools_qhash.cpp 4
1533
1534 There is also a value() overload that uses its second argument as
1535 a default value if there is no item with the specified key:
1536
1537 \snippet code/src_corelib_tools_qhash.cpp 5
1538
1539 In general, we recommend that you use contains() and value()
1540 rather than operator[]() for looking up a key in a hash. The
1541 reason is that operator[]() silently inserts an item into the
1542 hash if no item exists with the same key (unless the hash is
1543 const). For example, the following code snippet will create 1000
1544 items in memory:
1545
1546 \snippet code/src_corelib_tools_qhash.cpp 6
1547
1548 To avoid this problem, replace \c hash[i] with \c hash.value(i)
1549 in the code above.
1550
1551 Internally, QHash uses a hash table to perform lookups. This
1552 hash table automatically grows to
1553 provide fast lookups without wasting too much memory. You can
1554 still control the size of the hash table by calling reserve() if
1555 you already know approximately how many items the QHash will
1556 contain, but this isn't necessary to obtain good performance. You
1557 can also call capacity() to retrieve the hash table's size.
1558
1559 QHash will not shrink automatically if items are removed from the
1560 table. To minimize the memory used by the hash, call squeeze().
1561
1562 To iterate through all the (key, value) pairs stored in a
1563 QHash, use \l {asKeyValueRange}():
1564
1565 \snippet code/src_corelib_tools_qhash.cpp 8
1566
1567 This function returns a range object that can be used with structured
1568 bindings. For manual iterator control, you can also use traditional
1569 \l{STL-style iterators} (QHash::const_iterator and QHash::iterator):
1570
1571 \snippet code/src_corelib_tools_qhash.cpp qhash-iterator-stl-style
1572
1573 To modify values, use iterators:
1574
1575 \snippet code/src_corelib_tools_qhash.cpp qhash-iterator-modify-values
1576
1577 QHash also provides \l{Java-style iterators} (QHashIterator and
1578 QMutableHashIterator) for compatibility.
1579
1580 QHash is unordered, so an iterator's sequence cannot be assumed
1581 to be predictable. If ordering by key is required, use a QMap.
1582
1583 A QHash allows only one value per key. If you call
1584 insert() with a key that already exists in the QHash, the
1585 previous value is erased. For example:
1586
1587 \snippet code/src_corelib_tools_qhash.cpp 9
1588
1589 If you need to store multiple entries for the same key in the
1590 hash table, use \l{QMultiHash}.
1591
1592 If you only need to extract the values from a hash (not the keys),
1593 you can also use range-based for:
1594
1595 \snippet code/src_corelib_tools_qhash.cpp 12
1596
1597 Items can be removed from the hash in several ways. One way is to
1598 call remove(); this will remove any item with the given key.
1599 Another way is to use QMutableHashIterator::remove(). In addition,
1600 you can clear the entire hash using clear().
1601
1602 QHash's key and value data types must be \l{assignable data
1603 types}. You cannot, for example, store a QWidget as a value;
1604 instead, store a QWidget *.
1605
1606 \target qHash
1607 \section2 The hashing function
1608
1609 A QHash's key type has additional requirements other than being an
1610 assignable data type: it must provide operator==(), and there must also be
1611 a hashing function that returns a hash value for an argument of the
1612 key's type.
1613
1614 The hashing function computes a numeric value based on a key. It
1615 can use any algorithm imaginable, as long as it always returns
1616 the same value if given the same argument. In other words, if
1617 \c{e1 == e2}, then \c{hash(e1) == hash(e2)} must hold as well.
1618 However, to obtain good performance, the hashing function should
1619 attempt to return different hash values for different keys to the
1620 largest extent possible.
1621
1622 A hashing function for a key type \c{K} may be provided in two
1623 different ways.
1624
1625 The first way is by having an overload of \c{qHash()} in \c{K}'s
1626 namespace. The \c{qHash()} function must have one of these signatures:
1627
1628 \snippet code/src_corelib_tools_qhash.cpp 32
1629
1630 The two-arguments overloads take an unsigned integer that should be used to
1631 seed the calculation of the hash function. This seed is provided by QHash
1632 in order to prevent a family of \l{algorithmic complexity attacks}.
1633
1634 \note In Qt 6 it is possible to define a \c{qHash()} overload
1635 taking only one argument; support for this is deprecated. Starting
1636 with Qt 7, it will be mandatory to use a two-arguments overload. If
1637 both a one-argument and a two-arguments overload are defined for a
1638 key type, the latter is used by QHash (note that you can simply
1639 define a two-arguments version, and use a default value for the
1640 seed parameter). In Qt 6 it is possible to disable support for the
1641 single argument qHash overload by defining the
1642 \c{QT_NO_SINGLE_ARGUMENT_QHASH_OVERLOAD} macro.
1643
1644 The second way to provide a hashing function is by specializing
1645 the \c{std::hash} class for the key type \c{K}, and providing a
1646 suitable function call operator for it:
1647
1648 \snippet code/src_corelib_tools_qhash.cpp 33
1649
1650 The seed argument has the same meaning as for \c{qHash()},
1651 and may be left out.
1652
1653 This second way allows to reuse the same hash function between
1654 QHash and the C++ Standard Library unordered associative containers.
1655 If both a \c{qHash()} overload and a \c{std::hash} specializations
1656 are provided for a type, then the \c{qHash()} overload is preferred.
1657
1658 Here's a partial list of the C++ and Qt types that can serve as keys in a
1659 QHash: any integer type (char, unsigned long, etc.), any pointer type,
1660 QChar, QString, and QByteArray. For all of these, the \c <QHash> header
1661 defines a qHash() function that computes an adequate hash value. Many other
1662 Qt classes also declare a qHash overload for their type; please refer to
1663 the documentation of each class.
1664
1665 If you want to use other types as the key, make sure that you provide
1666 operator==() and a hash implementation.
1667
1668 The convenience qHashMulti() function can be used to implement
1669 qHash() for a custom type, where one usually wants to produce a
1670 hash value from multiple fields:
1671
1672 Example:
1673 \snippet code/src_corelib_tools_qhash.cpp 13
1674
1675 In the example above, we've relied on Qt's own implementation of
1676 qHash() for QString and QDate to give us a hash value for the
1677 employee's name and date of birth respectively.
1678
1679 Note that the implementation of the qHash() overloads offered by Qt
1680 may change at any time. You \b{must not} rely on the fact that qHash()
1681 will give the same results (for the same inputs) across different Qt
1682 versions.
1683
1684 \section2 Algorithmic complexity attacks
1685
1686 All hash tables are vulnerable to a particular class of denial of service
1687 attacks, in which the attacker carefully pre-computes a set of different
1688 keys that are going to be hashed in the same bucket of a hash table (or
1689 even have the very same hash value). The attack aims at getting the
1690 worst-case algorithmic behavior (O(n) instead of amortized O(1), see
1691 \l{Algorithmic Complexity} for the details) when the data is fed into the
1692 table.
1693
1694 In order to avoid this worst-case behavior, the calculation of the hash
1695 value done by qHash() can be salted by a random seed, that nullifies the
1696 attack's extent. This seed is automatically generated by QHash once per
1697 process, and then passed by QHash as the second argument of the
1698 two-arguments overload of the qHash() function.
1699
1700 This randomization of QHash is enabled by default. Even though programs
1701 should never depend on a particular QHash ordering, there may be situations
1702 where you temporarily need deterministic behavior, for example for debugging or
1703 regression testing. To disable the randomization, define the environment
1704 variable \c QT_HASH_SEED to have the value 0. Alternatively, you can call
1705 the QHashSeed::setDeterministicGlobalSeed() function.
1706
1707 \sa QHashIterator, QMutableHashIterator, QMap, QSet
1708*/
1709
1710/*! \fn template <class Key, class T> QHash<Key, T>::QHash()
1711
1712 Constructs an empty hash.
1713
1714 \sa clear()
1715*/
1716
1717/*!
1718 \fn template <class Key, class T> QHash<Key, T>::QHash(QHash &&other)
1719
1720 Move-constructs a QHash instance, making it point at the same
1721 object that \a other was pointing to.
1722
1723 \since 5.2
1724*/
1725
1726/*! \fn template <class Key, class T> QHash<Key, T>::QHash(std::initializer_list<std::pair<Key,T> > list)
1727 \since 5.1
1728
1729 Constructs a hash with a copy of each of the elements in the
1730 initializer list \a list.
1731*/
1732
1733/*! \fn template <class Key, class T> template <class InputIterator> QHash<Key, T>::QHash(InputIterator begin, InputIterator end)
1734 \since 5.14
1735
1736 Constructs a hash with a copy of each of the elements in the iterator range
1737 [\a begin, \a end). Either the elements iterated by the range must be
1738 objects with \c{first} and \c{second} data members (like \c{std::pair}),
1739 convertible to \c Key and to \c T respectively; or the
1740 iterators must have \c{key()} and \c{value()} member functions, returning a
1741 key convertible to \c Key and a value convertible to \c T respectively.
1742*/
1743
1744/*! \fn template <class Key, class T> QHash<Key, T>::QHash(const QHash &other)
1745
1746 Constructs a copy of \a other.
1747
1748 This operation occurs in \l{constant time}, because QHash is
1749 \l{implicitly shared}. This makes returning a QHash from a
1750 function very fast. If a shared instance is modified, it will be
1751 copied (copy-on-write), and this takes \l{linear time}.
1752
1753 \sa operator=()
1754*/
1755
1756/*! \fn template <class Key, class T> QHash<Key, T>::~QHash()
1757
1758 Destroys the hash. References to the values in the hash and all
1759 iterators of this hash become invalid.
1760*/
1761
1762/*! \fn template <class Key, class T> QHash &QHash<Key, T>::operator=(const QHash &other)
1763
1764 Assigns \a other to this hash and returns a reference to this hash.
1765*/
1766
1767/*!
1768 \fn template <class Key, class T> QHash &QHash<Key, T>::operator=(QHash &&other)
1769
1770 Move-assigns \a other to this QHash instance.
1771
1772 \since 5.2
1773*/
1774
1775/*! \fn template <class Key, class T> void QHash<Key, T>::swap(QHash &other)
1776 \since 4.8
1777 \memberswap{hash}
1778*/
1779
1780/*! \fn template <class Key, class T> void QMultiHash<Key, T>::swap(QMultiHash &other)
1781 \since 4.8
1782 \memberswap{multi-hash}
1783*/
1784
1785/*! \fn template <class Key, class T> bool QHash<Key, T>::operator==(const QHash &lhs, const QHash &rhs)
1786
1787 Returns \c true if \a lhs hash is equal to \a rhs hash; otherwise returns
1788 \c false.
1789
1790 Two hashes are considered equal if they contain the same (key,
1791 value) pairs.
1792
1793 This function requires the value type to implement \c operator==().
1794
1795 \sa operator!=()
1796*/
1797
1798/*! \fn template <class Key, class T> bool QHash<Key, T>::operator!=(const QHash &lhs, const QHash &rhs)
1799
1800 Returns \c true if \a lhs hash is not equal to \a rhs hash; otherwise
1801 returns \c false.
1802
1803 Two hashes are considered equal if they contain the same (key,
1804 value) pairs.
1805
1806 This function requires the value type to implement \c operator==().
1807
1808 \sa operator==()
1809*/
1810
1811/*! \fn template <class Key, class T> qsizetype QHash<Key, T>::size() const
1812
1813 Returns the number of items in the hash.
1814
1815 \sa isEmpty(), count()
1816*/
1817
1818/*! \fn template <class Key, class T> bool QHash<Key, T>::isEmpty() const
1819
1820 Returns \c true if the hash contains no items; otherwise returns
1821 false.
1822
1823 \sa size()
1824*/
1825
1826/*! \fn template <class Key, class T> qsizetype QHash<Key, T>::capacity() const
1827
1828 Returns the number of buckets in the QHash's internal hash table.
1829
1830 The sole purpose of this function is to provide a means of fine
1831 tuning QHash's memory usage. In general, you will rarely ever
1832 need to call this function. If you want to know how many items are
1833 in the hash, call size().
1834
1835 \sa reserve(), squeeze()
1836*/
1837
1838/*! \fn template <class Key, class T> float QHash<Key, T>::load_factor() const noexcept
1839
1840 Returns the current load factor of the QHash's internal hash table.
1841 This is the same as capacity()/size(). The implementation used
1842 will aim to keep the load factor between 0.25 and 0.5. This avoids
1843 having too many hash table collisions that would degrade performance.
1844
1845 Even with a low load factor, the implementation of the hash table has a
1846 very low memory overhead.
1847
1848 This method purely exists for diagnostic purposes and you should rarely
1849 need to call it yourself.
1850
1851 \sa reserve(), squeeze()
1852*/
1853
1854
1855/*! \fn template <class Key, class T> void QHash<Key, T>::reserve(qsizetype size)
1856
1857 Ensures that the QHash's internal hash table has space to store at
1858 least \a size items without having to grow the hash table.
1859
1860 This implies that the hash table will contain at least 2 * \a size buckets
1861 to ensure good performance
1862
1863 This function is useful for code that needs to build a huge hash
1864 and wants to avoid repeated reallocation. For example:
1865
1866 \snippet code/src_corelib_tools_qhash.cpp 14
1867
1868 Ideally, \a size should be the maximum number of items expected
1869 in the hash. QHash will then choose the smallest possible
1870 number of buckets that will allow storing \a size items in the table
1871 without having to grow the internal hash table. If \a size
1872 is an underestimate, the worst that will happen is that the QHash
1873 will be a bit slower.
1874
1875 In general, you will rarely ever need to call this function.
1876 QHash's internal hash table automatically grows to
1877 provide good performance without wasting too much memory.
1878
1879 \sa squeeze(), capacity()
1880*/
1881
1882/*! \fn template <class Key, class T> void QHash<Key, T>::squeeze()
1883
1884 Reduces the size of the QHash's internal hash table to save
1885 memory.
1886
1887 The sole purpose of this function is to provide a means of fine
1888 tuning QHash's memory usage. In general, you will rarely ever
1889 need to call this function.
1890
1891 \sa reserve(), capacity()
1892*/
1893
1894/*! \fn template <class Key, class T> void QHash<Key, T>::detach()
1895
1896 \internal
1897
1898 Detaches this hash from any other hashes with which it may share
1899 data.
1900
1901 \sa isDetached()
1902*/
1903
1904/*! \fn template <class Key, class T> bool QHash<Key, T>::isDetached() const
1905
1906 \internal
1907
1908 Returns \c true if the hash's internal data isn't shared with any
1909 other hash object; otherwise returns \c false.
1910
1911 \sa detach()
1912*/
1913
1914/*! \fn template <class Key, class T> bool QHash<Key, T>::isSharedWith(const QHash &other) const
1915
1916 \internal
1917
1918 Returns true if the internal hash table of this QHash is shared with \a other, otherwise false.
1919*/
1920
1921/*! \fn template <class Key, class T> void QHash<Key, T>::clear()
1922
1923 Removes all items from the hash and frees up all memory used by it.
1924
1925 \sa remove()
1926*/
1927
1928/*! \fn template <class Key, class T> bool QHash<Key, T>::remove(const Key &key)
1929
1930 Removes the item that has the \a key from the hash.
1931 Returns true if the key exists in the hash and the item has been removed,
1932 and false otherwise.
1933
1934 \sa clear(), take()
1935*/
1936
1937/*! \fn template <class Key, class T> template <typename Predicate> qsizetype QHash<Key, T>::removeIf(Predicate pred)
1938 \since 6.1
1939
1940 Removes all elements for which the predicate \a pred returns true
1941 from the hash.
1942
1943 The function supports predicates which take either an argument of
1944 type \c{QHash<Key, T>::iterator}, or an argument of type
1945 \c{std::pair<const Key &, T &>}.
1946
1947 Returns the number of elements removed, if any.
1948
1949 \sa clear(), take()
1950*/
1951
1952/*! \fn template <class Key, class T> T QHash<Key, T>::take(const Key &key)
1953
1954 Removes the item with the \a key from the hash and returns
1955 the value associated with it.
1956
1957 If the item does not exist in the hash, the function simply
1958 returns a \l{default-constructed value}.
1959
1960 If you don't use the return value, remove() is more efficient.
1961
1962 \sa remove()
1963*/
1964
1965/*! \fn template <class Key, class T> bool QHash<Key, T>::contains(const Key &key) const
1966
1967 Returns \c true if the hash contains an item with the \a key;
1968 otherwise returns \c false.
1969
1970 \sa count()
1971*/
1972
1973/*! \fn template <class Key, class T> T QHash<Key, T>::value(const Key &key) const
1974 \fn template <class Key, class T> T QHash<Key, T>::value(const Key &key, const T &defaultValue) const
1975 \overload
1976
1977 Returns the value associated with the \a key.
1978
1979 If the hash contains no item with the \a key, the function
1980 returns \a defaultValue, or a \l{default-constructed value} if this
1981 parameter has not been supplied.
1982*/
1983
1984/*! \fn template <class Key, class T> T &QHash<Key, T>::operator[](const Key &key)
1985
1986 Returns the value associated with the \a key as a modifiable
1987 reference.
1988
1989 If the hash contains no item with the \a key, the function inserts
1990 a \l{default-constructed value} into the hash with the \a key, and
1991 returns a reference to it.
1992
1993//! [qhash-iterator-invalidation-func-desc]
1994 \warning Returned iterators/references should be considered invalidated
1995 the next time you call a non-const function on the hash, or when the
1996 hash is destroyed.
1997//! [qhash-iterator-invalidation-func-desc]
1998
1999 \sa insert(), value()
2000*/
2001
2002/*! \fn template <class Key, class T> const T QHash<Key, T>::operator[](const Key &key) const
2003
2004 \overload
2005
2006 Same as value().
2007*/
2008
2009/*! \fn template <class Key, class T> QList<Key> QHash<Key, T>::keys() const
2010
2011 Returns a list containing all the keys in the hash, in an
2012 arbitrary order.
2013
2014 The order is guaranteed to be the same as that used by values().
2015
2016 This function creates a new list, in \l {linear time}. The time and memory
2017 use that entails can be avoided by iterating from \l keyBegin() to
2018 \l keyEnd().
2019
2020 \sa values(), key()
2021*/
2022
2023/*! \fn template <class Key, class T> QList<Key> QHash<Key, T>::keys(const T &value) const
2024
2025 \overload
2026
2027 Returns a list containing all the keys associated with value \a
2028 value, in an arbitrary order.
2029
2030 This function can be slow (\l{linear time}), because QHash's
2031 internal data structure is optimized for fast lookup by key, not
2032 by value.
2033*/
2034
2035/*! \fn template <class Key, class T> QList<T> QHash<Key, T>::values() const
2036
2037 Returns a list containing all the values in the hash, in an
2038 arbitrary order.
2039
2040 The order is guaranteed to be the same as that used by keys().
2041
2042 This function creates a new list, in \l {linear time}. The time and memory
2043 use that entails can be avoided by iterating from \l keyValueBegin() to
2044 \l keyValueEnd().
2045
2046 \sa keys(), value()
2047*/
2048
2049/*!
2050 \fn template <class Key, class T> Key QHash<Key, T>::key(const T &value) const
2051 \fn template <class Key, class T> Key QHash<Key, T>::key(const T &value, const Key &defaultKey) const
2052 \since 4.3
2053
2054 Returns the first key mapped to \a value. If the hash contains no item
2055 mapped to \a value, returns \a defaultKey, or a \l{default-constructed
2056 value}{default-constructed key} if this parameter has not been supplied.
2057
2058 This function can be slow (\l{linear time}), because QHash's
2059 internal data structure is optimized for fast lookup by key, not
2060 by value.
2061*/
2062
2063/*! \fn template <class Key, class T> qsizetype QHash<Key, T>::count(const Key &key) const
2064
2065 Returns the number of items associated with the \a key.
2066
2067 \sa contains()
2068*/
2069
2070/*! \fn template <class Key, class T> qsizetype QHash<Key, T>::count() const
2071
2072 \overload
2073
2074 Same as size().
2075*/
2076
2077/*! \fn template <class Key, class T> QHash<Key, T>::iterator QHash<Key, T>::begin()
2078
2079 Returns an \l{STL-style iterators}{STL-style iterator} pointing to the first item in
2080 the hash.
2081
2082 \include qhash.cpp qhash-iterator-invalidation-func-desc
2083
2084 \sa constBegin(), end()
2085*/
2086
2087/*! \fn template <class Key, class T> QHash<Key, T>::const_iterator QHash<Key, T>::begin() const
2088
2089 \overload
2090
2091 \include qhash.cpp qhash-iterator-invalidation-func-desc
2092*/
2093
2094/*! \fn template <class Key, class T> QHash<Key, T>::const_iterator QHash<Key, T>::cbegin() const
2095 \since 5.0
2096
2097 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the first item
2098 in the hash.
2099
2100 \include qhash.cpp qhash-iterator-invalidation-func-desc
2101
2102 \sa begin(), cend()
2103*/
2104
2105/*! \fn template <class Key, class T> QHash<Key, T>::const_iterator QHash<Key, T>::constBegin() const
2106
2107 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the first item
2108 in the hash.
2109
2110 \include qhash.cpp qhash-iterator-invalidation-func-desc
2111
2112 \sa begin(), constEnd()
2113*/
2114
2115/*! \fn template <class Key, class T> QHash<Key, T>::key_iterator QHash<Key, T>::keyBegin() const
2116 \since 5.6
2117
2118 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the first key
2119 in the hash.
2120
2121 \include qhash.cpp qhash-iterator-invalidation-func-desc
2122
2123 \sa keyEnd()
2124*/
2125
2126/*! \fn template <class Key, class T> QHash<Key, T>::iterator QHash<Key, T>::end()
2127
2128 Returns an \l{STL-style iterators}{STL-style iterator} pointing to the imaginary item
2129 after the last item in the hash.
2130
2131 \include qhash.cpp qhash-iterator-invalidation-func-desc
2132
2133 \sa begin(), constEnd()
2134*/
2135
2136/*! \fn template <class Key, class T> QHash<Key, T>::const_iterator QHash<Key, T>::end() const
2137
2138 \overload
2139
2140 \include qhash.cpp qhash-iterator-invalidation-func-desc
2141*/
2142
2143/*! \fn template <class Key, class T> QHash<Key, T>::const_iterator QHash<Key, T>::constEnd() const
2144
2145 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the imaginary
2146 item after the last item in the hash.
2147
2148 \include qhash.cpp qhash-iterator-invalidation-func-desc
2149
2150 \sa constBegin(), end()
2151*/
2152
2153/*! \fn template <class Key, class T> QHash<Key, T>::const_iterator QHash<Key, T>::cend() const
2154 \since 5.0
2155
2156 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the imaginary
2157 item after the last item in the hash.
2158
2159 \include qhash.cpp qhash-iterator-invalidation-func-desc
2160
2161 \sa cbegin(), end()
2162*/
2163
2164/*! \fn template <class Key, class T> QHash<Key, T>::key_iterator QHash<Key, T>::keyEnd() const
2165 \since 5.6
2166
2167 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the imaginary
2168 item after the last key in the hash.
2169
2170 \include qhash.cpp qhash-iterator-invalidation-func-desc
2171
2172 \sa keyBegin()
2173*/
2174
2175/*! \fn template <class Key, class T> QHash<Key, T>::key_value_iterator QHash<Key, T>::keyValueBegin()
2176 \since 5.10
2177
2178 Returns an \l{STL-style iterators}{STL-style iterator} pointing to the first entry
2179 in the hash.
2180
2181 \include qhash.cpp qhash-iterator-invalidation-func-desc
2182
2183 \sa keyValueEnd()
2184*/
2185
2186/*! \fn template <class Key, class T> QHash<Key, T>::key_value_iterator QHash<Key, T>::keyValueEnd()
2187 \since 5.10
2188
2189 Returns an \l{STL-style iterators}{STL-style iterator} pointing to the imaginary
2190 entry after the last entry in the hash.
2191
2192 \include qhash.cpp qhash-iterator-invalidation-func-desc
2193
2194 \sa keyValueBegin()
2195*/
2196
2197/*! \fn template <class Key, class T> QHash<Key, T>::const_key_value_iterator QHash<Key, T>::keyValueBegin() const
2198 \since 5.10
2199
2200 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the first entry
2201 in the hash.
2202
2203 \include qhash.cpp qhash-iterator-invalidation-func-desc
2204
2205 \sa keyValueEnd()
2206*/
2207
2208/*! \fn template <class Key, class T> QHash<Key, T>::const_key_value_iterator QHash<Key, T>::constKeyValueBegin() const
2209 \since 5.10
2210
2211 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the first entry
2212 in the hash.
2213
2214 \include qhash.cpp qhash-iterator-invalidation-func-desc
2215
2216 \sa keyValueBegin()
2217*/
2218
2219/*! \fn template <class Key, class T> QHash<Key, T>::const_key_value_iterator QHash<Key, T>::keyValueEnd() const
2220 \since 5.10
2221
2222 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the imaginary
2223 entry after the last entry in the hash.
2224
2225 \include qhash.cpp qhash-iterator-invalidation-func-desc
2226
2227 \sa keyValueBegin()
2228*/
2229
2230/*! \fn template <class Key, class T> QHash<Key, T>::const_key_value_iterator QHash<Key, T>::constKeyValueEnd() const
2231 \since 5.10
2232
2233 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the imaginary
2234 entry after the last entry in the hash.
2235
2236 \include qhash.cpp qhash-iterator-invalidation-func-desc
2237
2238 \sa constKeyValueBegin()
2239*/
2240
2241/*! \fn template <class Key, class T> auto QHash<Key, T>::asKeyValueRange() &
2242 \fn template <class Key, class T> auto QHash<Key, T>::asKeyValueRange() const &
2243 \fn template <class Key, class T> auto QHash<Key, T>::asKeyValueRange() &&
2244 \fn template <class Key, class T> auto QHash<Key, T>::asKeyValueRange() const &&
2245 \since 6.4
2246
2247 Returns a range object that allows iteration over this hash as
2248 key/value pairs. For instance, this range object can be used in a
2249 range-based for loop, in combination with a structured binding declaration:
2250
2251 \snippet code/src_corelib_tools_qhash.cpp 34
2252
2253 Note that both the key and the value obtained this way are
2254 references to the ones in the hash. Specifically, mutating the value
2255 will modify the hash itself.
2256
2257 \include qhash.cpp qhash-iterator-invalidation-func-desc
2258
2259 \sa QKeyValueIterator
2260*/
2261
2262/*! \fn template <class Key, class T> QHash<Key, T>::iterator QHash<Key, T>::erase(const_iterator pos)
2263 \since 5.7
2264
2265 Removes the (key, value) pair associated with the iterator \a pos
2266 from the hash, and returns an iterator to the next item in the
2267 hash.
2268
2269 This function never causes QHash to
2270 rehash its internal data structure. This means that it can safely
2271 be called while iterating, and won't affect the order of items in
2272 the hash. For example:
2273
2274 \snippet code/src_corelib_tools_qhash.cpp 15
2275
2276 \include qhash.cpp qhash-iterator-invalidation-func-desc
2277
2278 \sa remove(), take(), find()
2279*/
2280
2281/*! \fn template <class Key, class T> QHash<Key, T>::iterator QHash<Key, T>::find(const Key &key)
2282
2283 Returns an iterator pointing to the item with the \a key in the
2284 hash.
2285
2286 If the hash contains no item with the \a key, the function
2287 returns end().
2288
2289 If the hash contains multiple items with the \a key, this
2290 function returns an iterator that points to the most recently
2291 inserted value. The other values are accessible by incrementing
2292 the iterator. For example, here's some code that iterates over all
2293 the items with the same key:
2294
2295 \snippet code/src_corelib_tools_qhash.cpp 16
2296
2297 \include qhash.cpp qhash-iterator-invalidation-func-desc
2298
2299 \sa value(), values()
2300*/
2301
2302/*! \fn template <class Key, class T> QHash<Key, T>::const_iterator QHash<Key, T>::find(const Key &key) const
2303
2304 \overload
2305
2306 \include qhash.cpp qhash-iterator-invalidation-func-desc
2307*/
2308
2309/*! \fn template <class Key, class T> QHash<Key, T>::const_iterator QHash<Key, T>::constFind(const Key &key) const
2310 \since 4.1
2311
2312 Returns an iterator pointing to the item with the \a key in the
2313 hash.
2314
2315 If the hash contains no item with the \a key, the function
2316 returns constEnd().
2317
2318 \include qhash.cpp qhash-iterator-invalidation-func-desc
2319
2320 \sa find()
2321*/
2322
2323/*! \fn template <class Key, class T> QHash<Key, T>::iterator QHash<Key, T>::insert(const Key &key, const T &value)
2324
2325 Inserts a new item with the \a key and a value of \a value.
2326
2327 If there is already an item with the \a key, that item's value
2328 is replaced with \a value.
2329
2330 Returns an iterator pointing to the new/updated element.
2331
2332 \include qhash.cpp qhash-iterator-invalidation-func-desc
2333*/
2334
2335/*!
2336 \fn template <class Key, class T> QHash<Key, T>::iterator QHash<Key, T>::insert(const Key &key, T &&value)
2337 \fn template <class Key, class T> QHash<Key, T>::iterator QHash<Key, T>::insert(Key &&key, const T &value)
2338 \fn template <class Key, class T> QHash<Key, T>::iterator QHash<Key, T>::insert(Key &&key, T &&value)
2339 \since 6.11
2340 \overload
2341*/
2342
2343/*!
2344 \fn template <class Key, class T> template <typename ...Args> QHash<Key, T>::iterator QHash<Key, T>::emplace(const Key &key, Args&&... args)
2345 \fn template <class Key, class T> template <typename ...Args> QHash<Key, T>::iterator QHash<Key, T>::emplace(Key &&key, Args&&... args)
2346
2347 Inserts a new element into the container. This new element
2348 is constructed in-place using \a args as the arguments for its
2349 construction.
2350
2351 Returns an iterator pointing to the new element.
2352
2353 \include qhash.cpp qhash-iterator-invalidation-func-desc
2354*/
2355
2356/*!
2357 \class QHash::TryEmplaceResult
2358 \inmodule QtCore
2359 \since 6.9
2360 \ingroup tools
2361 \brief The TryEmplaceResult class is used to represent the result of a tryEmplace() operation.
2362
2363 The \c{TryEmplaceResult} class is used in QHash to represent the result
2364 of a tryEmplace() operation. It holds an \l{iterator} to the newly
2365 created item, or to the pre-existing item that prevented the insertion, and
2366 a boolean, \l{inserted}, denoting whether the insertion took place.
2367
2368 \sa QHash, QHash::tryEmplace()
2369*/
2370
2371/*!
2372 \variable QHash::TryEmplaceResult::iterator
2373
2374 Holds the iterator to the newly inserted element, or the element that
2375 prevented the insertion.
2376*/
2377
2378/*!
2379 \variable QHash::TryEmplaceResult::inserted
2380
2381 This value is \c{false} if there was already an entry with the same key.
2382*/
2383
2384/*!
2385 \fn template <class Key, class T> template <typename... Args> QHash<Key, T>::TryEmplaceResult QHash<Key, T>::tryEmplace(const Key &key, Args &&...args)
2386 \fn template <class Key, class T> template <typename... Args> QHash<Key, T>::TryEmplaceResult QHash<Key, T>::tryEmplace(Key &&key, Args &&...args)
2387 \fn template <class Key, class T> template <typename K, typename... Args, QHash<Key, T>::if_heterogeneously_searchable<K> = true, QHash<Key, T>::if_key_constructible_from<K> = true> QHash<Key, T>::TryEmplaceResult QHash<Key, T>::tryEmplace(K &&key, Args &&...args)
2388 \since 6.9
2389
2390 Inserts a new item with the \a key and a value constructed from \a args.
2391 If an item with \a key already exists, no insertion takes place.
2392
2393 Returns an instance of \l{TryEmplaceResult}, a structure that holds an
2394 \l{QHash::TryEmplaceResult::}{iterator} to the newly created item, or
2395 to the pre-existing item that prevented the insertion, and a boolean,
2396 \l{QHash::TryEmplaceResult::}{inserted}, denoting whether the insertion
2397 took place.
2398
2399 For example, this can be used to avoid the pattern of comparing old and
2400 new size or double-lookups. Where you might previously have written code like:
2401
2402 \code
2403 QHash<int, MyType> hash;
2404 // [...]
2405 int myKey = getKey();
2406 qsizetype oldSize = hash.size();
2407 MyType &elem = hash[myKey];
2408 if (oldSize != hash.size()) // Size changed: new element!
2409 initialize(elem);
2410 // [use elem...]
2411 \endcode
2412
2413 You can instead write:
2414
2415 \code
2416 QHash<int, MyType> hash;
2417 // [...]
2418 int myKey = getKey();
2419 auto result = hash.tryEmplace(myKey);
2420 if (result.inserted) // New element!
2421 initialize(*result.iterator);
2422 // [use result.iterator...]
2423 \endcode
2424
2425 \sa emplace(), tryInsert(), insertOrAssign()
2426*/
2427
2428/*!
2429 \fn template <class Key, class T> QHash<Key, T>::TryEmplaceResult QHash<Key, T>::tryInsert(const Key &key, const T &value)
2430 \fn template <class Key, class T> template <typename K, QHash<Key, T>::if_heterogeneously_searchable<K> = true, QHash<Key, T>::if_key_constructible_from<K> = true> QHash<Key, T>::TryEmplaceResult QHash<Key, T>::tryInsert(K &&key, const T &value)
2431 \since 6.9
2432
2433 Inserts a new item with the \a key and a value of \a value.
2434 If an item with \a key already exists, no insertion takes place.
2435
2436 Returns an instance of \l{TryEmplaceResult}, a structure that holds an
2437 \l{QHash::TryEmplaceResult::}{iterator} to the newly created item, or to the pre-existing item
2438 that prevented the insertion, and a boolean, \l{QHash::TryEmplaceResult::}{inserted}, denoting
2439 whether the insertion took place.
2440
2441 \sa insert(), tryEmplace(), insertOrAssign()
2442*/
2443
2444/*!
2445 \fn template <class Key, class T> template <typename K, typename... Args, QHash<Key, T>::if_heterogeneously_searchable<K> = true, QHash<Key, T>::if_key_constructible_from<K> = true> iterator QHash<Key, T>::try_emplace(const_iterator hint, K &&key, Args &&...args)
2446 \fn template <class Key, class T> template <typename... Args> iterator QHash<Key, T>::try_emplace(const_iterator hint, const Key &key, Args &&...args)
2447 \fn template <class Key, class T> template <typename... Args> iterator QHash<Key, T>::try_emplace(const_iterator hint, Key &&key, Args &&...args)
2448 \since 6.9
2449
2450 Inserts a new item with the \a key and a value constructed from \a args.
2451 If an item with \a key already exists, no insertion takes place.
2452
2453 Returns the iterator of the inserted item, or to the item that prevented the
2454 insertion.
2455
2456 \a hint is ignored.
2457
2458 These functions are provided for compatibility with the standard library.
2459
2460 \sa emplace(), tryEmplace(), tryInsert(), insertOrAssign()
2461*/
2462
2463/*!
2464 \fn template <class Key, class T> template <typename... Args> std::pair<iterator, bool> QHash<Key, T>::try_emplace(const Key &key, Args &&...args)
2465 \fn template <class Key, class T> template <typename... Args> std::pair<iterator, bool> QHash<Key, T>::try_emplace(Key &&key, Args &&...args)
2466 \fn template <class Key, class T> template <typename K, typename... Args, QHash<Key, T>::if_heterogeneously_searchable<K> = true, QHash<Key, T>::if_key_constructible_from<K> = true> std::pair<iterator, bool> QHash<Key, T>::try_emplace(K &&key, Args &&...args)
2467 \since 6.9
2468
2469 Inserts a new item with the \a key and a value constructed from \a args.
2470 If an item with \a key already exists, no insertion takes place.
2471
2472 Returns a pair consisting of an iterator to the inserted item (or to the
2473 item that prevented the insertion), and a bool denoting whether the
2474 insertion took place.
2475
2476 These functions are provided for compatibility with the standard library.
2477
2478 \sa emplace(), tryEmplace(), tryInsert(), insertOrAssign()
2479*/
2480
2481/*!
2482 \fn template <class Key, class T> template <typename Value> QHash<Key, T>::TryEmplaceResult QHash<Key, T>::insertOrAssign(const Key &key, Value &&value)
2483 \fn template <class Key, class T> template <typename Value> QHash<Key, T>::TryEmplaceResult QHash<Key, T>::insertOrAssign(Key &&key, Value &&value)
2484 \fn template <class Key, class T> template <typename K, typename Value, QHash<Key, T>::if_heterogeneously_searchable<K> = true, QHash<Key, T>::if_key_constructible_from<K> = true> QHash<Key, T>::TryEmplaceResult QHash<Key, T>::insertOrAssign(K &&key, Value &&value)
2485 \since 6.9
2486
2487 Attempts to insert an item with the \a key and \a value.
2488 If an item with \a key already exists its value is overwritten with \a value.
2489
2490 Returns an instance of \l{TryEmplaceResult}, a structure that holds an
2491 \l{QHash::TryEmplaceResult::}{iterator} to the item, and a boolean,
2492 \l{QHash::TryEmplaceResult::}{inserted}, denoting whether the item was newly created (\c{true})
2493 or if it previously existed (\c{false}).
2494
2495 \sa insert(), tryEmplace(), tryInsert()
2496*/
2497
2498/*!
2499 \fn template <class Key, class T> template <typename Value> std::pair<QHash<Key, T>::key_value_iterator, bool> QHash<Key, T>::insert_or_assign(const Key &key, Value &&value)
2500 \fn template <class Key, class T> template <typename Value> std::pair<QHash<Key, T>::key_value_iterator, bool> QHash<Key, T>::insert_or_assign(Key &&key, Value &&value)
2501 \fn template <class Key, class T> template <typename K, typename Value, QHash<Key, T>::if_heterogeneously_searchable<K> = true, QHash<Key, T>::if_key_constructible_from<K> = true> std::pair<QHash<Key, T>::key_value_iterator, bool> QHash<Key, T>::insert_or_assign(K &&key, Value &&value)
2502 \since 6.9
2503
2504 Attempts to insert an item with the \a key and \a value.
2505 If an item with \a key already exists its value is overwritten with \a value.
2506
2507 Returns a pair consisting of an iterator pointing to the item, and a
2508 boolean, denoting whether the item was newly created (\c{true}) or if it
2509 previously existed (\c{false}).
2510
2511 These functions are provided for compatibility with the standard library.
2512
2513 \sa insert(), tryEmplace(), tryInsert(), insertOrAssign()
2514*/
2515
2516/*!
2517 \fn template <class Key, class T> template <typename Value> std::pair<QHash<Key, T>::key_value_iterator, bool> QHash<Key, T>::insert_or_assign(const_iterator hint, const Key &key, Value &&value)
2518 \fn template <class Key, class T> template <typename Value> std::pair<QHash<Key, T>::key_value_iterator, bool> QHash<Key, T>::insert_or_assign(const_iterator hint, Key &&key, Value &&value)
2519 \fn template <class Key, class T> template <typename K, typename Value, QHash<Key, T>::if_heterogeneously_searchable<K> = true, QHash<Key, T>::if_key_constructible_from<K> = true> std::pair<QHash<Key, T>::key_value_iterator, bool> QHash<Key, T>::insert_or_assign(const_iterator hint, K &&key, Value &&value)
2520 \since 6.9
2521
2522 Attempts to insert an item with the \a key and \a value.
2523 If an item with \a key already exists its value is overwritten with \a value.
2524
2525 Returns a pair consisting of an iterator pointing to the item, and a
2526 boolean, denoting whether the item was newly created (\c{true}) or if it
2527 previously existed (\c{false}).
2528
2529 \a hint is ignored.
2530
2531 These functions are provided for compatibility with the standard library.
2532
2533 \sa insert(), tryEmplace(), insertOrAssign()
2534*/
2535
2536/*! \fn template <class Key, class T> void QHash<Key, T>::insert(const QHash &other)
2537 \since 5.15
2538
2539 Inserts all the items in the \a other hash into this hash.
2540
2541 If a key is common to both hashes, its value will be replaced with the
2542 value stored in \a other.
2543*/
2544
2545/*! \fn template <class Key, class T> bool QHash<Key, T>::empty() const
2546
2547 This function is provided for STL compatibility. It is equivalent
2548 to isEmpty(), returning true if the hash is empty; otherwise
2549 returns \c false.
2550*/
2551
2552/*! \fn template <class Key, class T> std::pair<iterator, iterator> QMultiHash<Key, T>::equal_range(const Key &key)
2553 \since 5.7
2554
2555 Returns a pair of iterators delimiting the range of values \c{[first, second)}, that
2556 are stored under \a key. If the range is empty then both iterators will be equal to end().
2557
2558 \include qhash.cpp qhash-iterator-invalidation-func-desc
2559*/
2560
2561/*!
2562 \fn template <class Key, class T> std::pair<const_iterator, const_iterator> QMultiHash<Key, T>::equal_range(const Key &key) const
2563 \overload
2564 \since 5.7
2565
2566 \include qhash.cpp qhash-iterator-invalidation-func-desc
2567*/
2568
2569/*! \typedef QHash::ConstIterator
2570
2571 Qt-style synonym for QHash::const_iterator.
2572*/
2573
2574/*! \typedef QHash::Iterator
2575
2576 Qt-style synonym for QHash::iterator.
2577*/
2578
2579/*! \typedef QHash::difference_type
2580
2581 Typedef for ptrdiff_t. Provided for STL compatibility.
2582*/
2583
2584/*! \typedef QHash::key_type
2585
2586 Typedef for Key. Provided for STL compatibility.
2587*/
2588
2589/*! \typedef QHash::mapped_type
2590
2591 Typedef for T. Provided for STL compatibility.
2592*/
2593
2594/*! \typedef QHash::size_type
2595
2596 Typedef for int. Provided for STL compatibility.
2597*/
2598
2599/*! \typedef QHash::iterator::difference_type
2600 \internal
2601*/
2602
2603/*! \typedef QHash::iterator::iterator_category
2604 \internal
2605*/
2606
2607/*! \typedef QHash::iterator::pointer
2608 \internal
2609*/
2610
2611/*! \typedef QHash::iterator::reference
2612 \internal
2613*/
2614
2615/*! \typedef QHash::iterator::value_type
2616 \internal
2617*/
2618
2619/*! \typedef QHash::const_iterator::difference_type
2620 \internal
2621*/
2622
2623/*! \typedef QHash::const_iterator::iterator_category
2624 \internal
2625*/
2626
2627/*! \typedef QHash::const_iterator::pointer
2628 \internal
2629*/
2630
2631/*! \typedef QHash::const_iterator::reference
2632 \internal
2633*/
2634
2635/*! \typedef QHash::const_iterator::value_type
2636 \internal
2637*/
2638
2639/*! \typedef QHash::key_iterator::difference_type
2640 \internal
2641*/
2642
2643/*! \typedef QHash::key_iterator::iterator_category
2644 \internal
2645*/
2646
2647/*! \typedef QHash::key_iterator::pointer
2648 \internal
2649*/
2650
2651/*! \typedef QHash::key_iterator::reference
2652 \internal
2653*/
2654
2655/*! \typedef QHash::key_iterator::value_type
2656 \internal
2657*/
2658
2659/*! \class QHash::iterator
2660 \inmodule QtCore
2661 \brief The QHash::iterator class provides an STL-style non-const iterator for QHash.
2662
2663 QHash<Key, T>::iterator allows you to iterate over a QHash
2664 and to modify the value (but not the key) associated
2665 with a particular key. If you want to iterate over a const QHash,
2666 you should use QHash::const_iterator. It is generally good
2667 practice to use QHash::const_iterator on a non-const QHash as
2668 well, unless you need to change the QHash through the iterator.
2669 Const iterators are slightly faster, and can improve code
2670 readability.
2671
2672 The default QHash::iterator constructor creates an uninitialized
2673 iterator. You must initialize it using a QHash function like
2674 QHash::begin(), QHash::end(), or QHash::find() before you can
2675 start iterating. Here's a typical loop that prints all the (key,
2676 value) pairs stored in a hash:
2677
2678 \snippet code/src_corelib_tools_qhash.cpp 17
2679
2680 Unlike QMap, which orders its items by key, QHash stores its
2681 items in an arbitrary order.
2682
2683 Here's an example that increments every value stored in the QHash
2684 by 2:
2685
2686 \snippet code/src_corelib_tools_qhash.cpp 18
2687
2688 To remove elements from a QHash you can use erase_if(QHash<Key, T> &map, Predicate pred):
2689
2690 \snippet code/src_corelib_tools_qhash.cpp 21
2691
2692 Multiple iterators can be used on the same hash. However, be aware
2693 that any modification performed directly on the QHash (inserting and
2694 removing items) can cause the iterators to become invalid.
2695
2696 Inserting items into the hash or calling methods such as QHash::reserve()
2697 or QHash::squeeze() can invalidate all iterators pointing into the hash.
2698 Iterators are guaranteed to stay valid only as long as the QHash doesn't have
2699 to grow/shrink its internal hash table.
2700 Using any iterator after a rehashing operation has occurred will lead to undefined behavior.
2701
2702 If you need to keep iterators over a long period of time, we recommend
2703 that you use QMap rather than QHash.
2704
2705 \warning Iterators on implicitly shared containers do not work
2706 exactly like STL-iterators. You should avoid copying a container
2707 while iterators are active on that container. For more information,
2708 read \l{Implicit sharing iterator problem}.
2709
2710 \sa QHash::const_iterator, QHash::key_iterator, QHash::key_value_iterator
2711*/
2712
2713/*! \fn template <class Key, class T> QHash<Key, T>::iterator::iterator()
2714
2715 Constructs an uninitialized iterator.
2716
2717 Functions like key(), value(), and operator++() must not be
2718 called on an uninitialized iterator. Use operator=() to assign a
2719 value to it before using it.
2720
2721 \sa QHash::begin(), QHash::end()
2722*/
2723
2724/*! \fn template <class Key, class T> const Key &QHash<Key, T>::iterator::key() const
2725
2726 Returns the current item's key as a const reference.
2727
2728 There is no direct way of changing an item's key through an
2729 iterator, although it can be done by calling QHash::erase()
2730 followed by QHash::insert().
2731
2732 \sa value()
2733*/
2734
2735/*! \fn template <class Key, class T> T &QHash<Key, T>::iterator::value() const
2736
2737 Returns a modifiable reference to the current item's value.
2738
2739 You can change the value of an item by using value() on
2740 the left side of an assignment, for example:
2741
2742 \snippet code/src_corelib_tools_qhash.cpp 22
2743
2744 \sa key(), operator*()
2745*/
2746
2747/*! \fn template <class Key, class T> T &QHash<Key, T>::iterator::operator*() const
2748
2749 Returns a modifiable reference to the current item's value.
2750
2751 Same as value().
2752
2753 \sa key()
2754*/
2755
2756/*! \fn template <class Key, class T> T *QHash<Key, T>::iterator::operator->() const
2757
2758 Returns a pointer to the current item's value.
2759
2760 \sa value()
2761*/
2762
2763/*!
2764 \fn template <class Key, class T> bool QHash<Key, T>::iterator::operator==(const iterator &other) const
2765 \fn template <class Key, class T> bool QHash<Key, T>::iterator::operator==(const const_iterator &other) const
2766
2767 Returns \c true if \a other points to the same item as this
2768 iterator; otherwise returns \c false.
2769
2770 \sa operator!=()
2771*/
2772
2773/*!
2774 \fn template <class Key, class T> bool QHash<Key, T>::iterator::operator!=(const iterator &other) const
2775 \fn template <class Key, class T> bool QHash<Key, T>::iterator::operator!=(const const_iterator &other) const
2776
2777 Returns \c true if \a other points to a different item than this
2778 iterator; otherwise returns \c false.
2779
2780 \sa operator==()
2781*/
2782
2783/*!
2784 \fn template <class Key, class T> QHash<Key, T>::iterator &QHash<Key, T>::iterator::operator++()
2785
2786 The prefix ++ operator (\c{++i}) advances the iterator to the
2787 next item in the hash and returns an iterator to the new current
2788 item.
2789
2790 Calling this function on QHash::end() leads to undefined results.
2791*/
2792
2793/*! \fn template <class Key, class T> QHash<Key, T>::iterator QHash<Key, T>::iterator::operator++(int)
2794
2795 \overload
2796
2797 The postfix ++ operator (\c{i++}) advances the iterator to the
2798 next item in the hash and returns an iterator to the previously
2799 current item.
2800*/
2801
2802/*! \class QHash::const_iterator
2803 \inmodule QtCore
2804 \brief The QHash::const_iterator class provides an STL-style const iterator for QHash.
2805
2806 QHash<Key, T>::const_iterator allows you to iterate over a
2807 QHash. If you want to modify the QHash as you
2808 iterate over it, you must use QHash::iterator instead. It is
2809 generally good practice to use QHash::const_iterator on a
2810 non-const QHash as well, unless you need to change the QHash
2811 through the iterator. Const iterators are slightly faster, and
2812 can improve code readability.
2813
2814 The default QHash::const_iterator constructor creates an
2815 uninitialized iterator. You must initialize it using a QHash
2816 function like QHash::cbegin(), QHash::cend(), or
2817 QHash::constFind() before you can start iterating. Here's a typical
2818 loop that prints all the (key, value) pairs stored in a hash:
2819
2820 \snippet code/src_corelib_tools_qhash.cpp 23
2821
2822 Unlike QMap, which orders its items by key, QHash stores its
2823 items in an arbitrary order. The only guarantee is that items that
2824 share the same key (because they were inserted using
2825 a QMultiHash) will appear consecutively, from the most
2826 recently to the least recently inserted value.
2827
2828 Multiple iterators can be used on the same hash. However, be aware
2829 that any modification performed directly on the QHash (inserting and
2830 removing items) can cause the iterators to become invalid.
2831
2832 Inserting items into the hash or calling methods such as QHash::reserve()
2833 or QHash::squeeze() can invalidate all iterators pointing into the hash.
2834 Iterators are guaranteed to stay valid only as long as the QHash doesn't have
2835 to grow/shrink its internal hash table.
2836 Using any iterator after a rehashing operation has occurred will lead to undefined behavior.
2837
2838 You can however safely use iterators to remove entries from the hash
2839 using the QHash::erase() method. This function can safely be called while
2840 iterating, and won't affect the order of items in the hash.
2841
2842 \warning Iterators on implicitly shared containers do not work
2843 exactly like STL-iterators. You should avoid copying a container
2844 while iterators are active on that container. For more information,
2845 read \l{Implicit sharing iterator problem}.
2846
2847 \sa QHash::iterator, QHash::key_iterator, QHash::const_key_value_iterator
2848*/
2849
2850/*! \fn template <class Key, class T> QHash<Key, T>::const_iterator::const_iterator()
2851
2852 Constructs an uninitialized iterator.
2853
2854 Functions like key(), value(), and operator++() must not be
2855 called on an uninitialized iterator. Use operator=() to assign a
2856 value to it before using it.
2857
2858 \sa QHash::constBegin(), QHash::constEnd()
2859*/
2860
2861/*! \fn template <class Key, class T> QHash<Key, T>::const_iterator::const_iterator(const iterator &other)
2862
2863 Constructs a copy of \a other.
2864*/
2865
2866/*! \fn template <class Key, class T> const Key &QHash<Key, T>::const_iterator::key() const
2867
2868 Returns the current item's key.
2869
2870 \sa value()
2871*/
2872
2873/*! \fn template <class Key, class T> const T &QHash<Key, T>::const_iterator::value() const
2874
2875 Returns the current item's value.
2876
2877 \sa key(), operator*()
2878*/
2879
2880/*! \fn template <class Key, class T> const T &QHash<Key, T>::const_iterator::operator*() const
2881
2882 Returns the current item's value.
2883
2884 Same as value().
2885
2886 \sa key()
2887*/
2888
2889/*! \fn template <class Key, class T> const T *QHash<Key, T>::const_iterator::operator->() const
2890
2891 Returns a pointer to the current item's value.
2892
2893 \sa value()
2894*/
2895
2896/*! \fn template <class Key, class T> bool QHash<Key, T>::const_iterator::operator==(const const_iterator &other) const
2897
2898 Returns \c true if \a other points to the same item as this
2899 iterator; otherwise returns \c false.
2900
2901 \sa operator!=()
2902*/
2903
2904/*! \fn template <class Key, class T> bool QHash<Key, T>::const_iterator::operator!=(const const_iterator &other) const
2905
2906 Returns \c true if \a other points to a different item than this
2907 iterator; otherwise returns \c false.
2908
2909 \sa operator==()
2910*/
2911
2912/*!
2913 \fn template <class Key, class T> QHash<Key, T>::const_iterator &QHash<Key, T>::const_iterator::operator++()
2914
2915 The prefix ++ operator (\c{++i}) advances the iterator to the
2916 next item in the hash and returns an iterator to the new current
2917 item.
2918
2919 Calling this function on QHash::end() leads to undefined results.
2920*/
2921
2922/*! \fn template <class Key, class T> QHash<Key, T>::const_iterator QHash<Key, T>::const_iterator::operator++(int)
2923
2924 \overload
2925
2926 The postfix ++ operator (\c{i++}) advances the iterator to the
2927 next item in the hash and returns an iterator to the previously
2928 current item.
2929*/
2930
2931/*! \class QHash::key_iterator
2932 \inmodule QtCore
2933 \since 5.6
2934 \brief The QHash::key_iterator class provides an STL-style const iterator for QHash keys.
2935
2936 QHash::key_iterator is essentially the same as QHash::const_iterator
2937 with the difference that operator*() and operator->() return a key
2938 instead of a value.
2939
2940 For most uses QHash::iterator and QHash::const_iterator should be used,
2941 you can easily access the key by calling QHash::iterator::key():
2942
2943 \snippet code/src_corelib_tools_qhash.cpp 27
2944
2945 However, to have interoperability between QHash's keys and STL-style
2946 algorithms we need an iterator that dereferences to a key instead
2947 of a value. With QHash::key_iterator we can apply an algorithm to a
2948 range of keys without having to call QHash::keys(), which is inefficient
2949 as it costs one QHash iteration and memory allocation to create a temporary
2950 QList.
2951
2952 \snippet code/src_corelib_tools_qhash.cpp 28
2953
2954 QHash::key_iterator is const, it's not possible to modify the key.
2955
2956 The default QHash::key_iterator constructor creates an uninitialized
2957 iterator. You must initialize it using a QHash function like
2958 QHash::keyBegin() or QHash::keyEnd().
2959
2960 \warning Iterators on implicitly shared containers do not work
2961 exactly like STL-iterators. You should avoid copying a container
2962 while iterators are active on that container. For more information,
2963 read \l{Implicit sharing iterator problem}.
2964
2965 \sa QHash::const_iterator, QHash::iterator
2966*/
2967
2968/*! \fn template <class Key, class T> const T &QHash<Key, T>::key_iterator::operator*() const
2969
2970 Returns the current item's key.
2971*/
2972
2973/*! \fn template <class Key, class T> const T *QHash<Key, T>::key_iterator::operator->() const
2974
2975 Returns a pointer to the current item's key.
2976*/
2977
2978/*! \fn template <class Key, class T> bool QHash<Key, T>::key_iterator::operator==(key_iterator other) const
2979
2980 Returns \c true if \a other points to the same item as this
2981 iterator; otherwise returns \c false.
2982
2983 \sa operator!=()
2984*/
2985
2986/*! \fn template <class Key, class T> bool QHash<Key, T>::key_iterator::operator!=(key_iterator other) const
2987
2988 Returns \c true if \a other points to a different item than this
2989 iterator; otherwise returns \c false.
2990
2991 \sa operator==()
2992*/
2993
2994/*!
2995 \fn template <class Key, class T> QHash<Key, T>::key_iterator &QHash<Key, T>::key_iterator::operator++()
2996
2997 The prefix ++ operator (\c{++i}) advances the iterator to the
2998 next item in the hash and returns an iterator to the new current
2999 item.
3000
3001 Calling this function on QHash::keyEnd() leads to undefined results.
3002
3003*/
3004
3005/*! \fn template <class Key, class T> QHash<Key, T>::key_iterator QHash<Key, T>::key_iterator::operator++(int)
3006
3007 \overload
3008
3009 The postfix ++ operator (\c{i++}) advances the iterator to the
3010 next item in the hash and returns an iterator to the previous
3011 item.
3012*/
3013
3014/*! \fn template <class Key, class T> const_iterator QHash<Key, T>::key_iterator::base() const
3015 Returns the underlying const_iterator this key_iterator is based on.
3016*/
3017
3018/*! \typedef QHash::const_key_value_iterator
3019 \inmodule QtCore
3020 \since 5.10
3021 \brief The QHash::const_key_value_iterator typedef provides an STL-style const iterator for QHash.
3022
3023 QHash::const_key_value_iterator is essentially the same as QHash::const_iterator
3024 with the difference that operator*() returns a key/value pair instead of a
3025 value.
3026
3027 \sa QKeyValueIterator
3028*/
3029
3030/*! \typedef QHash::key_value_iterator
3031 \inmodule QtCore
3032 \since 5.10
3033 \brief The QHash::key_value_iterator typedef provides an STL-style iterator for QHash.
3034
3035 QHash::key_value_iterator is essentially the same as QHash::iterator
3036 with the difference that operator*() returns a key/value pair instead of a
3037 value.
3038
3039 \sa QKeyValueIterator
3040*/
3041
3042/*! \fn template <class Key, class T> QDataStream &operator<<(QDataStream &out, const QHash<Key, T>& hash)
3043 \relates QHash
3044
3045 Writes the hash \a hash to stream \a out.
3046
3047 This function requires the key and value types to implement \c
3048 operator<<().
3049
3050 \sa {Serializing Qt Data Types}
3051*/
3052
3053/*! \fn template <class Key, class T> QDataStream &operator>>(QDataStream &in, QHash<Key, T> &hash)
3054 \relates QHash
3055
3056 Reads a hash from stream \a in into \a hash.
3057
3058 This function requires the key and value types to implement \c
3059 operator>>().
3060
3061 \sa {Serializing Qt Data Types}
3062*/
3063
3064/*! \class QMultiHash
3065 \inmodule QtCore
3066 \brief The QMultiHash class provides a multi-valued hash table.
3067 \compares equality
3068
3069 \ingroup tools
3070 \ingroup shared
3071
3072 \reentrant
3073
3074 QMultiHash<Key, T> is one of Qt's generic \l{container classes}, where
3075 \a Key is the type used for lookup keys and \a T is the mapped value type.
3076 It provides a hash table that allows multiple values for the same key.
3077
3078 QMultiHash mostly mirrors QHash's API. For example, you can use isEmpty() to test
3079 whether the hash is empty, and you can traverse a QMultiHash using
3080 QHash's iterator classes (for example, QHashIterator). But opposed to
3081 QHash, it provides an insert() function that allows the insertion of
3082 multiple items with the same key. The replace() function corresponds to
3083 QHash::insert(). It also provides convenient operator+() and
3084 operator+=().
3085
3086 Unlike QMultiMap, QMultiHash does not provide ordering of the
3087 inserted items. The only guarantee is that items that
3088 share the same key will appear consecutively, from the most
3089 recently to the least recently inserted value.
3090
3091 Example:
3092 \snippet code/src_corelib_tools_qhash.cpp 24
3093
3094 Unlike QHash, QMultiHash provides no operator[]. Use value() or
3095 replace() if you want to access the most recently inserted item
3096 with a certain key.
3097
3098 If you want to retrieve all the values for a single key, you can
3099 use values(const Key &key), which returns a QList<T>:
3100
3101 \snippet code/src_corelib_tools_qhash.cpp 25
3102
3103 The items that share the same key are available from most
3104 recently to least recently inserted.
3105
3106 A more efficient approach is to call find() to get
3107 the STL-style iterator for the first item with a key and iterate from
3108 there:
3109
3110 \snippet code/src_corelib_tools_qhash.cpp 26
3111
3112 QMultiHash's key and value data types must be \l{assignable data
3113 types}. You cannot, for example, store a QWidget as a value;
3114 instead, store a QWidget *. In addition, QMultiHash's key type
3115 must provide operator==(), and there must also be a qHash() function
3116 in the type's namespace that returns a hash value for an argument of the
3117 key's type. See the QHash documentation for details.
3118
3119 \sa QHash, QHashIterator, QMutableHashIterator, QMultiMap
3120*/
3121
3122/*! \fn template <class Key, class T> QMultiHash<Key, T>::QMultiHash()
3123
3124 Constructs an empty hash.
3125*/
3126
3127/*! \fn template <class Key, class T> QMultiHash<Key, T>::QMultiHash(std::initializer_list<std::pair<Key,T> > list)
3128 \since 5.1
3129
3130 Constructs a multi-hash with a copy of each of the elements in the
3131 initializer list \a list.
3132*/
3133
3134/*! \fn template <class Key, class T> QMultiHash<Key, T>::QMultiHash(const QHash<Key, T> &other)
3135
3136 Constructs a copy of \a other (which can be a QHash or a
3137 QMultiHash).
3138*/
3139
3140/*! \fn template <class Key, class T> template <class InputIterator> QMultiHash<Key, T>::QMultiHash(InputIterator begin, InputIterator end)
3141 \since 5.14
3142
3143 Constructs a multi-hash with a copy of each of the elements in the iterator range
3144 [\a begin, \a end). Either the elements iterated by the range must be
3145 objects with \c{first} and \c{second} data members (like \c{std::pair}),
3146 convertible to \c Key and to \c T respectively; or the
3147 iterators must have \c{key()} and \c{value()} member functions, returning a
3148 key convertible to \c Key and a value convertible to \c T respectively.
3149*/
3150
3151/*! \fn template <class Key, class T> QMultiHash<Key, T>::iterator QMultiHash<Key, T>::replace(const Key &key, const T &value)
3152
3153 Inserts a new item with the \a key and a value of \a value.
3154
3155 If there is already an item with the \a key, that item's value
3156 is replaced with \a value.
3157
3158 If there are multiple items with the \a key, the most
3159 recently inserted item's value is replaced with \a value.
3160
3161 Returns an iterator pointing to the new/updated element.
3162
3163 \include qhash.cpp qhash-iterator-invalidation-func-desc
3164
3165 \sa insert()
3166*/
3167
3168/*! \fn template <class Key, class T> QMultiHash<Key, T>::iterator QMultiHash<Key, T>::insert(const Key &key, const T &value)
3169
3170 Inserts a new item with the \a key and a value of \a value.
3171
3172 If there is already an item with the same key in the hash, this
3173 function will simply create a new one. (This behavior is
3174 different from replace(), which overwrites the value of an
3175 existing item.)
3176
3177 Returns an iterator pointing to the new element.
3178
3179 \include qhash.cpp qhash-iterator-invalidation-func-desc
3180
3181 \sa replace()
3182*/
3183
3184/*!
3185 \fn template <class Key, class T> QMultiHash<Key, T>::iterator QMultiHash<Key, T>::insert(const Key &key, T &&value)
3186 \fn template <class Key, class T> QMultiHash<Key, T>::iterator QMultiHash<Key, T>::insert(Key &&key, const T &value)
3187 \fn template <class Key, class T> QMultiHash<Key, T>::iterator QMultiHash<Key, T>::insert(Key &&key, T &&value)
3188 \since 6.11
3189 \overload
3190*/
3191
3192/*!
3193 \fn template <class Key, class T> template <typename ...Args> QMultiHash<Key, T>::iterator QMultiHash<Key, T>::emplace(const Key &key, Args&&... args)
3194 \fn template <class Key, class T> template <typename ...Args> QMultiHash<Key, T>::iterator QMultiHash<Key, T>::emplace(Key &&key, Args&&... args)
3195
3196 Inserts a new element into the container. This new element
3197 is constructed in-place using \a args as the arguments for its
3198 construction.
3199
3200 If there is already an item with the same key in the hash, this
3201 function will simply create a new one. (This behavior is
3202 different from replace(), which overwrites the value of an
3203 existing item.)
3204
3205 Returns an iterator pointing to the new element.
3206
3207 \include qhash.cpp qhash-iterator-invalidation-func-desc
3208
3209 \sa insert
3210*/
3211
3212/*!
3213 \fn template <class Key, class T> template <typename ...Args> QMultiHash<Key, T>::iterator QMultiHash<Key, T>::emplaceReplace(const Key &key, Args&&... args)
3214 \fn template <class Key, class T> template <typename ...Args> QMultiHash<Key, T>::iterator QMultiHash<Key, T>::emplaceReplace(Key &&key, Args&&... args)
3215
3216 Inserts a new element into the container. This new element
3217 is constructed in-place using \a args as the arguments for its
3218 construction.
3219
3220 If there is already an item with the same key in the hash, that item's
3221 value is replaced with a value constructed from \a args.
3222
3223 Returns an iterator pointing to the new element.
3224
3225 \include qhash.cpp qhash-iterator-invalidation-func-desc
3226
3227 \sa replace, emplace
3228*/
3229
3230/*! \fn template <class Key, class T> QMultiHash<Key, T>::iterator QMultiHash<Key, T>::erase(const_iterator pos)
3231 \since 5.7
3232
3233 Removes the (key, value) pair associated with the iterator \a pos
3234 from the hash, and returns an iterator to the next item in the
3235 hash.
3236
3237 This function never causes QMultiHash to
3238 rehash its internal data structure. This means that it can safely
3239 be called while iterating, and won't affect the order of items in
3240 the hash. For example:
3241
3242 \snippet code/src_corelib_tools_qhash.cpp 15multihash
3243
3244 \include qhash.cpp qhash-iterator-invalidation-func-desc
3245
3246 \sa remove(), take(), find()
3247*/
3248
3249/*! \fn template <class Key, class T> QMultiHash &QMultiHash<Key, T>::unite(const QMultiHash &other)
3250 \since 5.13
3251
3252 Inserts all the items in the \a other hash into this hash
3253 and returns a reference to this hash.
3254
3255 \sa insert()
3256*/
3257
3258
3259/*! \fn template <class Key, class T> QMultiHash &QMultiHash<Key, T>::unite(const QHash<Key, T> &other)
3260 \since 6.0
3261
3262 Inserts all the items in the \a other hash into this hash
3263 and returns a reference to this hash.
3264
3265 \sa insert()
3266*/
3267
3268/*! \fn template <class Key, class T> QList<Key> QMultiHash<Key, T>::uniqueKeys() const
3269 \since 5.13
3270
3271 Returns a list containing all the keys in the map. Keys that occur multiple
3272 times in the map occur only once in the returned list.
3273
3274 \sa keys(), values()
3275*/
3276
3277/*! \fn template <class Key, class T> T QMultiHash<Key, T>::value(const Key &key) const
3278 \fn template <class Key, class T> T QMultiHash<Key, T>::value(const Key &key, const T &defaultValue) const
3279
3280 Returns the value associated with the \a key.
3281
3282 If the hash contains no item with the \a key, the function
3283 returns \a defaultValue, or a \l{default-constructed value} if this
3284 parameter has not been supplied.
3285
3286 If there are multiple
3287 items for the \a key in the hash, the value of the most recently
3288 inserted one is returned.
3289*/
3290
3291/*! \fn template <class Key, class T> QList<T> QMultiHash<Key, T>::values(const Key &key) const
3292 \overload
3293
3294 Returns a list of all the values associated with the \a key,
3295 from the most recently inserted to the least recently inserted.
3296
3297 \sa count(), insert()
3298*/
3299
3300/*! \fn template <class Key, class T> T &QMultiHash<Key, T>::operator[](const Key &key)
3301
3302 Returns the value associated with the \a key as a modifiable reference.
3303
3304 If the hash contains no item with the \a key, the function inserts
3305 a \l{default-constructed value} into the hash with the \a key, and
3306 returns a reference to it.
3307
3308 If the hash contains multiple items with the \a key, this function returns
3309 a reference to the most recently inserted value.
3310
3311 \include qhash.cpp qhash-iterator-invalidation-func-desc
3312
3313 \sa insert(), value()
3314*/
3315
3316/*!
3317 \fn template <class Key, class T> bool QMultiHash<Key, T>::operator==(const QMultiHash &lhs, const QMultiHash &rhs)
3318
3319 Returns \c true if \a lhs multihash equals to the \a rhs multihash;
3320 otherwise returns \c false.
3321
3322 Two multihashes are considered equal if they contain the same (key, value)
3323 pairs.
3324
3325 This function requires the value type to implement \c {operator==()}.
3326
3327 \sa operator!=()
3328*/
3329
3330/*!
3331 \fn template <class Key, class T> bool QMultiHash<Key, T>::operator!=(const QMultiHash &lhs, const QMultiHash &rhs)
3332
3333 Returns \c true if \a lhs multihash is not equal to the \a rhs multihash;
3334 otherwise returns \c false.
3335
3336 Two multihashes are considered equal if they contain the same (key, value)
3337 pairs.
3338
3339 This function requires the value type to implement \c {operator==()}.
3340
3341 \sa operator==()
3342*/
3343
3344/*! \fn template <class Key, class T> QMultiHash &QMultiHash<Key, T>::operator+=(const QMultiHash &other)
3345
3346 Inserts all the items in the \a other hash into this hash
3347 and returns a reference to this hash.
3348
3349 \sa unite(), insert()
3350*/
3351
3352/*! \fn template <class Key, class T> QMultiHash QMultiHash<Key, T>::operator+(const QMultiHash &other) const
3353
3354 Returns a hash that contains all the items in this hash in
3355 addition to all the items in \a other. If a key is common to both
3356 hashes, the resulting hash will contain the key multiple times.
3357
3358 \sa operator+=()
3359*/
3360
3361/*!
3362 \fn template <class Key, class T> bool QMultiHash<Key, T>::contains(const Key &key, const T &value) const
3363 \since 4.3
3364
3365 Returns \c true if the hash contains an item with the \a key and
3366 \a value; otherwise returns \c false.
3367
3368 \sa count()
3369*/
3370
3371/*!
3372 \fn template <class Key, class T> qsizetype QMultiHash<Key, T>::remove(const Key &key)
3373 \since 4.3
3374
3375 Removes all the items that have the \a key from the hash.
3376 Returns the number of items removed.
3377
3378 \sa remove(const Key &key, const T &value)
3379*/
3380
3381/*!
3382 \fn template <class Key, class T> qsizetype QMultiHash<Key, T>::remove(const Key &key, const T &value)
3383 \since 4.3
3384
3385 Removes all the items that have the \a key and the value \a
3386 value from the hash. Returns the number of items removed.
3387
3388 \sa remove()
3389*/
3390
3391/*!
3392 \fn template <class Key, class T> void QMultiHash<Key, T>::clear()
3393 \since 4.3
3394
3395 Removes all items from the hash and frees up all memory used by it.
3396
3397 \sa remove()
3398*/
3399
3400/*! \fn template <class Key, class T> template <typename Predicate> qsizetype QMultiHash<Key, T>::removeIf(Predicate pred)
3401 \since 6.1
3402
3403 Removes all elements for which the predicate \a pred returns true
3404 from the multi hash.
3405
3406 The function supports predicates which take either an argument of
3407 type \c{QMultiHash<Key, T>::iterator}, or an argument of type
3408 \c{std::pair<const Key &, T &>}.
3409
3410 Returns the number of elements removed, if any.
3411
3412 \sa clear(), take()
3413*/
3414
3415/*! \fn template <class Key, class T> T QMultiHash<Key, T>::take(const Key &key)
3416
3417 Removes the item with the \a key from the hash and returns
3418 the value associated with it.
3419
3420 If the item does not exist in the hash, the function simply
3421 returns a \l{default-constructed value}. If there are multiple
3422 items for \a key in the hash, only the most recently inserted one
3423 is removed.
3424
3425 If you don't use the return value, remove() is more efficient.
3426
3427 \sa remove()
3428*/
3429
3430/*! \fn template <class Key, class T> QList<Key> QMultiHash<Key, T>::keys() const
3431
3432 Returns a list containing all the keys in the hash, in an
3433 arbitrary order. Keys that occur multiple times in the hash
3434 also occur multiple times in the list.
3435
3436 The order is guaranteed to be the same as that used by values().
3437
3438 This function creates a new list, in \l {linear time}. The time and memory
3439 use that entails can be avoided by iterating from \l keyBegin() to
3440 \l keyEnd().
3441
3442 \sa values(), key()
3443*/
3444
3445/*! \fn template <class Key, class T> QList<T> QMultiHash<Key, T>::values() const
3446
3447 Returns a list containing all the values in the hash, in an
3448 arbitrary order. If a key is associated with multiple values, all of
3449 its values will be in the list, and not just the most recently
3450 inserted one.
3451
3452 The order is guaranteed to be the same as that used by keys().
3453
3454 This function creates a new list, in \l {linear time}. The time and memory
3455 use that entails can be avoided by iterating from \l keyValueBegin() to
3456 \l keyValueEnd().
3457
3458 \sa keys(), value()
3459*/
3460
3461/*!
3462 \fn template <class Key, class T> Key QMultiHash<Key, T>::key(const T &value) const
3463 \fn template <class Key, class T> Key QMultiHash<Key, T>::key(const T &value, const Key &defaultKey) const
3464 \since 4.3
3465
3466 Returns the first key mapped to \a value. If the hash contains no item
3467 mapped to \a value, returns \a defaultKey, or a \l{default-constructed
3468 value}{default-constructed key} if this parameter has not been supplied.
3469
3470 This function can be slow (\l{linear time}), because QMultiHash's
3471 internal data structure is optimized for fast lookup by key, not
3472 by value.
3473*/
3474
3475/*!
3476 \fn template <class Key, class T> qsizetype QMultiHash<Key, T>::count(const Key &key, const T &value) const
3477 \since 4.3
3478
3479 Returns the number of items with the \a key and \a value.
3480
3481 \sa contains()
3482*/
3483
3484/*!
3485 \fn template <class Key, class T> typename QMultiHash<Key, T>::iterator QMultiHash<Key, T>::find(const Key &key, const T &value)
3486 \since 4.3
3487
3488 Returns an iterator pointing to the item with the \a key and \a value.
3489 If the hash contains no such item, the function returns end().
3490
3491 If the hash contains multiple items with the \a key and \a value, the
3492 iterator returned points to the most recently inserted item.
3493
3494 \include qhash.cpp qhash-iterator-invalidation-func-desc
3495*/
3496
3497/*!
3498 \fn template <class Key, class T> typename QMultiHash<Key, T>::const_iterator QMultiHash<Key, T>::find(const Key &key, const T &value) const
3499 \since 4.3
3500 \overload
3501
3502 \include qhash.cpp qhash-iterator-invalidation-func-desc
3503*/
3504
3505/*!
3506 \fn template <class Key, class T> typename QMultiHash<Key, T>::const_iterator QMultiHash<Key, T>::constFind(const Key &key, const T &value) const
3507 \since 4.3
3508
3509 Returns an iterator pointing to the item with the \a key and the
3510 \a value in the hash.
3511
3512 If the hash contains no such item, the function returns
3513 constEnd().
3514
3515 \include qhash.cpp qhash-iterator-invalidation-func-desc
3516*/
3517
3518/*! \fn template <class Key, class T> QMultiHash<Key, T>::iterator QMultiHash<Key, T>::begin()
3519
3520 Returns an \l{STL-style iterators}{STL-style iterator} pointing to the first item in
3521 the hash.
3522
3523 \include qhash.cpp qhash-iterator-invalidation-func-desc
3524
3525 \sa constBegin(), end()
3526*/
3527
3528/*! \fn template <class Key, class T> QMultiHash<Key, T>::const_iterator QMultiHash<Key, T>::begin() const
3529
3530 \overload
3531
3532 \include qhash.cpp qhash-iterator-invalidation-func-desc
3533*/
3534
3535/*! \fn template <class Key, class T> QMultiHash<Key, T>::const_iterator QMultiHash<Key, T>::cbegin() const
3536 \since 5.0
3537
3538 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the first item
3539 in the hash.
3540
3541 \include qhash.cpp qhash-iterator-invalidation-func-desc
3542
3543 \sa begin(), cend()
3544*/
3545
3546/*! \fn template <class Key, class T> QMultiHash<Key, T>::const_iterator QMultiHash<Key, T>::constBegin() const
3547
3548 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the first item
3549 in the hash.
3550
3551 \include qhash.cpp qhash-iterator-invalidation-func-desc
3552
3553 \sa begin(), constEnd()
3554*/
3555
3556/*! \fn template <class Key, class T> QMultiHash<Key, T>::key_iterator QMultiHash<Key, T>::keyBegin() const
3557 \since 5.6
3558
3559 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the first key
3560 in the hash.
3561
3562 \include qhash.cpp qhash-iterator-invalidation-func-desc
3563
3564 \sa keyEnd()
3565*/
3566
3567/*! \fn template <class Key, class T> QMultiHash<Key, T>::iterator QMultiHash<Key, T>::end()
3568
3569 Returns an \l{STL-style iterators}{STL-style iterator} pointing to the imaginary item
3570 after the last item in the hash.
3571
3572 \include qhash.cpp qhash-iterator-invalidation-func-desc
3573
3574 \sa begin(), constEnd()
3575*/
3576
3577/*! \fn template <class Key, class T> QMultiHash<Key, T>::const_iterator QMultiHash<Key, T>::end() const
3578
3579 \overload
3580*/
3581
3582/*! \fn template <class Key, class T> QMultiHash<Key, T>::const_iterator QMultiHash<Key, T>::constEnd() const
3583
3584 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the imaginary
3585 item after the last item in the hash.
3586
3587 \include qhash.cpp qhash-iterator-invalidation-func-desc
3588
3589 \sa constBegin(), end()
3590*/
3591
3592/*! \fn template <class Key, class T> QMultiHash<Key, T>::const_iterator QMultiHash<Key, T>::cend() const
3593 \since 5.0
3594
3595 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the imaginary
3596 item after the last item in the hash.
3597
3598 \include qhash.cpp qhash-iterator-invalidation-func-desc
3599
3600 \sa cbegin(), end()
3601*/
3602
3603/*! \fn template <class Key, class T> QMultiHash<Key, T>::key_iterator QMultiHash<Key, T>::keyEnd() const
3604 \since 5.6
3605
3606 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the imaginary
3607 item after the last key in the hash.
3608
3609 \include qhash.cpp qhash-iterator-invalidation-func-desc
3610
3611 \sa keyBegin()
3612*/
3613
3614/*! \fn template <class Key, class T> QMultiHash<Key, T>::key_value_iterator QMultiHash<Key, T>::keyValueBegin()
3615 \since 5.10
3616
3617 Returns an \l{STL-style iterators}{STL-style iterator} pointing to the first entry
3618 in the hash.
3619
3620 \include qhash.cpp qhash-iterator-invalidation-func-desc
3621
3622 \sa keyValueEnd()
3623*/
3624
3625/*! \fn template <class Key, class T> QMultiHash<Key, T>::key_value_iterator QMultiHash<Key, T>::keyValueEnd()
3626 \since 5.10
3627
3628 Returns an \l{STL-style iterators}{STL-style iterator} pointing to the imaginary
3629 entry after the last entry in the hash.
3630
3631 \include qhash.cpp qhash-iterator-invalidation-func-desc
3632
3633 \sa keyValueBegin()
3634*/
3635
3636/*! \fn template <class Key, class T> QMultiHash<Key, T>::const_key_value_iterator QMultiHash<Key, T>::keyValueBegin() const
3637 \since 5.10
3638
3639 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the first entry
3640 in the hash.
3641
3642 \include qhash.cpp qhash-iterator-invalidation-func-desc
3643
3644 \sa keyValueEnd()
3645*/
3646
3647/*! \fn template <class Key, class T> QMultiHash<Key, T>::const_key_value_iterator QMultiHash<Key, T>::constKeyValueBegin() const
3648 \since 5.10
3649
3650 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the first entry
3651 in the hash.
3652
3653 \include qhash.cpp qhash-iterator-invalidation-func-desc
3654
3655 \sa keyValueBegin()
3656*/
3657
3658/*! \fn template <class Key, class T> QMultiHash<Key, T>::const_key_value_iterator QMultiHash<Key, T>::keyValueEnd() const
3659 \since 5.10
3660
3661 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the imaginary
3662 entry after the last entry in the hash.
3663
3664 \include qhash.cpp qhash-iterator-invalidation-func-desc
3665
3666 \sa keyValueBegin()
3667*/
3668
3669/*! \fn template <class Key, class T> QMultiHash<Key, T>::const_key_value_iterator QMultiHash<Key, T>::constKeyValueEnd() const
3670 \since 5.10
3671
3672 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the imaginary
3673 entry after the last entry in the hash.
3674
3675 \include qhash.cpp qhash-iterator-invalidation-func-desc
3676
3677 \sa constKeyValueBegin()
3678*/
3679
3680/*! \fn template <class Key, class T> auto QMultiHash<Key, T>::asKeyValueRange() &
3681 \fn template <class Key, class T> auto QMultiHash<Key, T>::asKeyValueRange() const &
3682 \fn template <class Key, class T> auto QMultiHash<Key, T>::asKeyValueRange() &&
3683 \fn template <class Key, class T> auto QMultiHash<Key, T>::asKeyValueRange() const &&
3684 \since 6.4
3685
3686 Returns a range object that allows iteration over this hash as
3687 key/value pairs. For instance, this range object can be used in a
3688 range-based for loop, in combination with a structured binding declaration:
3689
3690 \snippet code/src_corelib_tools_qhash.cpp 35
3691
3692 Note that both the key and the value obtained this way are
3693 references to the ones in the hash. Specifically, mutating the value
3694 will modify the hash itself.
3695
3696 \include qhash.cpp qhash-iterator-invalidation-func-desc
3697
3698 \sa QKeyValueIterator
3699*/
3700
3701/*! \class QMultiHash::iterator
3702 \inmodule QtCore
3703 \brief The QMultiHash::iterator class provides an STL-style non-const iterator for QMultiHash.
3704
3705 QMultiHash<Key, T>::iterator allows you to iterate over a QMultiHash
3706 and to modify the value (but not the key) associated
3707 with a particular key. If you want to iterate over a const QMultiHash,
3708 you should use QMultiHash::const_iterator. It is generally good
3709 practice to use QMultiHash::const_iterator on a non-const QMultiHash as
3710 well, unless you need to change the QMultiHash through the iterator.
3711 Const iterators are slightly faster, and can improve code
3712 readability.
3713
3714 The default QMultiHash::iterator constructor creates an uninitialized
3715 iterator. You must initialize it using a QMultiHash function like
3716 QMultiHash::begin(), QMultiHash::end(), or QMultiHash::find() before you can
3717 start iterating. Here's a typical loop that prints all the (key,
3718 value) pairs stored in a hash:
3719
3720 \snippet code/src_corelib_tools_qhash.cpp 17
3721
3722 Unlike QMap, which orders its items by key, QMultiHash stores its
3723 items in an arbitrary order.
3724
3725 Here's an example that increments every value stored in the QMultiHash
3726 by 2:
3727
3728 \snippet code/src_corelib_tools_qhash.cpp 18
3729
3730 To remove elements from a QMultiHash you can use erase_if(QMultiHash<Key, T> &map, Predicate pred):
3731
3732 \snippet code/src_corelib_tools_qhash.cpp 21
3733
3734 Multiple iterators can be used on the same hash. However, be aware
3735 that any modification performed directly on the QHash (inserting and
3736 removing items) can cause the iterators to become invalid.
3737
3738 Inserting items into the hash or calling methods such as QHash::reserve()
3739 or QHash::squeeze() can invalidate all iterators pointing into the hash.
3740 Iterators are guaranteed to stay valid only as long as the QHash doesn't have
3741 to grow/shrink its internal hash table.
3742 Using any iterator after a rehashing operation has occurred will lead to undefined behavior.
3743
3744 If you need to keep iterators over a long period of time, we recommend
3745 that you use QMultiMap rather than QHash.
3746
3747 \warning Iterators on implicitly shared containers do not work
3748 exactly like STL-iterators. You should avoid copying a container
3749 while iterators are active on that container. For more information,
3750 read \l{Implicit sharing iterator problem}.
3751
3752 \sa QMultiHash::const_iterator, QMultiHash::key_iterator, QMultiHash::key_value_iterator
3753*/
3754
3755/*! \fn template <class Key, class T> QMultiHash<Key, T>::iterator::iterator()
3756
3757 Constructs an uninitialized iterator.
3758
3759 Functions like key(), value(), and operator++() must not be
3760 called on an uninitialized iterator. Use operator=() to assign a
3761 value to it before using it.
3762
3763 \sa QMultiHash::begin(), QMultiHash::end()
3764*/
3765
3766/*! \fn template <class Key, class T> const Key &QMultiHash<Key, T>::iterator::key() const
3767
3768 Returns the current item's key as a const reference.
3769
3770 There is no direct way of changing an item's key through an
3771 iterator, although it can be done by calling QMultiHash::erase()
3772 followed by QMultiHash::insert().
3773
3774 \sa value()
3775*/
3776
3777/*! \fn template <class Key, class T> T &QMultiHash<Key, T>::iterator::value() const
3778
3779 Returns a modifiable reference to the current item's value.
3780
3781 You can change the value of an item by using value() on
3782 the left side of an assignment, for example:
3783
3784 \snippet code/src_corelib_tools_qhash.cpp 22
3785
3786 \sa key(), operator*()
3787*/
3788
3789/*! \fn template <class Key, class T> T &QMultiHash<Key, T>::iterator::operator*() const
3790
3791 Returns a modifiable reference to the current item's value.
3792
3793 Same as value().
3794
3795 \sa key()
3796*/
3797
3798/*! \fn template <class Key, class T> T *QMultiHash<Key, T>::iterator::operator->() const
3799
3800 Returns a pointer to the current item's value.
3801
3802 \sa value()
3803*/
3804
3805/*!
3806 \fn template <class Key, class T> bool QMultiHash<Key, T>::iterator::operator==(const iterator &other) const
3807 \fn template <class Key, class T> bool QMultiHash<Key, T>::iterator::operator==(const const_iterator &other) const
3808
3809 Returns \c true if \a other points to the same item as this
3810 iterator; otherwise returns \c false.
3811
3812 \sa operator!=()
3813*/
3814
3815/*!
3816 \fn template <class Key, class T> bool QMultiHash<Key, T>::iterator::operator!=(const iterator &other) const
3817 \fn template <class Key, class T> bool QMultiHash<Key, T>::iterator::operator!=(const const_iterator &other) const
3818
3819 Returns \c true if \a other points to a different item than this
3820 iterator; otherwise returns \c false.
3821
3822 \sa operator==()
3823*/
3824
3825/*!
3826 \fn template <class Key, class T> QMultiHash<Key, T>::iterator &QMultiHash<Key, T>::iterator::operator++()
3827
3828 The prefix ++ operator (\c{++i}) advances the iterator to the
3829 next item in the hash and returns an iterator to the new current
3830 item.
3831
3832 Calling this function on QMultiHash::end() leads to undefined results.
3833*/
3834
3835/*! \fn template <class Key, class T> QMultiHash<Key, T>::iterator QMultiHash<Key, T>::iterator::operator++(int)
3836
3837 \overload
3838
3839 The postfix ++ operator (\c{i++}) advances the iterator to the
3840 next item in the hash and returns an iterator to the previously
3841 current item.
3842*/
3843
3844/*! \class QMultiHash::const_iterator
3845 \inmodule QtCore
3846 \brief The QMultiHash::const_iterator class provides an STL-style const iterator for QMultiHash.
3847
3848 QMultiHash<Key, T>::const_iterator allows you to iterate over a
3849 QMultiHash. If you want to modify the QMultiHash as you
3850 iterate over it, you must use QMultiHash::iterator instead. It is
3851 generally good practice to use QMultiHash::const_iterator on a
3852 non-const QMultiHash as well, unless you need to change the QMultiHash
3853 through the iterator. Const iterators are slightly faster, and
3854 can improve code readability.
3855
3856 The default QMultiHash::const_iterator constructor creates an
3857 uninitialized iterator. You must initialize it using a QMultiHash
3858 function like QMultiHash::cbegin(), QMultiHash::cend(), or
3859 QMultiHash::constFind() before you can start iterating. Here's a typical
3860 loop that prints all the (key, value) pairs stored in a hash:
3861
3862 \snippet code/src_corelib_tools_qhash.cpp 23
3863
3864 Unlike QMap, which orders its items by key, QMultiHash stores its
3865 items in an arbitrary order. The only guarantee is that items that
3866 share the same key (because they were inserted using
3867 a QMultiHash) will appear consecutively, from the most
3868 recently to the least recently inserted value.
3869
3870 Multiple iterators can be used on the same hash. However, be aware
3871 that any modification performed directly on the QMultiHash (inserting and
3872 removing items) can cause the iterators to become invalid.
3873
3874 Inserting items into the hash or calling methods such as QMultiHash::reserve()
3875 or QMultiHash::squeeze() can invalidate all iterators pointing into the hash.
3876 Iterators are guaranteed to stay valid only as long as the QMultiHash doesn't have
3877 to grow/shrink it's internal hash table.
3878 Using any iterator after a rehashing operation ahs occurred will lead to undefined behavior.
3879
3880 If you need to keep iterators over a long period of time, we recommend
3881 that you use QMultiMap rather than QMultiHash.
3882
3883 \warning Iterators on implicitly shared containers do not work
3884 exactly like STL-iterators. You should avoid copying a container
3885 while iterators are active on that container. For more information,
3886 read \l{Implicit sharing iterator problem}.
3887
3888 \sa QMultiHash::iterator, QMultiHash::key_iterator, QMultiHash::const_key_value_iterator
3889*/
3890
3891/*! \fn template <class Key, class T> QMultiHash<Key, T>::const_iterator::const_iterator()
3892
3893 Constructs an uninitialized iterator.
3894
3895 Functions like key(), value(), and operator++() must not be
3896 called on an uninitialized iterator. Use operator=() to assign a
3897 value to it before using it.
3898
3899 \sa QMultiHash::constBegin(), QMultiHash::constEnd()
3900*/
3901
3902/*! \fn template <class Key, class T> QMultiHash<Key, T>::const_iterator::const_iterator(const iterator &other)
3903
3904 Constructs a copy of \a other.
3905*/
3906
3907/*! \fn template <class Key, class T> const Key &QMultiHash<Key, T>::const_iterator::key() const
3908
3909 Returns the current item's key.
3910
3911 \sa value()
3912*/
3913
3914/*! \fn template <class Key, class T> const T &QMultiHash<Key, T>::const_iterator::value() const
3915
3916 Returns the current item's value.
3917
3918 \sa key(), operator*()
3919*/
3920
3921/*! \fn template <class Key, class T> const T &QMultiHash<Key, T>::const_iterator::operator*() const
3922
3923 Returns the current item's value.
3924
3925 Same as value().
3926
3927 \sa key()
3928*/
3929
3930/*! \fn template <class Key, class T> const T *QMultiHash<Key, T>::const_iterator::operator->() const
3931
3932 Returns a pointer to the current item's value.
3933
3934 \sa value()
3935*/
3936
3937/*! \fn template <class Key, class T> bool QMultiHash<Key, T>::const_iterator::operator==(const const_iterator &other) const
3938
3939 Returns \c true if \a other points to the same item as this
3940 iterator; otherwise returns \c false.
3941
3942 \sa operator!=()
3943*/
3944
3945/*! \fn template <class Key, class T> bool QMultiHash<Key, T>::const_iterator::operator!=(const const_iterator &other) const
3946
3947 Returns \c true if \a other points to a different item than this
3948 iterator; otherwise returns \c false.
3949
3950 \sa operator==()
3951*/
3952
3953/*!
3954 \fn template <class Key, class T> QMultiHash<Key, T>::const_iterator &QMultiHash<Key, T>::const_iterator::operator++()
3955
3956 The prefix ++ operator (\c{++i}) advances the iterator to the
3957 next item in the hash and returns an iterator to the new current
3958 item.
3959
3960 Calling this function on QMultiHash::end() leads to undefined results.
3961*/
3962
3963/*! \fn template <class Key, class T> QMultiHash<Key, T>::const_iterator QMultiHash<Key, T>::const_iterator::operator++(int)
3964
3965 \overload
3966
3967 The postfix ++ operator (\c{i++}) advances the iterator to the
3968 next item in the hash and returns an iterator to the previously
3969 current item.
3970*/
3971
3972/*! \class QMultiHash::key_iterator
3973 \inmodule QtCore
3974 \since 5.6
3975 \brief The QMultiHash::key_iterator class provides an STL-style const iterator for QMultiHash keys.
3976
3977 QMultiHash::key_iterator is essentially the same as QMultiHash::const_iterator
3978 with the difference that operator*() and operator->() return a key
3979 instead of a value.
3980
3981 For most uses QMultiHash::iterator and QMultiHash::const_iterator should be used,
3982 you can easily access the key by calling QMultiHash::iterator::key():
3983
3984 \snippet code/src_corelib_tools_qhash.cpp 27
3985
3986 However, to have interoperability between QMultiHash's keys and STL-style
3987 algorithms we need an iterator that dereferences to a key instead
3988 of a value. With QMultiHash::key_iterator we can apply an algorithm to a
3989 range of keys without having to call QMultiHash::keys(), which is inefficient
3990 as it costs one QMultiHash iteration and memory allocation to create a temporary
3991 QList.
3992
3993 \snippet code/src_corelib_tools_qhash.cpp 28
3994
3995 QMultiHash::key_iterator is const, it's not possible to modify the key.
3996
3997 The default QMultiHash::key_iterator constructor creates an uninitialized
3998 iterator. You must initialize it using a QMultiHash function like
3999 QMultiHash::keyBegin() or QMultiHash::keyEnd().
4000
4001 \warning Iterators on implicitly shared containers do not work
4002 exactly like STL-iterators. You should avoid copying a container
4003 while iterators are active on that container. For more information,
4004 read \l{Implicit sharing iterator problem}.
4005
4006 \sa QMultiHash::const_iterator, QMultiHash::iterator
4007*/
4008
4009/*! \fn template <class Key, class T> const T &QMultiHash<Key, T>::key_iterator::operator*() const
4010
4011 Returns the current item's key.
4012*/
4013
4014/*! \fn template <class Key, class T> const T *QMultiHash<Key, T>::key_iterator::operator->() const
4015
4016 Returns a pointer to the current item's key.
4017*/
4018
4019/*! \fn template <class Key, class T> bool QMultiHash<Key, T>::key_iterator::operator==(key_iterator other) const
4020
4021 Returns \c true if \a other points to the same item as this
4022 iterator; otherwise returns \c false.
4023
4024 \sa operator!=()
4025*/
4026
4027/*! \fn template <class Key, class T> bool QMultiHash<Key, T>::key_iterator::operator!=(key_iterator other) const
4028
4029 Returns \c true if \a other points to a different item than this
4030 iterator; otherwise returns \c false.
4031
4032 \sa operator==()
4033*/
4034
4035/*!
4036 \fn template <class Key, class T> QMultiHash<Key, T>::key_iterator &QMultiHash<Key, T>::key_iterator::operator++()
4037
4038 The prefix ++ operator (\c{++i}) advances the iterator to the
4039 next item in the hash and returns an iterator to the new current
4040 item.
4041
4042 Calling this function on QMultiHash::keyEnd() leads to undefined results.
4043*/
4044
4045/*! \fn template <class Key, class T> QMultiHash<Key, T>::key_iterator QMultiHash<Key, T>::key_iterator::operator++(int)
4046
4047 \overload
4048
4049 The postfix ++ operator (\c{i++}) advances the iterator to the
4050 next item in the hash and returns an iterator to the previous
4051 item.
4052*/
4053
4054/*! \fn template <class Key, class T> const_iterator QMultiHash<Key, T>::key_iterator::base() const
4055 Returns the underlying const_iterator this key_iterator is based on.
4056*/
4057
4058/*! \typedef QMultiHash::const_key_value_iterator
4059 \inmodule QtCore
4060 \since 5.10
4061 \brief The QMultiHash::const_key_value_iterator typedef provides an STL-style const iterator for QMultiHash.
4062
4063 QMultiHash::const_key_value_iterator is essentially the same as QMultiHash::const_iterator
4064 with the difference that operator*() returns a key/value pair instead of a
4065 value.
4066
4067 \sa QKeyValueIterator
4068*/
4069
4070/*! \typedef QMultiHash::key_value_iterator
4071 \inmodule QtCore
4072 \since 5.10
4073 \brief The QMultiHash::key_value_iterator typedef provides an STL-style iterator for QMultiHash.
4074
4075 QMultiHash::key_value_iterator is essentially the same as QMultiHash::iterator
4076 with the difference that operator*() returns a key/value pair instead of a
4077 value.
4078
4079 \sa QKeyValueIterator
4080*/
4081
4082/*! \fn template <class Key, class T> QDataStream &operator<<(QDataStream &out, const QMultiHash<Key, T>& hash)
4083 \relates QMultiHash
4084
4085 Writes the hash \a hash to stream \a out.
4086
4087 This function requires the key and value types to implement \c
4088 operator<<().
4089
4090 \sa {Serializing Qt Data Types}
4091*/
4092
4093/*! \fn template <class Key, class T> QDataStream &operator>>(QDataStream &in, QMultiHash<Key, T> &hash)
4094 \relates QMultiHash
4095
4096 Reads a hash from stream \a in into \a hash.
4097
4098 This function requires the key and value types to implement \c
4099 operator>>().
4100
4101 \sa {Serializing Qt Data Types}
4102*/
4103
4104/*!
4105 \fn template <class Key, class T> size_t qHash(const QHash<Key, T> &key, size_t seed = 0)
4106 \since 5.8
4107 \qhasholdTS{QHash}{Key}{T}
4108*/
4109
4110/*!
4111 \fn template <class Key, class T> size_t qHash(const QMultiHash<Key, T> &key, size_t seed = 0)
4112 \since 5.8
4113 \qhasholdTS{QMultiHash}{Key}{T}
4114*/
4115
4116/*! \fn template <typename Key, typename T, typename Predicate> qsizetype erase_if(QHash<Key, T> &hash, Predicate pred)
4117 \relates QHash
4118 \since 6.1
4119
4120 Removes all elements for which the predicate \a pred returns true
4121 from the hash \a hash.
4122
4123 The function supports predicates which take either an argument of
4124 type \c{QHash<Key, T>::iterator}, or an argument of type
4125 \c{std::pair<const Key &, T &>}.
4126
4127 Returns the number of elements removed, if any.
4128*/
4129
4130/*! \fn template <typename Key, typename T, typename Predicate> qsizetype erase_if(QMultiHash<Key, T> &hash, Predicate pred)
4131 \relates QMultiHash
4132 \since 6.1
4133
4134 Removes all elements for which the predicate \a pred returns true
4135 from the multi hash \a hash.
4136
4137 The function supports predicates which take either an argument of
4138 type \c{QMultiHash<Key, T>::iterator}, or an argument of type
4139 \c{std::pair<const Key &, T &>}.
4140
4141 Returns the number of elements removed, if any.
4142*/
4143
4144/*! \macro QT_NO_SINGLE_ARGUMENT_QHASH_OVERLOAD
4145 \relates QHash
4146 \since 6.11
4147
4148 Defining this macro disables the support for qHash overloads that only take
4149 one argument; in other words, for qHash overloads that do not also accept
4150 a seed. Support for the single-argument overloads of qHash is deprecated
4151 and will be removed in Qt 7.
4152
4153 \sa qHash
4154*/
4155
4156#ifdef QT_HAS_CONSTEXPR_BITOPS
4157namespace QHashPrivate {
4158static_assert(qPopulationCount(SpanConstants::NEntries) == 1,
4159 "NEntries must be a power of 2 for bucketForHash() to work.");
4160
4161// ensure the size of a Span does not depend on the template parameters
4162using Node1 = Node<int, int>;
4163static_assert(sizeof(Span<Node1>) == sizeof(Span<Node<char, void *>>));
4164static_assert(sizeof(Span<Node1>) == sizeof(Span<Node<qsizetype, QHashDummyValue>>));
4165static_assert(sizeof(Span<Node1>) == sizeof(Span<Node<QString, QVariant>>));
4166static_assert(sizeof(Span<Node1>) > SpanConstants::NEntries);
4167static_assert(qNextPowerOfTwo(sizeof(Span<Node1>)) == SpanConstants::NEntries * 2);
4168
4169// ensure allocations are always a power of two, at a minimum NEntries,
4170// obeying the fomula
4171// qNextPowerOfTwo(2 * N);
4172// without overflowing
4173static constexpr size_t NEntries = SpanConstants::NEntries;
4174static_assert(GrowthPolicy::bucketsForCapacity(1) == NEntries);
4175static_assert(GrowthPolicy::bucketsForCapacity(NEntries / 2 + 0) == NEntries);
4176static_assert(GrowthPolicy::bucketsForCapacity(NEntries / 2 + 1) == 2 * NEntries);
4177static_assert(GrowthPolicy::bucketsForCapacity(NEntries * 1 - 1) == 2 * NEntries);
4178static_assert(GrowthPolicy::bucketsForCapacity(NEntries * 1 + 0) == 4 * NEntries);
4179static_assert(GrowthPolicy::bucketsForCapacity(NEntries * 1 + 1) == 4 * NEntries);
4180static_assert(GrowthPolicy::bucketsForCapacity(NEntries * 2 - 1) == 4 * NEntries);
4181static_assert(GrowthPolicy::bucketsForCapacity(NEntries * 2 + 0) == 8 * NEntries);
4182static_assert(GrowthPolicy::bucketsForCapacity(SIZE_MAX / 4) == SIZE_MAX / 2 + 1);
4183static_assert(GrowthPolicy::bucketsForCapacity(SIZE_MAX / 2) == SIZE_MAX);
4184static_assert(GrowthPolicy::bucketsForCapacity(SIZE_MAX) == SIZE_MAX);
4185}
4186#endif
4187
4188QT_END_NAMESPACE
#define Q_BASIC_ATOMIC_INITIALIZER(a)
#define __has_feature(x)
size_t qHashBits(const void *p, size_t size, size_t seed) noexcept
Definition qhash.cpp:851
size_t qHashBits_fallback< None >(const uchar *p, size_t size, size_t seed, size_t seed2) noexcept
Definition qhash.cpp:299
size_t qHash(double key, size_t seed) noexcept
Definition qhash.cpp:1356
size_t qHashBits_fallback< ByteToWord >(const uchar *data, size_t size, size_t seed, size_t seed2) noexcept
Definition qhash.cpp:307
static size_t qHashBits_fallback(const uchar *p, size_t size, size_t seed, size_t seed2) noexcept
ZeroExtension
Definition qhash.cpp:292
@ ByteToWord
Definition qhash.cpp:294
@ None
Definition qhash.cpp:293
size_t qHash(QByteArrayView key, size_t seed) noexcept
Definition qhash.cpp:876
size_t qHash(long double key, size_t seed) noexcept
Definition qhash.cpp:1373
#define Q_DECL_HOT_FUNCTION
Definition qhash.cpp:51
uint qt_hash(QStringView key, uint chained) noexcept
Definition qhash.cpp:1087
constexpr size_t qHash(const QSize &s, size_t seed=0) noexcept
Definition qsize.h:192
Q_CORE_EXPORT void qt_from_latin1(char16_t *dst, const char *str, size_t size) noexcept
Definition qstring.cpp:921