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