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