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 if constexpr (QSysInfo::ByteOrder == QSysInfo::BigEndian)
1400 memset(buffer, 0, sizeof(long double) - Size);
1401 else
1402 memset(buffer + Size, 0, sizeof(long double) - Size);
1403 return murmurhash(buffer, sizeof(long double), seed);
1404 }
1405}
1406
1407/*!
1408 \fn template <typename Enum, std::enable_if_t<std::is_enum_v<Enum>, bool> = true> size_t qHash(Enum key, size_t seed)
1409 \since 6.5
1410 \qhashbuiltin
1411
1412 \note Prior to Qt 6.5, unscoped enums relied on the integer overloads of this
1413 function due to implicit conversion to their underlying integer types.
1414 For scoped enums, you had to implement an overload yourself. This is still the
1415 backwards-compatible fix to remain compatible with older Qt versions.
1416*/
1417
1418/*! \fn size_t qHash(const QChar key, size_t seed = 0)
1419 \since 5.0
1420 \qhashold{QHash}
1421*/
1422
1423/*! \fn size_t qHash(const QByteArray &key, size_t seed = 0)
1424 \since 5.0
1425 \qhashold{QHash}
1426*/
1427
1428/*! \fn size_t qHash(QByteArrayView key, size_t seed = 0)
1429 \since 6.0
1430 \qhashold{QHash}
1431*/
1432
1433/*! \fn size_t qHash(const QBitArray &key, size_t seed = 0)
1434 \since 5.0
1435 \qhashold{QHash}
1436*/
1437
1438/*! \fn size_t qHash(const QString &key, size_t seed = 0)
1439 \since 5.0
1440 \qhashold{QHash}
1441*/
1442
1443/*! \fn size_t qHash(QLatin1StringView key, size_t seed = 0)
1444 \since 5.0
1445 \qhashold{QHash}
1446*/
1447
1448/*! \fn template <class T> size_t qHash(const T *key, size_t seed = 0)
1449 \since 5.0
1450 \qhashbuiltin
1451*/
1452
1453/*! \fn size_t qHash(std::nullptr_t key, size_t seed = 0)
1454 \since 6.0
1455 \qhashbuiltin
1456*/
1457
1458/*! \fn template<typename T> bool qHashEquals(const T &a, const T &b)
1459 \relates QHash
1460 \since 6.0
1461 \internal
1462
1463 This method is being used by QHash to compare two keys. Returns true if the
1464 keys \a a and \a b are considered equal for hashing purposes.
1465
1466 The default implementation returns the result of (a == b). It can be reimplemented
1467 for a certain type if the equality operator is not suitable for hashing purposes.
1468 This is for example the case if the equality operator uses qFuzzyCompare to compare
1469 floating point values.
1470*/
1471
1472
1473/*!
1474 \class QHash
1475 \inmodule QtCore
1476 \brief The QHash class is a template class that provides a hash-table-based dictionary.
1477 \compares equality
1478
1479 \ingroup tools
1480 \ingroup shared
1481
1482 \reentrant
1483
1484 QHash<Key, T> is one of Qt's generic \l{container classes}, where
1485 \a Key is the type used for lookup keys and \a T is the mapped value
1486 type. It stores (key, value) pairs and provides very fast lookup of
1487 the value associated with a key.
1488
1489 QHash provides very similar functionality to QMap. The
1490 differences are:
1491
1492 \list
1493 \li QHash provides faster lookups than QMap. (See \l{Algorithmic
1494 Complexity} for details.)
1495 \li When iterating over a QMap, the items are always sorted by
1496 key. With QHash, the items are arbitrarily ordered.
1497 \li The key type of a QMap must provide operator<(). The key
1498 type of a QHash must provide operator==() and a global
1499 hash function called qHash() (see \l{qHash}).
1500 \endlist
1501
1502 Here's an example QHash with QString keys and \c int values:
1503 \snippet code/src_corelib_tools_qhash.cpp 0
1504
1505 To insert a (key, value) pair into the hash, you can use operator[]():
1506
1507 \snippet code/src_corelib_tools_qhash.cpp 1
1508
1509 This inserts the following three (key, value) pairs into the
1510 QHash: ("one", 1), ("three", 3), and ("seven", 7). Another way to
1511 insert items into the hash is to use insert():
1512
1513 \snippet code/src_corelib_tools_qhash.cpp 2
1514
1515 To look up a value, use operator[]() or value():
1516
1517 \snippet code/src_corelib_tools_qhash.cpp 3
1518
1519 If there is no item with the specified key in the hash, these
1520 functions return a \l{default-constructed value}.
1521
1522 If you want to check whether the hash contains a particular key,
1523 use contains():
1524
1525 \snippet code/src_corelib_tools_qhash.cpp 4
1526
1527 There is also a value() overload that uses its second argument as
1528 a default value if there is no item with the specified key:
1529
1530 \snippet code/src_corelib_tools_qhash.cpp 5
1531
1532 In general, we recommend that you use contains() and value()
1533 rather than operator[]() for looking up a key in a hash. The
1534 reason is that operator[]() silently inserts an item into the
1535 hash if no item exists with the same key (unless the hash is
1536 const). For example, the following code snippet will create 1000
1537 items in memory:
1538
1539 \snippet code/src_corelib_tools_qhash.cpp 6
1540
1541 To avoid this problem, replace \c hash[i] with \c hash.value(i)
1542 in the code above.
1543
1544 Internally, QHash uses a hash table to perform lookups. This
1545 hash table automatically grows to
1546 provide fast lookups without wasting too much memory. You can
1547 still control the size of the hash table by calling reserve() if
1548 you already know approximately how many items the QHash will
1549 contain, but this isn't necessary to obtain good performance. You
1550 can also call capacity() to retrieve the hash table's size.
1551
1552 QHash will not shrink automatically if items are removed from the
1553 table. To minimize the memory used by the hash, call squeeze().
1554
1555 To iterate through all the (key, value) pairs stored in a
1556 QHash, use \l {asKeyValueRange}():
1557
1558 \snippet code/src_corelib_tools_qhash.cpp 8
1559
1560 This function returns a range object that can be used with structured
1561 bindings. For manual iterator control, you can also use traditional
1562 \l{STL-style iterators} (QHash::const_iterator and QHash::iterator):
1563
1564 \snippet code/src_corelib_tools_qhash.cpp qhash-iterator-stl-style
1565
1566 To modify values, use iterators:
1567
1568 \snippet code/src_corelib_tools_qhash.cpp qhash-iterator-modify-values
1569
1570 QHash also provides \l{Java-style iterators} (QHashIterator and
1571 QMutableHashIterator) for compatibility.
1572
1573 QHash is unordered, so an iterator's sequence cannot be assumed
1574 to be predictable. If ordering by key is required, use a QMap.
1575
1576 A QHash allows only one value per key. If you call
1577 insert() with a key that already exists in the QHash, the
1578 previous value is erased. For example:
1579
1580 \snippet code/src_corelib_tools_qhash.cpp 9
1581
1582 If you need to store multiple entries for the same key in the
1583 hash table, use \l{QMultiHash}.
1584
1585 If you only need to extract the values from a hash (not the keys),
1586 you can also use range-based for:
1587
1588 \snippet code/src_corelib_tools_qhash.cpp 12
1589
1590 Items can be removed from the hash in several ways. One way is to
1591 call remove(); this will remove any item with the given key.
1592 Another way is to use QMutableHashIterator::remove(). In addition,
1593 you can clear the entire hash using clear().
1594
1595 QHash's key and value data types must be \l{assignable data
1596 types}. You cannot, for example, store a QWidget as a value;
1597 instead, store a QWidget *.
1598
1599 \target qHash
1600 \section2 The hashing function
1601
1602 A QHash's key type has additional requirements other than being an
1603 assignable data type: it must provide operator==(), and there must also be
1604 a hashing function that returns a hash value for an argument of the
1605 key's type.
1606
1607 The hashing function computes a numeric value based on a key. It
1608 can use any algorithm imaginable, as long as it always returns
1609 the same value if given the same argument. In other words, if
1610 \c{e1 == e2}, then \c{hash(e1) == hash(e2)} must hold as well.
1611 However, to obtain good performance, the hashing function should
1612 attempt to return different hash values for different keys to the
1613 largest extent possible.
1614
1615 A hashing function for a key type \c{K} may be provided in two
1616 different ways.
1617
1618 The first way is by having an overload of \c{qHash()} in \c{K}'s
1619 namespace. The \c{qHash()} function must have one of these signatures:
1620
1621 \snippet code/src_corelib_tools_qhash.cpp 32
1622
1623 The two-arguments overloads take an unsigned integer that should be used to
1624 seed the calculation of the hash function. This seed is provided by QHash
1625 in order to prevent a family of \l{algorithmic complexity attacks}.
1626
1627 \note In Qt 6 it is possible to define a \c{qHash()} overload
1628 taking only one argument; support for this is deprecated. Starting
1629 with Qt 7, it will be mandatory to use a two-arguments overload. If
1630 both a one-argument and a two-arguments overload are defined for a
1631 key type, the latter is used by QHash (note that you can simply
1632 define a two-arguments version, and use a default value for the
1633 seed parameter). In Qt 6 it is possible to disable support for the
1634 single argument qHash overload by defining the
1635 \c{QT_NO_SINGLE_ARGUMENT_QHASH_OVERLOAD} macro.
1636
1637 The second way to provide a hashing function is by specializing
1638 the \c{std::hash} class for the key type \c{K}, and providing a
1639 suitable function call operator for it:
1640
1641 \snippet code/src_corelib_tools_qhash.cpp 33
1642
1643 The seed argument has the same meaning as for \c{qHash()},
1644 and may be left out.
1645
1646 This second way allows to reuse the same hash function between
1647 QHash and the C++ Standard Library unordered associative containers.
1648 If both a \c{qHash()} overload and a \c{std::hash} specializations
1649 are provided for a type, then the \c{qHash()} overload is preferred.
1650
1651 Here's a partial list of the C++ and Qt types that can serve as keys in a
1652 QHash: any integer type (char, unsigned long, etc.), any pointer type,
1653 QChar, QString, and QByteArray. For all of these, the \c <QHash> header
1654 defines a qHash() function that computes an adequate hash value. Many other
1655 Qt classes also declare a qHash overload for their type; please refer to
1656 the documentation of each class.
1657
1658 If you want to use other types as the key, make sure that you provide
1659 operator==() and a hash implementation.
1660
1661 The convenience qHashMulti() function can be used to implement
1662 qHash() for a custom type, where one usually wants to produce a
1663 hash value from multiple fields:
1664
1665 Example:
1666 \snippet code/src_corelib_tools_qhash.cpp 13
1667
1668 In the example above, we've relied on Qt's own implementation of
1669 qHash() for QString and QDate to give us a hash value for the
1670 employee's name and date of birth respectively.
1671
1672 Note that the implementation of the qHash() overloads offered by Qt
1673 may change at any time. You \b{must not} rely on the fact that qHash()
1674 will give the same results (for the same inputs) across different Qt
1675 versions.
1676
1677 \section2 Algorithmic complexity attacks
1678
1679 All hash tables are vulnerable to a particular class of denial of service
1680 attacks, in which the attacker carefully pre-computes a set of different
1681 keys that are going to be hashed in the same bucket of a hash table (or
1682 even have the very same hash value). The attack aims at getting the
1683 worst-case algorithmic behavior (O(n) instead of amortized O(1), see
1684 \l{Algorithmic Complexity} for the details) when the data is fed into the
1685 table.
1686
1687 In order to avoid this worst-case behavior, the calculation of the hash
1688 value done by qHash() can be salted by a random seed, that nullifies the
1689 attack's extent. This seed is automatically generated by QHash once per
1690 process, and then passed by QHash as the second argument of the
1691 two-arguments overload of the qHash() function.
1692
1693 This randomization of QHash is enabled by default. Even though programs
1694 should never depend on a particular QHash ordering, there may be situations
1695 where you temporarily need deterministic behavior, for example for debugging or
1696 regression testing. To disable the randomization, define the environment
1697 variable \c QT_HASH_SEED to have the value 0. Alternatively, you can call
1698 the QHashSeed::setDeterministicGlobalSeed() function.
1699
1700 \sa QHashIterator, QMutableHashIterator, QMap, QSet
1701*/
1702
1703/*! \fn template <class Key, class T> QHash<Key, T>::QHash()
1704
1705 Constructs an empty hash.
1706
1707 \sa clear()
1708*/
1709
1710/*!
1711 \fn template <class Key, class T> QHash<Key, T>::QHash(QHash &&other)
1712
1713 Move-constructs a QHash instance, making it point at the same
1714 object that \a other was pointing to.
1715
1716 \since 5.2
1717*/
1718
1719/*! \fn template <class Key, class T> QHash<Key, T>::QHash(std::initializer_list<std::pair<Key,T> > list)
1720 \since 5.1
1721
1722 Constructs a hash with a copy of each of the elements in the
1723 initializer list \a list.
1724*/
1725
1726/*! \fn template <class Key, class T> template <class InputIterator> QHash<Key, T>::QHash(InputIterator begin, InputIterator end)
1727 \since 5.14
1728
1729 Constructs a hash with a copy of each of the elements in the iterator range
1730 [\a begin, \a end). Either the elements iterated by the range must be
1731 objects with \c{first} and \c{second} data members (like \c{std::pair}),
1732 convertible to \c Key and to \c T respectively; or the
1733 iterators must have \c{key()} and \c{value()} member functions, returning a
1734 key convertible to \c Key and a value convertible to \c T respectively.
1735*/
1736
1737/*! \fn template <class Key, class T> QHash<Key, T>::QHash(const QHash &other)
1738
1739 Constructs a copy of \a other.
1740
1741 This operation occurs in \l{constant time}, because QHash is
1742 \l{implicitly shared}. This makes returning a QHash from a
1743 function very fast. If a shared instance is modified, it will be
1744 copied (copy-on-write), and this takes \l{linear time}.
1745
1746 \sa operator=()
1747*/
1748
1749/*! \fn template <class Key, class T> QHash<Key, T>::~QHash()
1750
1751 Destroys the hash. References to the values in the hash and all
1752 iterators of this hash become invalid.
1753*/
1754
1755/*! \fn template <class Key, class T> QHash &QHash<Key, T>::operator=(const QHash &other)
1756
1757 Assigns \a other to this hash and returns a reference to this hash.
1758*/
1759
1760/*!
1761 \fn template <class Key, class T> QHash &QHash<Key, T>::operator=(QHash &&other)
1762
1763 Move-assigns \a other to this QHash instance.
1764
1765 \since 5.2
1766*/
1767
1768/*! \fn template <class Key, class T> void QHash<Key, T>::swap(QHash &other)
1769 \since 4.8
1770 \memberswap{hash}
1771*/
1772
1773/*! \fn template <class Key, class T> void QMultiHash<Key, T>::swap(QMultiHash &other)
1774 \since 4.8
1775 \memberswap{multi-hash}
1776*/
1777
1778/*! \fn template <class Key, class T> bool QHash<Key, T>::operator==(const QHash &lhs, const QHash &rhs)
1779
1780 Returns \c true if \a lhs hash is equal to \a rhs hash; otherwise returns
1781 \c false.
1782
1783 Two hashes are considered equal if they contain the same (key,
1784 value) pairs.
1785
1786 This function requires the value type to implement \c operator==().
1787
1788 \sa operator!=()
1789*/
1790
1791/*! \fn template <class Key, class T> bool QHash<Key, T>::operator!=(const QHash &lhs, const QHash &rhs)
1792
1793 Returns \c true if \a lhs hash is not equal to \a rhs hash; otherwise
1794 returns \c false.
1795
1796 Two hashes are considered equal if they contain the same (key,
1797 value) pairs.
1798
1799 This function requires the value type to implement \c operator==().
1800
1801 \sa operator==()
1802*/
1803
1804/*! \fn template <class Key, class T> qsizetype QHash<Key, T>::size() const
1805
1806 Returns the number of items in the hash.
1807
1808 \sa isEmpty(), count()
1809*/
1810
1811/*! \fn template <class Key, class T> bool QHash<Key, T>::isEmpty() const
1812
1813 Returns \c true if the hash contains no items; otherwise returns
1814 false.
1815
1816 \sa size()
1817*/
1818
1819/*! \fn template <class Key, class T> qsizetype QHash<Key, T>::capacity() const
1820
1821 Returns the number of buckets in the QHash's internal hash table.
1822
1823 The sole purpose of this function is to provide a means of fine
1824 tuning QHash's memory usage. In general, you will rarely ever
1825 need to call this function. If you want to know how many items are
1826 in the hash, call size().
1827
1828 \sa reserve(), squeeze()
1829*/
1830
1831/*! \fn template <class Key, class T> float QHash<Key, T>::load_factor() const noexcept
1832
1833 Returns the current load factor of the QHash's internal hash table.
1834 This is the same as capacity()/size(). The implementation used
1835 will aim to keep the load factor between 0.25 and 0.5. This avoids
1836 having too many hash table collisions that would degrade performance.
1837
1838 Even with a low load factor, the implementation of the hash table has a
1839 very low memory overhead.
1840
1841 This method purely exists for diagnostic purposes and you should rarely
1842 need to call it yourself.
1843
1844 \sa reserve(), squeeze()
1845*/
1846
1847
1848/*! \fn template <class Key, class T> void QHash<Key, T>::reserve(qsizetype size)
1849
1850 Ensures that the QHash's internal hash table has space to store at
1851 least \a size items without having to grow the hash table.
1852
1853 This implies that the hash table will contain at least 2 * \a size buckets
1854 to ensure good performance
1855
1856 This function is useful for code that needs to build a huge hash
1857 and wants to avoid repeated reallocation. For example:
1858
1859 \snippet code/src_corelib_tools_qhash.cpp 14
1860
1861 Ideally, \a size should be the maximum number of items expected
1862 in the hash. QHash will then choose the smallest possible
1863 number of buckets that will allow storing \a size items in the table
1864 without having to grow the internal hash table. If \a size
1865 is an underestimate, the worst that will happen is that the QHash
1866 will be a bit slower.
1867
1868 In general, you will rarely ever need to call this function.
1869 QHash's internal hash table automatically grows to
1870 provide good performance without wasting too much memory.
1871
1872 \sa squeeze(), capacity()
1873*/
1874
1875/*! \fn template <class Key, class T> void QHash<Key, T>::squeeze()
1876
1877 Reduces the size of the QHash's internal hash table to save
1878 memory.
1879
1880 The sole purpose of this function is to provide a means of fine
1881 tuning QHash's memory usage. In general, you will rarely ever
1882 need to call this function.
1883
1884 \sa reserve(), capacity()
1885*/
1886
1887/*! \fn template <class Key, class T> void QHash<Key, T>::detach()
1888
1889 \internal
1890
1891 Detaches this hash from any other hashes with which it may share
1892 data.
1893
1894 \sa isDetached()
1895*/
1896
1897/*! \fn template <class Key, class T> bool QHash<Key, T>::isDetached() const
1898
1899 \internal
1900
1901 Returns \c true if the hash's internal data isn't shared with any
1902 other hash object; otherwise returns \c false.
1903
1904 \sa detach()
1905*/
1906
1907/*! \fn template <class Key, class T> bool QHash<Key, T>::isSharedWith(const QHash &other) const
1908
1909 \internal
1910
1911 Returns true if the internal hash table of this QHash is shared with \a other, otherwise false.
1912*/
1913
1914/*! \fn template <class Key, class T> void QHash<Key, T>::clear()
1915
1916 Removes all items from the hash and frees up all memory used by it.
1917
1918 \sa remove()
1919*/
1920
1921/*! \fn template <class Key, class T> bool QHash<Key, T>::remove(const Key &key)
1922
1923 Removes the item that has the \a key from the hash.
1924 Returns true if the key exists in the hash and the item has been removed,
1925 and false otherwise.
1926
1927 \sa clear(), take()
1928*/
1929
1930/*! \fn template <class Key, class T> template <typename Predicate> qsizetype QHash<Key, T>::removeIf(Predicate pred)
1931 \since 6.1
1932
1933 Removes all elements for which the predicate \a pred returns true
1934 from the hash.
1935
1936 The function supports predicates which take either an argument of
1937 type \c{QHash<Key, T>::iterator}, or an argument of type
1938 \c{std::pair<const Key &, T &>}.
1939
1940 Returns the number of elements removed, if any.
1941
1942 \sa clear(), take()
1943*/
1944
1945/*! \fn template <class Key, class T> T QHash<Key, T>::take(const Key &key)
1946
1947 Removes the item with the \a key from the hash and returns
1948 the value associated with it.
1949
1950 If the item does not exist in the hash, the function simply
1951 returns a \l{default-constructed value}.
1952
1953 If you don't use the return value, remove() is more efficient.
1954
1955 \sa remove()
1956*/
1957
1958/*! \fn template <class Key, class T> bool QHash<Key, T>::contains(const Key &key) const
1959
1960 Returns \c true if the hash contains an item with the \a key;
1961 otherwise returns \c false.
1962
1963 \sa count()
1964*/
1965
1966/*! \fn template <class Key, class T> T QHash<Key, T>::value(const Key &key) const
1967 \fn template <class Key, class T> T QHash<Key, T>::value(const Key &key, const T &defaultValue) const
1968 \overload
1969
1970 Returns the value associated with the \a key.
1971
1972 If the hash contains no item with the \a key, the function
1973 returns \a defaultValue, or a \l{default-constructed value} if this
1974 parameter has not been supplied.
1975*/
1976
1977/*! \fn template <class Key, class T> T &QHash<Key, T>::operator[](const Key &key)
1978
1979 Returns the value associated with the \a key as a modifiable
1980 reference.
1981
1982 If the hash contains no item with the \a key, the function inserts
1983 a \l{default-constructed value} into the hash with the \a key, and
1984 returns a reference to it.
1985
1986//! [qhash-iterator-invalidation-func-desc]
1987 \warning Returned iterators/references should be considered invalidated
1988 the next time you call a non-const function on the hash, or when the
1989 hash is destroyed.
1990//! [qhash-iterator-invalidation-func-desc]
1991
1992 \sa insert(), value()
1993*/
1994
1995/*! \fn template <class Key, class T> const T QHash<Key, T>::operator[](const Key &key) const
1996
1997 \overload
1998
1999 Same as value().
2000*/
2001
2002/*! \fn template <class Key, class T> QList<Key> QHash<Key, T>::keys() const
2003
2004 Returns a list containing all the keys in the hash, in an
2005 arbitrary order.
2006
2007 The order is guaranteed to be the same as that used by values().
2008
2009 This function creates a new list, in \l {linear time}. The time and memory
2010 use that entails can be avoided by iterating from \l keyBegin() to
2011 \l keyEnd().
2012
2013 \sa values(), key()
2014*/
2015
2016/*! \fn template <class Key, class T> QList<Key> QHash<Key, T>::keys(const T &value) const
2017
2018 \overload
2019
2020 Returns a list containing all the keys associated with value \a
2021 value, in an arbitrary order.
2022
2023 This function can be slow (\l{linear time}), because QHash's
2024 internal data structure is optimized for fast lookup by key, not
2025 by value.
2026*/
2027
2028/*! \fn template <class Key, class T> QList<T> QHash<Key, T>::values() const
2029
2030 Returns a list containing all the values in the hash, in an
2031 arbitrary order.
2032
2033 The order is guaranteed to be the same as that used by keys().
2034
2035 This function creates a new list, in \l {linear time}. The time and memory
2036 use that entails can be avoided by iterating from \l keyValueBegin() to
2037 \l keyValueEnd().
2038
2039 \sa keys(), value()
2040*/
2041
2042/*!
2043 \fn template <class Key, class T> Key QHash<Key, T>::key(const T &value) const
2044 \fn template <class Key, class T> Key QHash<Key, T>::key(const T &value, const Key &defaultKey) const
2045 \since 4.3
2046
2047 Returns the first key mapped to \a value. If the hash contains no item
2048 mapped to \a value, returns \a defaultKey, or a \l{default-constructed
2049 value}{default-constructed key} if this parameter has not been supplied.
2050
2051 This function can be slow (\l{linear time}), because QHash's
2052 internal data structure is optimized for fast lookup by key, not
2053 by value.
2054*/
2055
2056/*! \fn template <class Key, class T> qsizetype QHash<Key, T>::count(const Key &key) const
2057
2058 Returns the number of items associated with the \a key.
2059
2060 \sa contains()
2061*/
2062
2063/*! \fn template <class Key, class T> qsizetype QHash<Key, T>::count() const
2064
2065 \overload
2066
2067 Same as size().
2068*/
2069
2070/*! \fn template <class Key, class T> QHash<Key, T>::iterator QHash<Key, T>::begin()
2071
2072 Returns an \l{STL-style iterators}{STL-style iterator} pointing to the first item in
2073 the hash.
2074
2075 \include qhash.cpp qhash-iterator-invalidation-func-desc
2076
2077 \sa constBegin(), end()
2078*/
2079
2080/*! \fn template <class Key, class T> QHash<Key, T>::const_iterator QHash<Key, T>::begin() const
2081
2082 \overload
2083
2084 \include qhash.cpp qhash-iterator-invalidation-func-desc
2085*/
2086
2087/*! \fn template <class Key, class T> QHash<Key, T>::const_iterator QHash<Key, T>::cbegin() const
2088 \since 5.0
2089
2090 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the first item
2091 in the hash.
2092
2093 \include qhash.cpp qhash-iterator-invalidation-func-desc
2094
2095 \sa begin(), cend()
2096*/
2097
2098/*! \fn template <class Key, class T> QHash<Key, T>::const_iterator QHash<Key, T>::constBegin() const
2099
2100 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the first item
2101 in the hash.
2102
2103 \include qhash.cpp qhash-iterator-invalidation-func-desc
2104
2105 \sa begin(), constEnd()
2106*/
2107
2108/*! \fn template <class Key, class T> QHash<Key, T>::key_iterator QHash<Key, T>::keyBegin() const
2109 \since 5.6
2110
2111 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the first key
2112 in the hash.
2113
2114 \include qhash.cpp qhash-iterator-invalidation-func-desc
2115
2116 \sa keyEnd()
2117*/
2118
2119/*! \fn template <class Key, class T> QHash<Key, T>::iterator QHash<Key, T>::end()
2120
2121 Returns an \l{STL-style iterators}{STL-style iterator} pointing to the imaginary item
2122 after the last item in the hash.
2123
2124 \include qhash.cpp qhash-iterator-invalidation-func-desc
2125
2126 \sa begin(), constEnd()
2127*/
2128
2129/*! \fn template <class Key, class T> QHash<Key, T>::const_iterator QHash<Key, T>::end() const
2130
2131 \overload
2132
2133 \include qhash.cpp qhash-iterator-invalidation-func-desc
2134*/
2135
2136/*! \fn template <class Key, class T> QHash<Key, T>::const_iterator QHash<Key, T>::constEnd() const
2137
2138 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the imaginary
2139 item after the last item in the hash.
2140
2141 \include qhash.cpp qhash-iterator-invalidation-func-desc
2142
2143 \sa constBegin(), end()
2144*/
2145
2146/*! \fn template <class Key, class T> QHash<Key, T>::const_iterator QHash<Key, T>::cend() const
2147 \since 5.0
2148
2149 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the imaginary
2150 item after the last item in the hash.
2151
2152 \include qhash.cpp qhash-iterator-invalidation-func-desc
2153
2154 \sa cbegin(), end()
2155*/
2156
2157/*! \fn template <class Key, class T> QHash<Key, T>::key_iterator QHash<Key, T>::keyEnd() const
2158 \since 5.6
2159
2160 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the imaginary
2161 item after the last key in the hash.
2162
2163 \include qhash.cpp qhash-iterator-invalidation-func-desc
2164
2165 \sa keyBegin()
2166*/
2167
2168/*! \fn template <class Key, class T> QHash<Key, T>::key_value_iterator QHash<Key, T>::keyValueBegin()
2169 \since 5.10
2170
2171 Returns an \l{STL-style iterators}{STL-style iterator} pointing to the first entry
2172 in the hash.
2173
2174 \include qhash.cpp qhash-iterator-invalidation-func-desc
2175
2176 \sa keyValueEnd()
2177*/
2178
2179/*! \fn template <class Key, class T> QHash<Key, T>::key_value_iterator QHash<Key, T>::keyValueEnd()
2180 \since 5.10
2181
2182 Returns an \l{STL-style iterators}{STL-style iterator} pointing to the imaginary
2183 entry after the last entry in the hash.
2184
2185 \include qhash.cpp qhash-iterator-invalidation-func-desc
2186
2187 \sa keyValueBegin()
2188*/
2189
2190/*! \fn template <class Key, class T> QHash<Key, T>::const_key_value_iterator QHash<Key, T>::keyValueBegin() const
2191 \since 5.10
2192
2193 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the first entry
2194 in the hash.
2195
2196 \include qhash.cpp qhash-iterator-invalidation-func-desc
2197
2198 \sa keyValueEnd()
2199*/
2200
2201/*! \fn template <class Key, class T> QHash<Key, T>::const_key_value_iterator QHash<Key, T>::constKeyValueBegin() const
2202 \since 5.10
2203
2204 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the first entry
2205 in the hash.
2206
2207 \include qhash.cpp qhash-iterator-invalidation-func-desc
2208
2209 \sa keyValueBegin()
2210*/
2211
2212/*! \fn template <class Key, class T> QHash<Key, T>::const_key_value_iterator QHash<Key, T>::keyValueEnd() const
2213 \since 5.10
2214
2215 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the imaginary
2216 entry after the last entry in the hash.
2217
2218 \include qhash.cpp qhash-iterator-invalidation-func-desc
2219
2220 \sa keyValueBegin()
2221*/
2222
2223/*! \fn template <class Key, class T> QHash<Key, T>::const_key_value_iterator QHash<Key, T>::constKeyValueEnd() const
2224 \since 5.10
2225
2226 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the imaginary
2227 entry after the last entry in the hash.
2228
2229 \include qhash.cpp qhash-iterator-invalidation-func-desc
2230
2231 \sa constKeyValueBegin()
2232*/
2233
2234/*! \fn template <class Key, class T> auto QHash<Key, T>::asKeyValueRange() &
2235 \fn template <class Key, class T> auto QHash<Key, T>::asKeyValueRange() const &
2236 \fn template <class Key, class T> auto QHash<Key, T>::asKeyValueRange() &&
2237 \fn template <class Key, class T> auto QHash<Key, T>::asKeyValueRange() const &&
2238 \since 6.4
2239
2240 Returns a range object that allows iteration over this hash as
2241 key/value pairs. For instance, this range object can be used in a
2242 range-based for loop, in combination with a structured binding declaration:
2243
2244 \snippet code/src_corelib_tools_qhash.cpp 34
2245
2246 Note that both the key and the value obtained this way are
2247 references to the ones in the hash. Specifically, mutating the value
2248 will modify the hash itself.
2249
2250 \include qhash.cpp qhash-iterator-invalidation-func-desc
2251
2252 \sa QKeyValueIterator
2253*/
2254
2255/*! \fn template <class Key, class T> QHash<Key, T>::iterator QHash<Key, T>::erase(const_iterator pos)
2256 \since 5.7
2257
2258 Removes the (key, value) pair associated with the iterator \a pos
2259 from the hash, and returns an iterator to the next item in the
2260 hash.
2261
2262 This function never causes QHash to
2263 rehash its internal data structure. This means that it can safely
2264 be called while iterating, and won't affect the order of items in
2265 the hash. For example:
2266
2267 \snippet code/src_corelib_tools_qhash.cpp 15
2268
2269 \include qhash.cpp qhash-iterator-invalidation-func-desc
2270
2271 \sa remove(), take(), find()
2272*/
2273
2274/*! \fn template <class Key, class T> QHash<Key, T>::iterator QHash<Key, T>::find(const Key &key)
2275
2276 Returns an iterator pointing to the item with the \a key in the
2277 hash.
2278
2279 If the hash contains no item with the \a key, the function
2280 returns end().
2281
2282 If the hash contains multiple items with the \a key, this
2283 function returns an iterator that points to the most recently
2284 inserted value. The other values are accessible by incrementing
2285 the iterator. For example, here's some code that iterates over all
2286 the items with the same key:
2287
2288 \snippet code/src_corelib_tools_qhash.cpp 16
2289
2290 \include qhash.cpp qhash-iterator-invalidation-func-desc
2291
2292 \sa value(), values()
2293*/
2294
2295/*! \fn template <class Key, class T> QHash<Key, T>::const_iterator QHash<Key, T>::find(const Key &key) const
2296
2297 \overload
2298
2299 \include qhash.cpp qhash-iterator-invalidation-func-desc
2300*/
2301
2302/*! \fn template <class Key, class T> QHash<Key, T>::const_iterator QHash<Key, T>::constFind(const Key &key) const
2303 \since 4.1
2304
2305 Returns an iterator pointing to the item with the \a key in the
2306 hash.
2307
2308 If the hash contains no item with the \a key, the function
2309 returns constEnd().
2310
2311 \include qhash.cpp qhash-iterator-invalidation-func-desc
2312
2313 \sa find()
2314*/
2315
2316/*! \fn template <class Key, class T> QHash<Key, T>::iterator QHash<Key, T>::insert(const Key &key, const T &value)
2317
2318 Inserts a new item with the \a key and a value of \a value.
2319
2320 If there is already an item with the \a key, that item's value
2321 is replaced with \a value.
2322
2323 Returns an iterator pointing to the new/updated element.
2324
2325 \include qhash.cpp qhash-iterator-invalidation-func-desc
2326*/
2327
2328/*!
2329 \fn template <class Key, class T> QHash<Key, T>::iterator QHash<Key, T>::insert(const Key &key, T &&value)
2330 \fn template <class Key, class T> QHash<Key, T>::iterator QHash<Key, T>::insert(Key &&key, const T &value)
2331 \fn template <class Key, class T> QHash<Key, T>::iterator QHash<Key, T>::insert(Key &&key, T &&value)
2332 \since 6.11
2333 \overload
2334*/
2335
2336/*!
2337 \fn template <class Key, class T> template <typename ...Args> QHash<Key, T>::iterator QHash<Key, T>::emplace(const Key &key, Args&&... args)
2338 \fn template <class Key, class T> template <typename ...Args> QHash<Key, T>::iterator QHash<Key, T>::emplace(Key &&key, Args&&... args)
2339
2340 Inserts a new element into the container. This new element
2341 is constructed in-place using \a args as the arguments for its
2342 construction.
2343
2344 Returns an iterator pointing to the new element.
2345
2346 \include qhash.cpp qhash-iterator-invalidation-func-desc
2347*/
2348
2349/*!
2350 \class QHash::TryEmplaceResult
2351 \inmodule QtCore
2352 \since 6.9
2353 \ingroup tools
2354 \brief The TryEmplaceResult class is used to represent the result of a tryEmplace() operation.
2355
2356 The \c{TryEmplaceResult} class is used in QHash to represent the result
2357 of a tryEmplace() operation. It holds an \l{iterator} to the newly
2358 created item, or to the pre-existing item that prevented the insertion, and
2359 a boolean, \l{inserted}, denoting whether the insertion took place.
2360
2361 \sa QHash, QHash::tryEmplace()
2362*/
2363
2364/*!
2365 \variable QHash::TryEmplaceResult::iterator
2366
2367 Holds the iterator to the newly inserted element, or the element that
2368 prevented the insertion.
2369*/
2370
2371/*!
2372 \variable QHash::TryEmplaceResult::inserted
2373
2374 This value is \c{false} if there was already an entry with the same key.
2375*/
2376
2377/*!
2378 \fn template <class Key, class T> template <typename... Args> QHash<Key, T>::TryEmplaceResult QHash<Key, T>::tryEmplace(const Key &key, Args &&...args)
2379 \fn template <class Key, class T> template <typename... Args> QHash<Key, T>::TryEmplaceResult QHash<Key, T>::tryEmplace(Key &&key, Args &&...args)
2380 \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)
2381 \since 6.9
2382
2383 Inserts a new item with the \a key and a value constructed from \a args.
2384 If an item with \a key already exists, no insertion takes place.
2385
2386 Returns an instance of \l{TryEmplaceResult}, a structure that holds an
2387 \l{QHash::TryEmplaceResult::}{iterator} to the newly created item, or
2388 to the pre-existing item that prevented the insertion, and a boolean,
2389 \l{QHash::TryEmplaceResult::}{inserted}, denoting whether the insertion
2390 took place.
2391
2392 For example, this can be used to avoid the pattern of comparing old and
2393 new size or double-lookups. Where you might previously have written code like:
2394
2395 \code
2396 QHash<int, MyType> hash;
2397 // [...]
2398 int myKey = getKey();
2399 qsizetype oldSize = hash.size();
2400 MyType &elem = hash[myKey];
2401 if (oldSize != hash.size()) // Size changed: new element!
2402 initialize(elem);
2403 // [use elem...]
2404 \endcode
2405
2406 You can instead write:
2407
2408 \code
2409 QHash<int, MyType> hash;
2410 // [...]
2411 int myKey = getKey();
2412 auto result = hash.tryEmplace(myKey);
2413 if (result.inserted) // New element!
2414 initialize(*result.iterator);
2415 // [use result.iterator...]
2416 \endcode
2417
2418 \sa emplace(), tryInsert(), insertOrAssign()
2419*/
2420
2421/*!
2422 \fn template <class Key, class T> QHash<Key, T>::TryEmplaceResult QHash<Key, T>::tryInsert(const Key &key, const T &value)
2423 \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)
2424 \since 6.9
2425
2426 Inserts a new item with the \a key and a value of \a value.
2427 If an item with \a key already exists, no insertion takes place.
2428
2429 Returns an instance of \l{TryEmplaceResult}, a structure that holds an
2430 \l{QHash::TryEmplaceResult::}{iterator} to the newly created item, or to the pre-existing item
2431 that prevented the insertion, and a boolean, \l{QHash::TryEmplaceResult::}{inserted}, denoting
2432 whether the insertion took place.
2433
2434 \sa insert(), tryEmplace(), insertOrAssign()
2435*/
2436
2437/*!
2438 \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)
2439 \fn template <class Key, class T> template <typename... Args> iterator QHash<Key, T>::try_emplace(const_iterator hint, const Key &key, Args &&...args)
2440 \fn template <class Key, class T> template <typename... Args> iterator QHash<Key, T>::try_emplace(const_iterator hint, Key &&key, Args &&...args)
2441 \since 6.9
2442
2443 Inserts a new item with the \a key and a value constructed from \a args.
2444 If an item with \a key already exists, no insertion takes place.
2445
2446 Returns the iterator of the inserted item, or to the item that prevented the
2447 insertion.
2448
2449 \a hint is ignored.
2450
2451 These functions are provided for compatibility with the standard library.
2452
2453 \sa emplace(), tryEmplace(), tryInsert(), insertOrAssign()
2454*/
2455
2456/*!
2457 \fn template <class Key, class T> template <typename... Args> std::pair<iterator, bool> QHash<Key, T>::try_emplace(const Key &key, Args &&...args)
2458 \fn template <class Key, class T> template <typename... Args> std::pair<iterator, bool> QHash<Key, T>::try_emplace(Key &&key, Args &&...args)
2459 \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)
2460 \since 6.9
2461
2462 Inserts a new item with the \a key and a value constructed from \a args.
2463 If an item with \a key already exists, no insertion takes place.
2464
2465 Returns a pair consisting of an iterator to the inserted item (or to the
2466 item that prevented the insertion), and a bool denoting whether the
2467 insertion took place.
2468
2469 These functions are provided for compatibility with the standard library.
2470
2471 \sa emplace(), tryEmplace(), tryInsert(), insertOrAssign()
2472*/
2473
2474/*!
2475 \fn template <class Key, class T> template <typename Value> QHash<Key, T>::TryEmplaceResult QHash<Key, T>::insertOrAssign(const Key &key, Value &&value)
2476 \fn template <class Key, class T> template <typename Value> QHash<Key, T>::TryEmplaceResult QHash<Key, T>::insertOrAssign(Key &&key, Value &&value)
2477 \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)
2478 \since 6.9
2479
2480 Attempts to insert an item with the \a key and \a value.
2481 If an item with \a key already exists its value is overwritten with \a value.
2482
2483 Returns an instance of \l{TryEmplaceResult}, a structure that holds an
2484 \l{QHash::TryEmplaceResult::}{iterator} to the item, and a boolean,
2485 \l{QHash::TryEmplaceResult::}{inserted}, denoting whether the item was newly created (\c{true})
2486 or if it previously existed (\c{false}).
2487
2488 \sa insert(), tryEmplace(), tryInsert()
2489*/
2490
2491/*!
2492 \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)
2493 \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)
2494 \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)
2495 \since 6.9
2496
2497 Attempts to insert an item with the \a key and \a value.
2498 If an item with \a key already exists its value is overwritten with \a value.
2499
2500 Returns a pair consisting of an iterator pointing to the item, and a
2501 boolean, denoting whether the item was newly created (\c{true}) or if it
2502 previously existed (\c{false}).
2503
2504 These functions are provided for compatibility with the standard library.
2505
2506 \sa insert(), tryEmplace(), tryInsert(), insertOrAssign()
2507*/
2508
2509/*!
2510 \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)
2511 \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)
2512 \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)
2513 \since 6.9
2514
2515 Attempts to insert an item with the \a key and \a value.
2516 If an item with \a key already exists its value is overwritten with \a value.
2517
2518 Returns a pair consisting of an iterator pointing to the item, and a
2519 boolean, denoting whether the item was newly created (\c{true}) or if it
2520 previously existed (\c{false}).
2521
2522 \a hint is ignored.
2523
2524 These functions are provided for compatibility with the standard library.
2525
2526 \sa insert(), tryEmplace(), insertOrAssign()
2527*/
2528
2529/*! \fn template <class Key, class T> void QHash<Key, T>::insert(const QHash &other)
2530 \since 5.15
2531
2532 Inserts all the items in the \a other hash into this hash.
2533
2534 If a key is common to both hashes, its value will be replaced with the
2535 value stored in \a other.
2536*/
2537
2538/*! \fn template <class Key, class T> bool QHash<Key, T>::empty() const
2539
2540 This function is provided for STL compatibility. It is equivalent
2541 to isEmpty(), returning true if the hash is empty; otherwise
2542 returns \c false.
2543*/
2544
2545/*! \fn template <class Key, class T> std::pair<iterator, iterator> QMultiHash<Key, T>::equal_range(const Key &key)
2546 \since 5.7
2547
2548 Returns a pair of iterators delimiting the range of values \c{[first, second)}, that
2549 are stored under \a key. If the range is empty then both iterators will be equal to end().
2550
2551 \include qhash.cpp qhash-iterator-invalidation-func-desc
2552*/
2553
2554/*!
2555 \fn template <class Key, class T> std::pair<const_iterator, const_iterator> QMultiHash<Key, T>::equal_range(const Key &key) const
2556 \overload
2557 \since 5.7
2558
2559 \include qhash.cpp qhash-iterator-invalidation-func-desc
2560*/
2561
2562/*! \typedef QHash::ConstIterator
2563
2564 Qt-style synonym for QHash::const_iterator.
2565*/
2566
2567/*! \typedef QHash::Iterator
2568
2569 Qt-style synonym for QHash::iterator.
2570*/
2571
2572/*! \typedef QHash::difference_type
2573
2574 Typedef for ptrdiff_t. Provided for STL compatibility.
2575*/
2576
2577/*! \typedef QHash::key_type
2578
2579 Typedef for Key. Provided for STL compatibility.
2580*/
2581
2582/*! \typedef QHash::mapped_type
2583
2584 Typedef for T. Provided for STL compatibility.
2585*/
2586
2587/*! \typedef QHash::size_type
2588
2589 Typedef for int. Provided for STL compatibility.
2590*/
2591
2592/*! \typedef QHash::iterator::difference_type
2593 \internal
2594*/
2595
2596/*! \typedef QHash::iterator::iterator_category
2597 \internal
2598*/
2599
2600/*! \typedef QHash::iterator::pointer
2601 \internal
2602*/
2603
2604/*! \typedef QHash::iterator::reference
2605 \internal
2606*/
2607
2608/*! \typedef QHash::iterator::value_type
2609 \internal
2610*/
2611
2612/*! \typedef QHash::const_iterator::difference_type
2613 \internal
2614*/
2615
2616/*! \typedef QHash::const_iterator::iterator_category
2617 \internal
2618*/
2619
2620/*! \typedef QHash::const_iterator::pointer
2621 \internal
2622*/
2623
2624/*! \typedef QHash::const_iterator::reference
2625 \internal
2626*/
2627
2628/*! \typedef QHash::const_iterator::value_type
2629 \internal
2630*/
2631
2632/*! \typedef QHash::key_iterator::difference_type
2633 \internal
2634*/
2635
2636/*! \typedef QHash::key_iterator::iterator_category
2637 \internal
2638*/
2639
2640/*! \typedef QHash::key_iterator::pointer
2641 \internal
2642*/
2643
2644/*! \typedef QHash::key_iterator::reference
2645 \internal
2646*/
2647
2648/*! \typedef QHash::key_iterator::value_type
2649 \internal
2650*/
2651
2652/*! \class QHash::iterator
2653 \inmodule QtCore
2654 \brief The QHash::iterator class provides an STL-style non-const iterator for QHash.
2655
2656 QHash<Key, T>::iterator allows you to iterate over a QHash
2657 and to modify the value (but not the key) associated
2658 with a particular key. If you want to iterate over a const QHash,
2659 you should use QHash::const_iterator. It is generally good
2660 practice to use QHash::const_iterator on a non-const QHash as
2661 well, unless you need to change the QHash through the iterator.
2662 Const iterators are slightly faster, and can improve code
2663 readability.
2664
2665 The default QHash::iterator constructor creates an uninitialized
2666 iterator. You must initialize it using a QHash function like
2667 QHash::begin(), QHash::end(), or QHash::find() before you can
2668 start iterating. Here's a typical loop that prints all the (key,
2669 value) pairs stored in a hash:
2670
2671 \snippet code/src_corelib_tools_qhash.cpp 17
2672
2673 Unlike QMap, which orders its items by key, QHash stores its
2674 items in an arbitrary order.
2675
2676 Here's an example that increments every value stored in the QHash
2677 by 2:
2678
2679 \snippet code/src_corelib_tools_qhash.cpp 18
2680
2681 To remove elements from a QHash you can use erase_if(QHash<Key, T> &map, Predicate pred):
2682
2683 \snippet code/src_corelib_tools_qhash.cpp 21
2684
2685 Multiple iterators can be used on the same hash. However, be aware
2686 that any modification performed directly on the QHash (inserting and
2687 removing items) can cause the iterators to become invalid.
2688
2689 Inserting items into the hash or calling methods such as QHash::reserve()
2690 or QHash::squeeze() can invalidate all iterators pointing into the hash.
2691 Iterators are guaranteed to stay valid only as long as the QHash doesn't have
2692 to grow/shrink its internal hash table.
2693 Using any iterator after a rehashing operation has occurred will lead to undefined behavior.
2694
2695 If you need to keep iterators over a long period of time, we recommend
2696 that you use QMap rather than QHash.
2697
2698 \warning Iterators on implicitly shared containers do not work
2699 exactly like STL-iterators. You should avoid copying a container
2700 while iterators are active on that container. For more information,
2701 read \l{Implicit sharing iterator problem}.
2702
2703 \sa QHash::const_iterator, QHash::key_iterator, QHash::key_value_iterator
2704*/
2705
2706/*! \fn template <class Key, class T> QHash<Key, T>::iterator::iterator()
2707
2708 Constructs an uninitialized iterator.
2709
2710 Functions like key(), value(), and operator++() must not be
2711 called on an uninitialized iterator. Use operator=() to assign a
2712 value to it before using it.
2713
2714 \sa QHash::begin(), QHash::end()
2715*/
2716
2717/*! \fn template <class Key, class T> const Key &QHash<Key, T>::iterator::key() const
2718
2719 Returns the current item's key as a const reference.
2720
2721 There is no direct way of changing an item's key through an
2722 iterator, although it can be done by calling QHash::erase()
2723 followed by QHash::insert().
2724
2725 \sa value()
2726*/
2727
2728/*! \fn template <class Key, class T> T &QHash<Key, T>::iterator::value() const
2729
2730 Returns a modifiable reference to the current item's value.
2731
2732 You can change the value of an item by using value() on
2733 the left side of an assignment, for example:
2734
2735 \snippet code/src_corelib_tools_qhash.cpp 22
2736
2737 \sa key(), operator*()
2738*/
2739
2740/*! \fn template <class Key, class T> T &QHash<Key, T>::iterator::operator*() const
2741
2742 Returns a modifiable reference to the current item's value.
2743
2744 Same as value().
2745
2746 \sa key()
2747*/
2748
2749/*! \fn template <class Key, class T> T *QHash<Key, T>::iterator::operator->() const
2750
2751 Returns a pointer to the current item's value.
2752
2753 \sa value()
2754*/
2755
2756/*!
2757 \fn template <class Key, class T> bool QHash<Key, T>::iterator::operator==(const iterator &other) const
2758 \fn template <class Key, class T> bool QHash<Key, T>::iterator::operator==(const const_iterator &other) const
2759
2760 Returns \c true if \a other points to the same item as this
2761 iterator; otherwise returns \c false.
2762
2763 \sa operator!=()
2764*/
2765
2766/*!
2767 \fn template <class Key, class T> bool QHash<Key, T>::iterator::operator!=(const iterator &other) const
2768 \fn template <class Key, class T> bool QHash<Key, T>::iterator::operator!=(const const_iterator &other) const
2769
2770 Returns \c true if \a other points to a different item than this
2771 iterator; otherwise returns \c false.
2772
2773 \sa operator==()
2774*/
2775
2776/*!
2777 \fn template <class Key, class T> QHash<Key, T>::iterator &QHash<Key, T>::iterator::operator++()
2778
2779 The prefix ++ operator (\c{++i}) advances the iterator to the
2780 next item in the hash and returns an iterator to the new current
2781 item.
2782
2783 Calling this function on QHash::end() leads to undefined results.
2784*/
2785
2786/*! \fn template <class Key, class T> QHash<Key, T>::iterator QHash<Key, T>::iterator::operator++(int)
2787
2788 \overload
2789
2790 The postfix ++ operator (\c{i++}) advances the iterator to the
2791 next item in the hash and returns an iterator to the previously
2792 current item.
2793*/
2794
2795/*! \class QHash::const_iterator
2796 \inmodule QtCore
2797 \brief The QHash::const_iterator class provides an STL-style const iterator for QHash.
2798
2799 QHash<Key, T>::const_iterator allows you to iterate over a
2800 QHash. If you want to modify the QHash as you
2801 iterate over it, you must use QHash::iterator instead. It is
2802 generally good practice to use QHash::const_iterator on a
2803 non-const QHash as well, unless you need to change the QHash
2804 through the iterator. Const iterators are slightly faster, and
2805 can improve code readability.
2806
2807 The default QHash::const_iterator constructor creates an
2808 uninitialized iterator. You must initialize it using a QHash
2809 function like QHash::cbegin(), QHash::cend(), or
2810 QHash::constFind() before you can start iterating. Here's a typical
2811 loop that prints all the (key, value) pairs stored in a hash:
2812
2813 \snippet code/src_corelib_tools_qhash.cpp 23
2814
2815 Unlike QMap, which orders its items by key, QHash stores its
2816 items in an arbitrary order. The only guarantee is that items that
2817 share the same key (because they were inserted using
2818 a QMultiHash) will appear consecutively, from the most
2819 recently to the least recently inserted value.
2820
2821 Multiple iterators can be used on the same hash. However, be aware
2822 that any modification performed directly on the QHash (inserting and
2823 removing items) can cause the iterators to become invalid.
2824
2825 Inserting items into the hash or calling methods such as QHash::reserve()
2826 or QHash::squeeze() can invalidate all iterators pointing into the hash.
2827 Iterators are guaranteed to stay valid only as long as the QHash doesn't have
2828 to grow/shrink its internal hash table.
2829 Using any iterator after a rehashing operation has occurred will lead to undefined behavior.
2830
2831 You can however safely use iterators to remove entries from the hash
2832 using the QHash::erase() method. This function can safely be called while
2833 iterating, and won't affect the order of items in the hash.
2834
2835 \warning Iterators on implicitly shared containers do not work
2836 exactly like STL-iterators. You should avoid copying a container
2837 while iterators are active on that container. For more information,
2838 read \l{Implicit sharing iterator problem}.
2839
2840 \sa QHash::iterator, QHash::key_iterator, QHash::const_key_value_iterator
2841*/
2842
2843/*! \fn template <class Key, class T> QHash<Key, T>::const_iterator::const_iterator()
2844
2845 Constructs an uninitialized iterator.
2846
2847 Functions like key(), value(), and operator++() must not be
2848 called on an uninitialized iterator. Use operator=() to assign a
2849 value to it before using it.
2850
2851 \sa QHash::constBegin(), QHash::constEnd()
2852*/
2853
2854/*! \fn template <class Key, class T> QHash<Key, T>::const_iterator::const_iterator(const iterator &other)
2855
2856 Constructs a copy of \a other.
2857*/
2858
2859/*! \fn template <class Key, class T> const Key &QHash<Key, T>::const_iterator::key() const
2860
2861 Returns the current item's key.
2862
2863 \sa value()
2864*/
2865
2866/*! \fn template <class Key, class T> const T &QHash<Key, T>::const_iterator::value() const
2867
2868 Returns the current item's value.
2869
2870 \sa key(), operator*()
2871*/
2872
2873/*! \fn template <class Key, class T> const T &QHash<Key, T>::const_iterator::operator*() const
2874
2875 Returns the current item's value.
2876
2877 Same as value().
2878
2879 \sa key()
2880*/
2881
2882/*! \fn template <class Key, class T> const T *QHash<Key, T>::const_iterator::operator->() const
2883
2884 Returns a pointer to the current item's value.
2885
2886 \sa value()
2887*/
2888
2889/*! \fn template <class Key, class T> bool QHash<Key, T>::const_iterator::operator==(const const_iterator &other) const
2890
2891 Returns \c true if \a other points to the same item as this
2892 iterator; otherwise returns \c false.
2893
2894 \sa operator!=()
2895*/
2896
2897/*! \fn template <class Key, class T> bool QHash<Key, T>::const_iterator::operator!=(const const_iterator &other) const
2898
2899 Returns \c true if \a other points to a different item than this
2900 iterator; otherwise returns \c false.
2901
2902 \sa operator==()
2903*/
2904
2905/*!
2906 \fn template <class Key, class T> QHash<Key, T>::const_iterator &QHash<Key, T>::const_iterator::operator++()
2907
2908 The prefix ++ operator (\c{++i}) advances the iterator to the
2909 next item in the hash and returns an iterator to the new current
2910 item.
2911
2912 Calling this function on QHash::end() leads to undefined results.
2913*/
2914
2915/*! \fn template <class Key, class T> QHash<Key, T>::const_iterator QHash<Key, T>::const_iterator::operator++(int)
2916
2917 \overload
2918
2919 The postfix ++ operator (\c{i++}) advances the iterator to the
2920 next item in the hash and returns an iterator to the previously
2921 current item.
2922*/
2923
2924/*! \class QHash::key_iterator
2925 \inmodule QtCore
2926 \since 5.6
2927 \brief The QHash::key_iterator class provides an STL-style const iterator for QHash keys.
2928
2929 QHash::key_iterator is essentially the same as QHash::const_iterator
2930 with the difference that operator*() and operator->() return a key
2931 instead of a value.
2932
2933 For most uses QHash::iterator and QHash::const_iterator should be used,
2934 you can easily access the key by calling QHash::iterator::key():
2935
2936 \snippet code/src_corelib_tools_qhash.cpp 27
2937
2938 However, to have interoperability between QHash's keys and STL-style
2939 algorithms we need an iterator that dereferences to a key instead
2940 of a value. With QHash::key_iterator we can apply an algorithm to a
2941 range of keys without having to call QHash::keys(), which is inefficient
2942 as it costs one QHash iteration and memory allocation to create a temporary
2943 QList.
2944
2945 \snippet code/src_corelib_tools_qhash.cpp 28
2946
2947 QHash::key_iterator is const, it's not possible to modify the key.
2948
2949 The default QHash::key_iterator constructor creates an uninitialized
2950 iterator. You must initialize it using a QHash function like
2951 QHash::keyBegin() or QHash::keyEnd().
2952
2953 \warning Iterators on implicitly shared containers do not work
2954 exactly like STL-iterators. You should avoid copying a container
2955 while iterators are active on that container. For more information,
2956 read \l{Implicit sharing iterator problem}.
2957
2958 \sa QHash::const_iterator, QHash::iterator
2959*/
2960
2961/*! \fn template <class Key, class T> const T &QHash<Key, T>::key_iterator::operator*() const
2962
2963 Returns the current item's key.
2964*/
2965
2966/*! \fn template <class Key, class T> const T *QHash<Key, T>::key_iterator::operator->() const
2967
2968 Returns a pointer to the current item's key.
2969*/
2970
2971/*! \fn template <class Key, class T> bool QHash<Key, T>::key_iterator::operator==(key_iterator other) const
2972
2973 Returns \c true if \a other points to the same item as this
2974 iterator; otherwise returns \c false.
2975
2976 \sa operator!=()
2977*/
2978
2979/*! \fn template <class Key, class T> bool QHash<Key, T>::key_iterator::operator!=(key_iterator other) const
2980
2981 Returns \c true if \a other points to a different item than this
2982 iterator; otherwise returns \c false.
2983
2984 \sa operator==()
2985*/
2986
2987/*!
2988 \fn template <class Key, class T> QHash<Key, T>::key_iterator &QHash<Key, T>::key_iterator::operator++()
2989
2990 The prefix ++ operator (\c{++i}) advances the iterator to the
2991 next item in the hash and returns an iterator to the new current
2992 item.
2993
2994 Calling this function on QHash::keyEnd() leads to undefined results.
2995
2996*/
2997
2998/*! \fn template <class Key, class T> QHash<Key, T>::key_iterator QHash<Key, T>::key_iterator::operator++(int)
2999
3000 \overload
3001
3002 The postfix ++ operator (\c{i++}) advances the iterator to the
3003 next item in the hash and returns an iterator to the previous
3004 item.
3005*/
3006
3007/*! \fn template <class Key, class T> const_iterator QHash<Key, T>::key_iterator::base() const
3008 Returns the underlying const_iterator this key_iterator is based on.
3009*/
3010
3011/*! \typedef QHash::const_key_value_iterator
3012 \inmodule QtCore
3013 \since 5.10
3014 \brief The QHash::const_key_value_iterator typedef provides an STL-style const iterator for QHash.
3015
3016 QHash::const_key_value_iterator is essentially the same as QHash::const_iterator
3017 with the difference that operator*() returns a key/value pair instead of a
3018 value.
3019
3020 \sa QKeyValueIterator
3021*/
3022
3023/*! \typedef QHash::key_value_iterator
3024 \inmodule QtCore
3025 \since 5.10
3026 \brief The QHash::key_value_iterator typedef provides an STL-style iterator for QHash.
3027
3028 QHash::key_value_iterator is essentially the same as QHash::iterator
3029 with the difference that operator*() returns a key/value pair instead of a
3030 value.
3031
3032 \sa QKeyValueIterator
3033*/
3034
3035/*! \fn template <class Key, class T> QDataStream &operator<<(QDataStream &out, const QHash<Key, T>& hash)
3036 \relates QHash
3037
3038 Writes the hash \a hash to stream \a out.
3039
3040 This function requires the key and value types to implement \c
3041 operator<<().
3042
3043 \sa {Serializing Qt Data Types}
3044*/
3045
3046/*! \fn template <class Key, class T> QDataStream &operator>>(QDataStream &in, QHash<Key, T> &hash)
3047 \relates QHash
3048
3049 Reads a hash from stream \a in into \a hash.
3050
3051 This function requires the key and value types to implement \c
3052 operator>>().
3053
3054 \sa {Serializing Qt Data Types}
3055*/
3056
3057/*! \class QMultiHash
3058 \inmodule QtCore
3059 \brief The QMultiHash class provides a multi-valued hash table.
3060 \compares equality
3061
3062 \ingroup tools
3063 \ingroup shared
3064
3065 \reentrant
3066
3067 QMultiHash<Key, T> is one of Qt's generic \l{container classes}, where
3068 \a Key is the type used for lookup keys and \a T is the mapped value type.
3069 It provides a hash table that allows multiple values for the same key.
3070
3071 QMultiHash mostly mirrors QHash's API. For example, you can use isEmpty() to test
3072 whether the hash is empty, and you can traverse a QMultiHash using
3073 QHash's iterator classes (for example, QHashIterator). But opposed to
3074 QHash, it provides an insert() function that allows the insertion of
3075 multiple items with the same key. The replace() function corresponds to
3076 QHash::insert(). It also provides convenient operator+() and
3077 operator+=().
3078
3079 Unlike QMultiMap, QMultiHash does not provide ordering of the
3080 inserted items. The only guarantee is that items that
3081 share the same key will appear consecutively, from the most
3082 recently to the least recently inserted value.
3083
3084 Example:
3085 \snippet code/src_corelib_tools_qhash.cpp 24
3086
3087 Unlike QHash, QMultiHash provides no operator[]. Use value() or
3088 replace() if you want to access the most recently inserted item
3089 with a certain key.
3090
3091 If you want to retrieve all the values for a single key, you can
3092 use values(const Key &key), which returns a QList<T>:
3093
3094 \snippet code/src_corelib_tools_qhash.cpp 25
3095
3096 The items that share the same key are available from most
3097 recently to least recently inserted.
3098
3099 A more efficient approach is to call find() to get
3100 the STL-style iterator for the first item with a key and iterate from
3101 there:
3102
3103 \snippet code/src_corelib_tools_qhash.cpp 26
3104
3105 QMultiHash's key and value data types must be \l{assignable data
3106 types}. You cannot, for example, store a QWidget as a value;
3107 instead, store a QWidget *. In addition, QMultiHash's key type
3108 must provide operator==(), and there must also be a qHash() function
3109 in the type's namespace that returns a hash value for an argument of the
3110 key's type. See the QHash documentation for details.
3111
3112 \sa QHash, QHashIterator, QMutableHashIterator, QMultiMap
3113*/
3114
3115/*! \fn template <class Key, class T> QMultiHash<Key, T>::QMultiHash()
3116
3117 Constructs an empty hash.
3118*/
3119
3120/*! \fn template <class Key, class T> QMultiHash<Key, T>::QMultiHash(std::initializer_list<std::pair<Key,T> > list)
3121 \since 5.1
3122
3123 Constructs a multi-hash with a copy of each of the elements in the
3124 initializer list \a list.
3125*/
3126
3127/*! \fn template <class Key, class T> QMultiHash<Key, T>::QMultiHash(const QHash<Key, T> &other)
3128
3129 Constructs a copy of \a other (which can be a QHash or a
3130 QMultiHash).
3131*/
3132
3133/*! \fn template <class Key, class T> template <class InputIterator> QMultiHash<Key, T>::QMultiHash(InputIterator begin, InputIterator end)
3134 \since 5.14
3135
3136 Constructs a multi-hash with a copy of each of the elements in the iterator range
3137 [\a begin, \a end). Either the elements iterated by the range must be
3138 objects with \c{first} and \c{second} data members (like \c{std::pair}),
3139 convertible to \c Key and to \c T respectively; or the
3140 iterators must have \c{key()} and \c{value()} member functions, returning a
3141 key convertible to \c Key and a value convertible to \c T respectively.
3142*/
3143
3144/*! \fn template <class Key, class T> QMultiHash<Key, T>::iterator QMultiHash<Key, T>::replace(const Key &key, const T &value)
3145
3146 Inserts a new item with the \a key and a value of \a value.
3147
3148 If there is already an item with the \a key, that item's value
3149 is replaced with \a value.
3150
3151 If there are multiple items with the \a key, the most
3152 recently inserted item's value is replaced with \a value.
3153
3154 Returns an iterator pointing to the new/updated element.
3155
3156 \include qhash.cpp qhash-iterator-invalidation-func-desc
3157
3158 \sa insert()
3159*/
3160
3161/*! \fn template <class Key, class T> QMultiHash<Key, T>::iterator QMultiHash<Key, T>::insert(const Key &key, const T &value)
3162
3163 Inserts a new item with the \a key and a value of \a value.
3164
3165 If there is already an item with the same key in the hash, this
3166 function will simply create a new one. (This behavior is
3167 different from replace(), which overwrites the value of an
3168 existing item.)
3169
3170 Returns an iterator pointing to the new element.
3171
3172 \include qhash.cpp qhash-iterator-invalidation-func-desc
3173
3174 \sa replace()
3175*/
3176
3177/*!
3178 \fn template <class Key, class T> QMultiHash<Key, T>::iterator QMultiHash<Key, T>::insert(const Key &key, T &&value)
3179 \fn template <class Key, class T> QMultiHash<Key, T>::iterator QMultiHash<Key, T>::insert(Key &&key, const T &value)
3180 \fn template <class Key, class T> QMultiHash<Key, T>::iterator QMultiHash<Key, T>::insert(Key &&key, T &&value)
3181 \since 6.11
3182 \overload
3183*/
3184
3185/*!
3186 \fn template <class Key, class T> template <typename ...Args> QMultiHash<Key, T>::iterator QMultiHash<Key, T>::emplace(const Key &key, Args&&... args)
3187 \fn template <class Key, class T> template <typename ...Args> QMultiHash<Key, T>::iterator QMultiHash<Key, T>::emplace(Key &&key, Args&&... args)
3188
3189 Inserts a new element into the container. This new element
3190 is constructed in-place using \a args as the arguments for its
3191 construction.
3192
3193 If there is already an item with the same key in the hash, this
3194 function will simply create a new one. (This behavior is
3195 different from replace(), which overwrites the value of an
3196 existing item.)
3197
3198 Returns an iterator pointing to the new element.
3199
3200 \include qhash.cpp qhash-iterator-invalidation-func-desc
3201
3202 \sa insert
3203*/
3204
3205/*!
3206 \fn template <class Key, class T> template <typename ...Args> QMultiHash<Key, T>::iterator QMultiHash<Key, T>::emplaceReplace(const Key &key, Args&&... args)
3207 \fn template <class Key, class T> template <typename ...Args> QMultiHash<Key, T>::iterator QMultiHash<Key, T>::emplaceReplace(Key &&key, Args&&... args)
3208
3209 Inserts a new element into the container. This new element
3210 is constructed in-place using \a args as the arguments for its
3211 construction.
3212
3213 If there is already an item with the same key in the hash, that item's
3214 value is replaced with a value constructed from \a args.
3215
3216 Returns an iterator pointing to the new element.
3217
3218 \include qhash.cpp qhash-iterator-invalidation-func-desc
3219
3220 \sa replace, emplace
3221*/
3222
3223/*! \fn template <class Key, class T> QMultiHash<Key, T>::iterator QMultiHash<Key, T>::erase(const_iterator pos)
3224 \since 5.7
3225
3226 Removes the (key, value) pair associated with the iterator \a pos
3227 from the hash, and returns an iterator to the next item in the
3228 hash.
3229
3230 This function never causes QMultiHash to
3231 rehash its internal data structure. This means that it can safely
3232 be called while iterating, and won't affect the order of items in
3233 the hash. For example:
3234
3235 \snippet code/src_corelib_tools_qhash.cpp 15multihash
3236
3237 \include qhash.cpp qhash-iterator-invalidation-func-desc
3238
3239 \sa remove(), take(), find()
3240*/
3241
3242/*! \fn template <class Key, class T> QMultiHash &QMultiHash<Key, T>::unite(const QMultiHash &other)
3243 \since 5.13
3244
3245 Inserts all the items in the \a other hash into this hash
3246 and returns a reference to this hash.
3247
3248 \sa insert()
3249*/
3250
3251
3252/*! \fn template <class Key, class T> QMultiHash &QMultiHash<Key, T>::unite(const QHash<Key, T> &other)
3253 \since 6.0
3254
3255 Inserts all the items in the \a other hash into this hash
3256 and returns a reference to this hash.
3257
3258 \sa insert()
3259*/
3260
3261/*! \fn template <class Key, class T> QList<Key> QMultiHash<Key, T>::uniqueKeys() const
3262 \since 5.13
3263
3264 Returns a list containing all the keys in the map. Keys that occur multiple
3265 times in the map occur only once in the returned list.
3266
3267 \sa keys(), values()
3268*/
3269
3270/*! \fn template <class Key, class T> T QMultiHash<Key, T>::value(const Key &key) const
3271 \fn template <class Key, class T> T QMultiHash<Key, T>::value(const Key &key, const T &defaultValue) const
3272
3273 Returns the value associated with the \a key.
3274
3275 If the hash contains no item with the \a key, the function
3276 returns \a defaultValue, or a \l{default-constructed value} if this
3277 parameter has not been supplied.
3278
3279 If there are multiple
3280 items for the \a key in the hash, the value of the most recently
3281 inserted one is returned.
3282*/
3283
3284/*! \fn template <class Key, class T> QList<T> QMultiHash<Key, T>::values(const Key &key) const
3285 \overload
3286
3287 Returns a list of all the values associated with the \a key,
3288 from the most recently inserted to the least recently inserted.
3289
3290 \sa count(), insert()
3291*/
3292
3293/*! \fn template <class Key, class T> T &QMultiHash<Key, T>::operator[](const Key &key)
3294
3295 Returns the value associated with the \a key as a modifiable reference.
3296
3297 If the hash contains no item with the \a key, the function inserts
3298 a \l{default-constructed value} into the hash with the \a key, and
3299 returns a reference to it.
3300
3301 If the hash contains multiple items with the \a key, this function returns
3302 a reference to the most recently inserted value.
3303
3304 \include qhash.cpp qhash-iterator-invalidation-func-desc
3305
3306 \sa insert(), value()
3307*/
3308
3309/*!
3310 \fn template <class Key, class T> bool QMultiHash<Key, T>::operator==(const QMultiHash &lhs, const QMultiHash &rhs)
3311
3312 Returns \c true if \a lhs multihash equals to the \a rhs multihash;
3313 otherwise returns \c false.
3314
3315 Two multihashes are considered equal if they contain the same (key, value)
3316 pairs.
3317
3318 This function requires the value type to implement \c {operator==()}.
3319
3320 \sa operator!=()
3321*/
3322
3323/*!
3324 \fn template <class Key, class T> bool QMultiHash<Key, T>::operator!=(const QMultiHash &lhs, const QMultiHash &rhs)
3325
3326 Returns \c true if \a lhs multihash is not equal to the \a rhs multihash;
3327 otherwise returns \c false.
3328
3329 Two multihashes are considered equal if they contain the same (key, value)
3330 pairs.
3331
3332 This function requires the value type to implement \c {operator==()}.
3333
3334 \sa operator==()
3335*/
3336
3337/*! \fn template <class Key, class T> QMultiHash &QMultiHash<Key, T>::operator+=(const QMultiHash &other)
3338
3339 Inserts all the items in the \a other hash into this hash
3340 and returns a reference to this hash.
3341
3342 \sa unite(), insert()
3343*/
3344
3345/*! \fn template <class Key, class T> QMultiHash QMultiHash<Key, T>::operator+(const QMultiHash &other) const
3346
3347 Returns a hash that contains all the items in this hash in
3348 addition to all the items in \a other. If a key is common to both
3349 hashes, the resulting hash will contain the key multiple times.
3350
3351 \sa operator+=()
3352*/
3353
3354/*!
3355 \fn template <class Key, class T> bool QMultiHash<Key, T>::contains(const Key &key, const T &value) const
3356 \since 4.3
3357
3358 Returns \c true if the hash contains an item with the \a key and
3359 \a value; otherwise returns \c false.
3360
3361 \sa count()
3362*/
3363
3364/*!
3365 \fn template <class Key, class T> qsizetype QMultiHash<Key, T>::remove(const Key &key)
3366 \since 4.3
3367
3368 Removes all the items that have the \a key from the hash.
3369 Returns the number of items removed.
3370
3371 \sa remove(const Key &key, const T &value)
3372*/
3373
3374/*!
3375 \fn template <class Key, class T> qsizetype QMultiHash<Key, T>::remove(const Key &key, const T &value)
3376 \since 4.3
3377
3378 Removes all the items that have the \a key and the value \a
3379 value from the hash. Returns the number of items removed.
3380
3381 \sa remove()
3382*/
3383
3384/*!
3385 \fn template <class Key, class T> void QMultiHash<Key, T>::clear()
3386 \since 4.3
3387
3388 Removes all items from the hash and frees up all memory used by it.
3389
3390 \sa remove()
3391*/
3392
3393/*! \fn template <class Key, class T> template <typename Predicate> qsizetype QMultiHash<Key, T>::removeIf(Predicate pred)
3394 \since 6.1
3395
3396 Removes all elements for which the predicate \a pred returns true
3397 from the multi hash.
3398
3399 The function supports predicates which take either an argument of
3400 type \c{QMultiHash<Key, T>::iterator}, or an argument of type
3401 \c{std::pair<const Key &, T &>}.
3402
3403 Returns the number of elements removed, if any.
3404
3405 \sa clear(), take()
3406*/
3407
3408/*! \fn template <class Key, class T> T QMultiHash<Key, T>::take(const Key &key)
3409
3410 Removes the item with the \a key from the hash and returns
3411 the value associated with it.
3412
3413 If the item does not exist in the hash, the function simply
3414 returns a \l{default-constructed value}. If there are multiple
3415 items for \a key in the hash, only the most recently inserted one
3416 is removed.
3417
3418 If you don't use the return value, remove() is more efficient.
3419
3420 \sa remove()
3421*/
3422
3423/*! \fn template <class Key, class T> QList<Key> QMultiHash<Key, T>::keys() const
3424
3425 Returns a list containing all the keys in the hash, in an
3426 arbitrary order. Keys that occur multiple times in the hash
3427 also occur multiple times in the list.
3428
3429 The order is guaranteed to be the same as that used by values().
3430
3431 This function creates a new list, in \l {linear time}. The time and memory
3432 use that entails can be avoided by iterating from \l keyBegin() to
3433 \l keyEnd().
3434
3435 \sa values(), key()
3436*/
3437
3438/*! \fn template <class Key, class T> QList<T> QMultiHash<Key, T>::values() const
3439
3440 Returns a list containing all the values in the hash, in an
3441 arbitrary order. If a key is associated with multiple values, all of
3442 its values will be in the list, and not just the most recently
3443 inserted one.
3444
3445 The order is guaranteed to be the same as that used by keys().
3446
3447 This function creates a new list, in \l {linear time}. The time and memory
3448 use that entails can be avoided by iterating from \l keyValueBegin() to
3449 \l keyValueEnd().
3450
3451 \sa keys(), value()
3452*/
3453
3454/*!
3455 \fn template <class Key, class T> Key QMultiHash<Key, T>::key(const T &value) const
3456 \fn template <class Key, class T> Key QMultiHash<Key, T>::key(const T &value, const Key &defaultKey) const
3457 \since 4.3
3458
3459 Returns the first key mapped to \a value. If the hash contains no item
3460 mapped to \a value, returns \a defaultKey, or a \l{default-constructed
3461 value}{default-constructed key} if this parameter has not been supplied.
3462
3463 This function can be slow (\l{linear time}), because QMultiHash's
3464 internal data structure is optimized for fast lookup by key, not
3465 by value.
3466*/
3467
3468/*!
3469 \fn template <class Key, class T> qsizetype QMultiHash<Key, T>::count(const Key &key, const T &value) const
3470 \since 4.3
3471
3472 Returns the number of items with the \a key and \a value.
3473
3474 \sa contains()
3475*/
3476
3477/*!
3478 \fn template <class Key, class T> typename QMultiHash<Key, T>::iterator QMultiHash<Key, T>::find(const Key &key, const T &value)
3479 \since 4.3
3480
3481 Returns an iterator pointing to the item with the \a key and \a value.
3482 If the hash contains no such item, the function returns end().
3483
3484 If the hash contains multiple items with the \a key and \a value, the
3485 iterator returned points to the most recently inserted item.
3486
3487 \include qhash.cpp qhash-iterator-invalidation-func-desc
3488*/
3489
3490/*!
3491 \fn template <class Key, class T> typename QMultiHash<Key, T>::const_iterator QMultiHash<Key, T>::find(const Key &key, const T &value) const
3492 \since 4.3
3493 \overload
3494
3495 \include qhash.cpp qhash-iterator-invalidation-func-desc
3496*/
3497
3498/*!
3499 \fn template <class Key, class T> typename QMultiHash<Key, T>::const_iterator QMultiHash<Key, T>::constFind(const Key &key, const T &value) const
3500 \since 4.3
3501
3502 Returns an iterator pointing to the item with the \a key and the
3503 \a value in the hash.
3504
3505 If the hash contains no such item, the function returns
3506 constEnd().
3507
3508 \include qhash.cpp qhash-iterator-invalidation-func-desc
3509*/
3510
3511/*! \fn template <class Key, class T> QMultiHash<Key, T>::iterator QMultiHash<Key, T>::begin()
3512
3513 Returns an \l{STL-style iterators}{STL-style iterator} pointing to the first item in
3514 the hash.
3515
3516 \include qhash.cpp qhash-iterator-invalidation-func-desc
3517
3518 \sa constBegin(), end()
3519*/
3520
3521/*! \fn template <class Key, class T> QMultiHash<Key, T>::const_iterator QMultiHash<Key, T>::begin() const
3522
3523 \overload
3524
3525 \include qhash.cpp qhash-iterator-invalidation-func-desc
3526*/
3527
3528/*! \fn template <class Key, class T> QMultiHash<Key, T>::const_iterator QMultiHash<Key, T>::cbegin() const
3529 \since 5.0
3530
3531 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the first item
3532 in the hash.
3533
3534 \include qhash.cpp qhash-iterator-invalidation-func-desc
3535
3536 \sa begin(), cend()
3537*/
3538
3539/*! \fn template <class Key, class T> QMultiHash<Key, T>::const_iterator QMultiHash<Key, T>::constBegin() const
3540
3541 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the first item
3542 in the hash.
3543
3544 \include qhash.cpp qhash-iterator-invalidation-func-desc
3545
3546 \sa begin(), constEnd()
3547*/
3548
3549/*! \fn template <class Key, class T> QMultiHash<Key, T>::key_iterator QMultiHash<Key, T>::keyBegin() const
3550 \since 5.6
3551
3552 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the first key
3553 in the hash.
3554
3555 \include qhash.cpp qhash-iterator-invalidation-func-desc
3556
3557 \sa keyEnd()
3558*/
3559
3560/*! \fn template <class Key, class T> QMultiHash<Key, T>::iterator QMultiHash<Key, T>::end()
3561
3562 Returns an \l{STL-style iterators}{STL-style iterator} pointing to the imaginary item
3563 after the last item in the hash.
3564
3565 \include qhash.cpp qhash-iterator-invalidation-func-desc
3566
3567 \sa begin(), constEnd()
3568*/
3569
3570/*! \fn template <class Key, class T> QMultiHash<Key, T>::const_iterator QMultiHash<Key, T>::end() const
3571
3572 \overload
3573*/
3574
3575/*! \fn template <class Key, class T> QMultiHash<Key, T>::const_iterator QMultiHash<Key, T>::constEnd() const
3576
3577 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the imaginary
3578 item after the last item in the hash.
3579
3580 \include qhash.cpp qhash-iterator-invalidation-func-desc
3581
3582 \sa constBegin(), end()
3583*/
3584
3585/*! \fn template <class Key, class T> QMultiHash<Key, T>::const_iterator QMultiHash<Key, T>::cend() const
3586 \since 5.0
3587
3588 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the imaginary
3589 item after the last item in the hash.
3590
3591 \include qhash.cpp qhash-iterator-invalidation-func-desc
3592
3593 \sa cbegin(), end()
3594*/
3595
3596/*! \fn template <class Key, class T> QMultiHash<Key, T>::key_iterator QMultiHash<Key, T>::keyEnd() const
3597 \since 5.6
3598
3599 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the imaginary
3600 item after the last key in the hash.
3601
3602 \include qhash.cpp qhash-iterator-invalidation-func-desc
3603
3604 \sa keyBegin()
3605*/
3606
3607/*! \fn template <class Key, class T> QMultiHash<Key, T>::key_value_iterator QMultiHash<Key, T>::keyValueBegin()
3608 \since 5.10
3609
3610 Returns an \l{STL-style iterators}{STL-style iterator} pointing to the first entry
3611 in the hash.
3612
3613 \include qhash.cpp qhash-iterator-invalidation-func-desc
3614
3615 \sa keyValueEnd()
3616*/
3617
3618/*! \fn template <class Key, class T> QMultiHash<Key, T>::key_value_iterator QMultiHash<Key, T>::keyValueEnd()
3619 \since 5.10
3620
3621 Returns an \l{STL-style iterators}{STL-style iterator} pointing to the imaginary
3622 entry after the last entry in the hash.
3623
3624 \include qhash.cpp qhash-iterator-invalidation-func-desc
3625
3626 \sa keyValueBegin()
3627*/
3628
3629/*! \fn template <class Key, class T> QMultiHash<Key, T>::const_key_value_iterator QMultiHash<Key, T>::keyValueBegin() const
3630 \since 5.10
3631
3632 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the first entry
3633 in the hash.
3634
3635 \include qhash.cpp qhash-iterator-invalidation-func-desc
3636
3637 \sa keyValueEnd()
3638*/
3639
3640/*! \fn template <class Key, class T> QMultiHash<Key, T>::const_key_value_iterator QMultiHash<Key, T>::constKeyValueBegin() const
3641 \since 5.10
3642
3643 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the first entry
3644 in the hash.
3645
3646 \include qhash.cpp qhash-iterator-invalidation-func-desc
3647
3648 \sa keyValueBegin()
3649*/
3650
3651/*! \fn template <class Key, class T> QMultiHash<Key, T>::const_key_value_iterator QMultiHash<Key, T>::keyValueEnd() const
3652 \since 5.10
3653
3654 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the imaginary
3655 entry after the last entry in the hash.
3656
3657 \include qhash.cpp qhash-iterator-invalidation-func-desc
3658
3659 \sa keyValueBegin()
3660*/
3661
3662/*! \fn template <class Key, class T> QMultiHash<Key, T>::const_key_value_iterator QMultiHash<Key, T>::constKeyValueEnd() const
3663 \since 5.10
3664
3665 Returns a const \l{STL-style iterators}{STL-style iterator} pointing to the imaginary
3666 entry after the last entry in the hash.
3667
3668 \include qhash.cpp qhash-iterator-invalidation-func-desc
3669
3670 \sa constKeyValueBegin()
3671*/
3672
3673/*! \fn template <class Key, class T> auto QMultiHash<Key, T>::asKeyValueRange() &
3674 \fn template <class Key, class T> auto QMultiHash<Key, T>::asKeyValueRange() const &
3675 \fn template <class Key, class T> auto QMultiHash<Key, T>::asKeyValueRange() &&
3676 \fn template <class Key, class T> auto QMultiHash<Key, T>::asKeyValueRange() const &&
3677 \since 6.4
3678
3679 Returns a range object that allows iteration over this hash as
3680 key/value pairs. For instance, this range object can be used in a
3681 range-based for loop, in combination with a structured binding declaration:
3682
3683 \snippet code/src_corelib_tools_qhash.cpp 35
3684
3685 Note that both the key and the value obtained this way are
3686 references to the ones in the hash. Specifically, mutating the value
3687 will modify the hash itself.
3688
3689 \include qhash.cpp qhash-iterator-invalidation-func-desc
3690
3691 \sa QKeyValueIterator
3692*/
3693
3694/*! \class QMultiHash::iterator
3695 \inmodule QtCore
3696 \brief The QMultiHash::iterator class provides an STL-style non-const iterator for QMultiHash.
3697
3698 QMultiHash<Key, T>::iterator allows you to iterate over a QMultiHash
3699 and to modify the value (but not the key) associated
3700 with a particular key. If you want to iterate over a const QMultiHash,
3701 you should use QMultiHash::const_iterator. It is generally good
3702 practice to use QMultiHash::const_iterator on a non-const QMultiHash as
3703 well, unless you need to change the QMultiHash through the iterator.
3704 Const iterators are slightly faster, and can improve code
3705 readability.
3706
3707 The default QMultiHash::iterator constructor creates an uninitialized
3708 iterator. You must initialize it using a QMultiHash function like
3709 QMultiHash::begin(), QMultiHash::end(), or QMultiHash::find() before you can
3710 start iterating. Here's a typical loop that prints all the (key,
3711 value) pairs stored in a hash:
3712
3713 \snippet code/src_corelib_tools_qhash.cpp 17
3714
3715 Unlike QMap, which orders its items by key, QMultiHash stores its
3716 items in an arbitrary order.
3717
3718 Here's an example that increments every value stored in the QMultiHash
3719 by 2:
3720
3721 \snippet code/src_corelib_tools_qhash.cpp 18
3722
3723 To remove elements from a QMultiHash you can use erase_if(QMultiHash<Key, T> &map, Predicate pred):
3724
3725 \snippet code/src_corelib_tools_qhash.cpp 21
3726
3727 Multiple iterators can be used on the same hash. However, be aware
3728 that any modification performed directly on the QHash (inserting and
3729 removing items) can cause the iterators to become invalid.
3730
3731 Inserting items into the hash or calling methods such as QHash::reserve()
3732 or QHash::squeeze() can invalidate all iterators pointing into the hash.
3733 Iterators are guaranteed to stay valid only as long as the QHash doesn't have
3734 to grow/shrink its internal hash table.
3735 Using any iterator after a rehashing operation has occurred will lead to undefined behavior.
3736
3737 If you need to keep iterators over a long period of time, we recommend
3738 that you use QMultiMap rather than QHash.
3739
3740 \warning Iterators on implicitly shared containers do not work
3741 exactly like STL-iterators. You should avoid copying a container
3742 while iterators are active on that container. For more information,
3743 read \l{Implicit sharing iterator problem}.
3744
3745 \sa QMultiHash::const_iterator, QMultiHash::key_iterator, QMultiHash::key_value_iterator
3746*/
3747
3748/*! \fn template <class Key, class T> QMultiHash<Key, T>::iterator::iterator()
3749
3750 Constructs an uninitialized iterator.
3751
3752 Functions like key(), value(), and operator++() must not be
3753 called on an uninitialized iterator. Use operator=() to assign a
3754 value to it before using it.
3755
3756 \sa QMultiHash::begin(), QMultiHash::end()
3757*/
3758
3759/*! \fn template <class Key, class T> const Key &QMultiHash<Key, T>::iterator::key() const
3760
3761 Returns the current item's key as a const reference.
3762
3763 There is no direct way of changing an item's key through an
3764 iterator, although it can be done by calling QMultiHash::erase()
3765 followed by QMultiHash::insert().
3766
3767 \sa value()
3768*/
3769
3770/*! \fn template <class Key, class T> T &QMultiHash<Key, T>::iterator::value() const
3771
3772 Returns a modifiable reference to the current item's value.
3773
3774 You can change the value of an item by using value() on
3775 the left side of an assignment, for example:
3776
3777 \snippet code/src_corelib_tools_qhash.cpp 22
3778
3779 \sa key(), operator*()
3780*/
3781
3782/*! \fn template <class Key, class T> T &QMultiHash<Key, T>::iterator::operator*() const
3783
3784 Returns a modifiable reference to the current item's value.
3785
3786 Same as value().
3787
3788 \sa key()
3789*/
3790
3791/*! \fn template <class Key, class T> T *QMultiHash<Key, T>::iterator::operator->() const
3792
3793 Returns a pointer to the current item's value.
3794
3795 \sa value()
3796*/
3797
3798/*!
3799 \fn template <class Key, class T> bool QMultiHash<Key, T>::iterator::operator==(const iterator &other) const
3800 \fn template <class Key, class T> bool QMultiHash<Key, T>::iterator::operator==(const const_iterator &other) const
3801
3802 Returns \c true if \a other points to the same item as this
3803 iterator; otherwise returns \c false.
3804
3805 \sa operator!=()
3806*/
3807
3808/*!
3809 \fn template <class Key, class T> bool QMultiHash<Key, T>::iterator::operator!=(const iterator &other) const
3810 \fn template <class Key, class T> bool QMultiHash<Key, T>::iterator::operator!=(const const_iterator &other) const
3811
3812 Returns \c true if \a other points to a different item than this
3813 iterator; otherwise returns \c false.
3814
3815 \sa operator==()
3816*/
3817
3818/*!
3819 \fn template <class Key, class T> QMultiHash<Key, T>::iterator &QMultiHash<Key, T>::iterator::operator++()
3820
3821 The prefix ++ operator (\c{++i}) advances the iterator to the
3822 next item in the hash and returns an iterator to the new current
3823 item.
3824
3825 Calling this function on QMultiHash::end() leads to undefined results.
3826*/
3827
3828/*! \fn template <class Key, class T> QMultiHash<Key, T>::iterator QMultiHash<Key, T>::iterator::operator++(int)
3829
3830 \overload
3831
3832 The postfix ++ operator (\c{i++}) advances the iterator to the
3833 next item in the hash and returns an iterator to the previously
3834 current item.
3835*/
3836
3837/*! \class QMultiHash::const_iterator
3838 \inmodule QtCore
3839 \brief The QMultiHash::const_iterator class provides an STL-style const iterator for QMultiHash.
3840
3841 QMultiHash<Key, T>::const_iterator allows you to iterate over a
3842 QMultiHash. If you want to modify the QMultiHash as you
3843 iterate over it, you must use QMultiHash::iterator instead. It is
3844 generally good practice to use QMultiHash::const_iterator on a
3845 non-const QMultiHash as well, unless you need to change the QMultiHash
3846 through the iterator. Const iterators are slightly faster, and
3847 can improve code readability.
3848
3849 The default QMultiHash::const_iterator constructor creates an
3850 uninitialized iterator. You must initialize it using a QMultiHash
3851 function like QMultiHash::cbegin(), QMultiHash::cend(), or
3852 QMultiHash::constFind() before you can start iterating. Here's a typical
3853 loop that prints all the (key, value) pairs stored in a hash:
3854
3855 \snippet code/src_corelib_tools_qhash.cpp 23
3856
3857 Unlike QMap, which orders its items by key, QMultiHash stores its
3858 items in an arbitrary order. The only guarantee is that items that
3859 share the same key (because they were inserted using
3860 a QMultiHash) will appear consecutively, from the most
3861 recently to the least recently inserted value.
3862
3863 Multiple iterators can be used on the same hash. However, be aware
3864 that any modification performed directly on the QMultiHash (inserting and
3865 removing items) can cause the iterators to become invalid.
3866
3867 Inserting items into the hash or calling methods such as QMultiHash::reserve()
3868 or QMultiHash::squeeze() can invalidate all iterators pointing into the hash.
3869 Iterators are guaranteed to stay valid only as long as the QMultiHash doesn't have
3870 to grow/shrink it's internal hash table.
3871 Using any iterator after a rehashing operation ahs occurred will lead to undefined behavior.
3872
3873 If you need to keep iterators over a long period of time, we recommend
3874 that you use QMultiMap rather than QMultiHash.
3875
3876 \warning Iterators on implicitly shared containers do not work
3877 exactly like STL-iterators. You should avoid copying a container
3878 while iterators are active on that container. For more information,
3879 read \l{Implicit sharing iterator problem}.
3880
3881 \sa QMultiHash::iterator, QMultiHash::key_iterator, QMultiHash::const_key_value_iterator
3882*/
3883
3884/*! \fn template <class Key, class T> QMultiHash<Key, T>::const_iterator::const_iterator()
3885
3886 Constructs an uninitialized iterator.
3887
3888 Functions like key(), value(), and operator++() must not be
3889 called on an uninitialized iterator. Use operator=() to assign a
3890 value to it before using it.
3891
3892 \sa QMultiHash::constBegin(), QMultiHash::constEnd()
3893*/
3894
3895/*! \fn template <class Key, class T> QMultiHash<Key, T>::const_iterator::const_iterator(const iterator &other)
3896
3897 Constructs a copy of \a other.
3898*/
3899
3900/*! \fn template <class Key, class T> const Key &QMultiHash<Key, T>::const_iterator::key() const
3901
3902 Returns the current item's key.
3903
3904 \sa value()
3905*/
3906
3907/*! \fn template <class Key, class T> const T &QMultiHash<Key, T>::const_iterator::value() const
3908
3909 Returns the current item's value.
3910
3911 \sa key(), operator*()
3912*/
3913
3914/*! \fn template <class Key, class T> const T &QMultiHash<Key, T>::const_iterator::operator*() const
3915
3916 Returns the current item's value.
3917
3918 Same as value().
3919
3920 \sa key()
3921*/
3922
3923/*! \fn template <class Key, class T> const T *QMultiHash<Key, T>::const_iterator::operator->() const
3924
3925 Returns a pointer to the current item's value.
3926
3927 \sa value()
3928*/
3929
3930/*! \fn template <class Key, class T> bool QMultiHash<Key, T>::const_iterator::operator==(const const_iterator &other) const
3931
3932 Returns \c true if \a other points to the same item as this
3933 iterator; otherwise returns \c false.
3934
3935 \sa operator!=()
3936*/
3937
3938/*! \fn template <class Key, class T> bool QMultiHash<Key, T>::const_iterator::operator!=(const const_iterator &other) const
3939
3940 Returns \c true if \a other points to a different item than this
3941 iterator; otherwise returns \c false.
3942
3943 \sa operator==()
3944*/
3945
3946/*!
3947 \fn template <class Key, class T> QMultiHash<Key, T>::const_iterator &QMultiHash<Key, T>::const_iterator::operator++()
3948
3949 The prefix ++ operator (\c{++i}) advances the iterator to the
3950 next item in the hash and returns an iterator to the new current
3951 item.
3952
3953 Calling this function on QMultiHash::end() leads to undefined results.
3954*/
3955
3956/*! \fn template <class Key, class T> QMultiHash<Key, T>::const_iterator QMultiHash<Key, T>::const_iterator::operator++(int)
3957
3958 \overload
3959
3960 The postfix ++ operator (\c{i++}) advances the iterator to the
3961 next item in the hash and returns an iterator to the previously
3962 current item.
3963*/
3964
3965/*! \class QMultiHash::key_iterator
3966 \inmodule QtCore
3967 \since 5.6
3968 \brief The QMultiHash::key_iterator class provides an STL-style const iterator for QMultiHash keys.
3969
3970 QMultiHash::key_iterator is essentially the same as QMultiHash::const_iterator
3971 with the difference that operator*() and operator->() return a key
3972 instead of a value.
3973
3974 For most uses QMultiHash::iterator and QMultiHash::const_iterator should be used,
3975 you can easily access the key by calling QMultiHash::iterator::key():
3976
3977 \snippet code/src_corelib_tools_qhash.cpp 27
3978
3979 However, to have interoperability between QMultiHash's keys and STL-style
3980 algorithms we need an iterator that dereferences to a key instead
3981 of a value. With QMultiHash::key_iterator we can apply an algorithm to a
3982 range of keys without having to call QMultiHash::keys(), which is inefficient
3983 as it costs one QMultiHash iteration and memory allocation to create a temporary
3984 QList.
3985
3986 \snippet code/src_corelib_tools_qhash.cpp 28
3987
3988 QMultiHash::key_iterator is const, it's not possible to modify the key.
3989
3990 The default QMultiHash::key_iterator constructor creates an uninitialized
3991 iterator. You must initialize it using a QMultiHash function like
3992 QMultiHash::keyBegin() or QMultiHash::keyEnd().
3993
3994 \warning Iterators on implicitly shared containers do not work
3995 exactly like STL-iterators. You should avoid copying a container
3996 while iterators are active on that container. For more information,
3997 read \l{Implicit sharing iterator problem}.
3998
3999 \sa QMultiHash::const_iterator, QMultiHash::iterator
4000*/
4001
4002/*! \fn template <class Key, class T> const T &QMultiHash<Key, T>::key_iterator::operator*() const
4003
4004 Returns the current item's key.
4005*/
4006
4007/*! \fn template <class Key, class T> const T *QMultiHash<Key, T>::key_iterator::operator->() const
4008
4009 Returns a pointer to the current item's key.
4010*/
4011
4012/*! \fn template <class Key, class T> bool QMultiHash<Key, T>::key_iterator::operator==(key_iterator other) const
4013
4014 Returns \c true if \a other points to the same item as this
4015 iterator; otherwise returns \c false.
4016
4017 \sa operator!=()
4018*/
4019
4020/*! \fn template <class Key, class T> bool QMultiHash<Key, T>::key_iterator::operator!=(key_iterator other) const
4021
4022 Returns \c true if \a other points to a different item than this
4023 iterator; otherwise returns \c false.
4024
4025 \sa operator==()
4026*/
4027
4028/*!
4029 \fn template <class Key, class T> QMultiHash<Key, T>::key_iterator &QMultiHash<Key, T>::key_iterator::operator++()
4030
4031 The prefix ++ operator (\c{++i}) advances the iterator to the
4032 next item in the hash and returns an iterator to the new current
4033 item.
4034
4035 Calling this function on QMultiHash::keyEnd() leads to undefined results.
4036*/
4037
4038/*! \fn template <class Key, class T> QMultiHash<Key, T>::key_iterator QMultiHash<Key, T>::key_iterator::operator++(int)
4039
4040 \overload
4041
4042 The postfix ++ operator (\c{i++}) advances the iterator to the
4043 next item in the hash and returns an iterator to the previous
4044 item.
4045*/
4046
4047/*! \fn template <class Key, class T> const_iterator QMultiHash<Key, T>::key_iterator::base() const
4048 Returns the underlying const_iterator this key_iterator is based on.
4049*/
4050
4051/*! \typedef QMultiHash::const_key_value_iterator
4052 \inmodule QtCore
4053 \since 5.10
4054 \brief The QMultiHash::const_key_value_iterator typedef provides an STL-style const iterator for QMultiHash.
4055
4056 QMultiHash::const_key_value_iterator is essentially the same as QMultiHash::const_iterator
4057 with the difference that operator*() returns a key/value pair instead of a
4058 value.
4059
4060 \sa QKeyValueIterator
4061*/
4062
4063/*! \typedef QMultiHash::key_value_iterator
4064 \inmodule QtCore
4065 \since 5.10
4066 \brief The QMultiHash::key_value_iterator typedef provides an STL-style iterator for QMultiHash.
4067
4068 QMultiHash::key_value_iterator is essentially the same as QMultiHash::iterator
4069 with the difference that operator*() returns a key/value pair instead of a
4070 value.
4071
4072 \sa QKeyValueIterator
4073*/
4074
4075/*! \fn template <class Key, class T> QDataStream &operator<<(QDataStream &out, const QMultiHash<Key, T>& hash)
4076 \relates QMultiHash
4077
4078 Writes the hash \a hash to stream \a out.
4079
4080 This function requires the key and value types to implement \c
4081 operator<<().
4082
4083 \sa {Serializing Qt Data Types}
4084*/
4085
4086/*! \fn template <class Key, class T> QDataStream &operator>>(QDataStream &in, QMultiHash<Key, T> &hash)
4087 \relates QMultiHash
4088
4089 Reads a hash from stream \a in into \a hash.
4090
4091 This function requires the key and value types to implement \c
4092 operator>>().
4093
4094 \sa {Serializing Qt Data Types}
4095*/
4096
4097/*!
4098 \fn template <class Key, class T> size_t qHash(const QHash<Key, T> &key, size_t seed = 0)
4099 \since 5.8
4100 \qhasholdTS{QHash}{Key}{T}
4101*/
4102
4103/*!
4104 \fn template <class Key, class T> size_t qHash(const QMultiHash<Key, T> &key, size_t seed = 0)
4105 \since 5.8
4106 \qhasholdTS{QMultiHash}{Key}{T}
4107*/
4108
4109/*! \fn template <typename Key, typename T, typename Predicate> qsizetype erase_if(QHash<Key, T> &hash, Predicate pred)
4110 \relates QHash
4111 \since 6.1
4112
4113 Removes all elements for which the predicate \a pred returns true
4114 from the hash \a hash.
4115
4116 The function supports predicates which take either an argument of
4117 type \c{QHash<Key, T>::iterator}, or an argument of type
4118 \c{std::pair<const Key &, T &>}.
4119
4120 Returns the number of elements removed, if any.
4121*/
4122
4123/*! \fn template <typename Key, typename T, typename Predicate> qsizetype erase_if(QMultiHash<Key, T> &hash, Predicate pred)
4124 \relates QMultiHash
4125 \since 6.1
4126
4127 Removes all elements for which the predicate \a pred returns true
4128 from the multi hash \a hash.
4129
4130 The function supports predicates which take either an argument of
4131 type \c{QMultiHash<Key, T>::iterator}, or an argument of type
4132 \c{std::pair<const Key &, T &>}.
4133
4134 Returns the number of elements removed, if any.
4135*/
4136
4137/*! \macro QT_NO_SINGLE_ARGUMENT_QHASH_OVERLOAD
4138 \relates QHash
4139 \since 6.11
4140
4141 Defining this macro disables the support for qHash overloads that only take
4142 one argument; in other words, for qHash overloads that do not also accept
4143 a seed. Support for the single-argument overloads of qHash is deprecated
4144 and will be removed in Qt 7.
4145
4146 \sa qHash
4147*/
4148
4149#ifdef QT_HAS_CONSTEXPR_BITOPS
4150namespace QHashPrivate {
4151static_assert(qPopulationCount(SpanConstants::NEntries) == 1,
4152 "NEntries must be a power of 2 for bucketForHash() to work.");
4153
4154// ensure the size of a Span does not depend on the template parameters
4155using Node1 = Node<int, int>;
4156static_assert(sizeof(Span<Node1>) == sizeof(Span<Node<char, void *>>));
4157static_assert(sizeof(Span<Node1>) == sizeof(Span<Node<qsizetype, QHashDummyValue>>));
4158static_assert(sizeof(Span<Node1>) == sizeof(Span<Node<QString, QVariant>>));
4159static_assert(sizeof(Span<Node1>) > SpanConstants::NEntries);
4160static_assert(qNextPowerOfTwo(sizeof(Span<Node1>)) == SpanConstants::NEntries * 2);
4161
4162// ensure allocations are always a power of two, at a minimum NEntries,
4163// obeying the fomula
4164// qNextPowerOfTwo(2 * N);
4165// without overflowing
4166static constexpr size_t NEntries = SpanConstants::NEntries;
4167static_assert(GrowthPolicy::bucketsForCapacity(1) == NEntries);
4168static_assert(GrowthPolicy::bucketsForCapacity(NEntries / 2 + 0) == NEntries);
4169static_assert(GrowthPolicy::bucketsForCapacity(NEntries / 2 + 1) == 2 * NEntries);
4170static_assert(GrowthPolicy::bucketsForCapacity(NEntries * 1 - 1) == 2 * NEntries);
4171static_assert(GrowthPolicy::bucketsForCapacity(NEntries * 1 + 0) == 4 * NEntries);
4172static_assert(GrowthPolicy::bucketsForCapacity(NEntries * 1 + 1) == 4 * NEntries);
4173static_assert(GrowthPolicy::bucketsForCapacity(NEntries * 2 - 1) == 4 * NEntries);
4174static_assert(GrowthPolicy::bucketsForCapacity(NEntries * 2 + 0) == 8 * NEntries);
4175static_assert(GrowthPolicy::bucketsForCapacity(SIZE_MAX / 4) == SIZE_MAX / 2 + 1);
4176static_assert(GrowthPolicy::bucketsForCapacity(SIZE_MAX / 2) == SIZE_MAX);
4177static_assert(GrowthPolicy::bucketsForCapacity(SIZE_MAX) == SIZE_MAX);
4178}
4179#endif
4180
4181QT_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