日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問(wèn) 生活随笔!

生活随笔

當(dāng)前位置: 首頁(yè) > 编程资源 > 编程问答 >内容正文

编程问答

redis 源码 dict.c 实现

發(fā)布時(shí)間:2023/12/19 编程问答 45 豆豆
生活随笔 收集整理的這篇文章主要介紹了 redis 源码 dict.c 实现 小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

dict.c文件代碼:

#include "fmacros.h"#include <stdio.h> #include <stdlib.h> #include <string.h> #include <stdarg.h> #include <limits.h> #include <sys/time.h> #include <ctype.h>#include "dict.h" #include "zmalloc.h" #include "redisassert.h"/* Using dictEnableResize() / dictDisableResize() we make possible to* enable/disable resizing of the hash table as needed. This is very important* for Redis, as we use copy-on-write and don't want to move too much memory* around when there is a child performing saving operations.** Note that even when dict_can_resize is set to 0, not all resizes are* prevented: a hash table is still allowed to grow if the ratio between* the number of elements and the buckets > dict_force_resize_ratio. */ static int dict_can_resize = 1; static unsigned int dict_force_resize_ratio = 5;/* -------------------------- private prototypes ---------------------------- */static int _dictExpandIfNeeded(dict *ht); static unsigned long _dictNextPower(unsigned long size); static int _dictKeyIndex(dict *ht, const void *key); static int _dictInit(dict *ht, dictType *type, void *privDataPtr);/* -------------------------- hash functions -------------------------------- *//* Thomas Wang's 32 bit Mix Function */ unsigned int dictIntHashFunction(unsigned int key) {key += ~(key << 15);key ^= (key >> 10);key += (key << 3);key ^= (key >> 6);key += ~(key << 11);key ^= (key >> 16);return key; }static uint32_t dict_hash_function_seed = 5381;void dictSetHashFunctionSeed(uint32_t seed) {dict_hash_function_seed = seed; }uint32_t dictGetHashFunctionSeed(void) {return dict_hash_function_seed; }/* MurmurHash2, by Austin Appleby* Note - This code makes a few assumptions about how your machine behaves -* 1. We can read a 4-byte value from any address without crashing* 2. sizeof(int) == 4** And it has a few limitations -** 1. It will not work incrementally.* 2. It will not produce the same results on little-endian and big-endian* machines.*/ unsigned int dictGenHashFunction(const void *key, int len) {/* 'm' and 'r' are mixing constants generated offline.They're not really 'magic', they just happen to work well. */uint32_t seed = dict_hash_function_seed;const uint32_t m = 0x5bd1e995;const int r = 24;/* Initialize the hash to a 'random' value */uint32_t h = seed ^ len;/* Mix 4 bytes at a time into the hash */const unsigned char *data = (const unsigned char *)key;while(len >= 4) {uint32_t k = *(uint32_t*)data;k *= m;k ^= k >> r;k *= m;h *= m;h ^= k;data += 4;len -= 4;}/* Handle the last few bytes of the input array */switch(len) {case 3: h ^= data[2] << 16;case 2: h ^= data[1] << 8;case 1: h ^= data[0]; h *= m;};/* Do a few final mixes of the hash to ensure the last few* bytes are well-incorporated. */h ^= h >> 13;h *= m;h ^= h >> 15;return (unsigned int)h; }/* And a case insensitive hash function (based on djb hash) */ unsigned int dictGenCaseHashFunction(const unsigned char *buf, int len) {unsigned int hash = (unsigned int)dict_hash_function_seed;while (len--)hash = ((hash << 5) + hash) + (tolower(*buf++)); /* hash * 33 + c */return hash; }/* ----------------------------- API implementation ------------------------- *//* Reset a hash table already initialized with ht_init().* NOTE: This function should only be called by ht_destroy(). */ static void _dictReset(dictht *ht) {ht->table = NULL;ht->size = 0;ht->sizemask = 0;ht->used = 0; }/* Create a new hash table */ dict *dictCreate(dictType *type,void *privDataPtr) {dict *d = zmalloc(sizeof(*d));_dictInit(d,type,privDataPtr);return d; }/* Initialize the hash table */ int _dictInit(dict *d, dictType *type,void *privDataPtr) {_dictReset(&d->ht[0]);_dictReset(&d->ht[1]);d->type = type;d->privdata = privDataPtr;d->rehashidx = -1;d->iterators = 0;return DICT_OK; }/* Resize the table to the minimal size that contains all the elements,* but with the invariant of a USED/BUCKETS ratio near to <= 1 */ int dictResize(dict *d) {int minimal;if (!dict_can_resize || dictIsRehashing(d)) return DICT_ERR;minimal = d->ht[0].used;if (minimal < DICT_HT_INITIAL_SIZE)minimal = DICT_HT_INITIAL_SIZE;return dictExpand(d, minimal); }/* Expand or create the hash table */ int dictExpand(dict *d, unsigned long size) {dictht n; /* the new hash table */unsigned long realsize = _dictNextPower(size);/* the size is invalid if it is smaller than the number of* elements already inside the hash table */if (dictIsRehashing(d) || d->ht[0].used > size)return DICT_ERR;/* Allocate the new hash table and initialize all pointers to NULL */n.size = realsize;n.sizemask = realsize-1;n.table = zcalloc(realsize*sizeof(dictEntry*));n.used = 0;/* Is this the first initialization? If so it's not really a rehashing* we just set the first hash table so that it can accept keys. */if (d->ht[0].table == NULL) {d->ht[0] = n;return DICT_OK;}/* Prepare a second hash table for incremental rehashing */d->ht[1] = n;d->rehashidx = 0;return DICT_OK; }/* Performs N steps of incremental rehashing. Returns 1 if there are still* keys to move from the old to the new hash table, otherwise 0 is returned.* Note that a rehashing step consists in moving a bucket (that may have more* than one key as we use chaining) from the old to the new hash table. */ int dictRehash(dict *d, int n) {if (!dictIsRehashing(d)) return 0;while(n--) {dictEntry *de, *nextde;/* Check if we already rehashed the whole table... */if (d->ht[0].used == 0) {zfree(d->ht[0].table);d->ht[0] = d->ht[1];_dictReset(&d->ht[1]);d->rehashidx = -1;return 0;}/* Note that rehashidx can't overflow as we are sure there are more* elements because ht[0].used != 0 */assert(d->ht[0].size > (unsigned long)d->rehashidx);while(d->ht[0].table[d->rehashidx] == NULL) d->rehashidx++;de = d->ht[0].table[d->rehashidx];/* Move all the keys in this bucket from the old to the new hash HT */while(de) {unsigned int h;nextde = de->next;/* Get the index in the new hash table */h = dictHashKey(d, de->key) & d->ht[1].sizemask;de->next = d->ht[1].table[h];d->ht[1].table[h] = de;d->ht[0].used--;d->ht[1].used++;de = nextde;}d->ht[0].table[d->rehashidx] = NULL;d->rehashidx++;}return 1; }long long timeInMilliseconds(void) {struct timeval tv;gettimeofday(&tv,NULL);return (((long long)tv.tv_sec)*1000)+(tv.tv_usec/1000); }/* Rehash for an amount of time between ms milliseconds and ms+1 milliseconds */ int dictRehashMilliseconds(dict *d, int ms) {long long start = timeInMilliseconds();int rehashes = 0;while(dictRehash(d,100)) {rehashes += 100;if (timeInMilliseconds()-start > ms) break;}return rehashes; }/* This function performs just a step of rehashing, and only if there are* no safe iterators bound to our hash table. When we have iterators in the* middle of a rehashing we can't mess with the two hash tables otherwise* some element can be missed or duplicated.** This function is called by common lookup or update operations in the* dictionary so that the hash table automatically migrates from H1 to H2* while it is actively used. */ static void _dictRehashStep(dict *d) {if (d->iterators == 0) dictRehash(d,1); }/* Add an element to the target hash table */ int dictAdd(dict *d, void *key, void *val) {dictEntry *entry = dictAddRaw(d,key);if (!entry) return DICT_ERR;dictSetVal(d, entry, val);return DICT_OK; }/* Low level add. This function adds the entry but instead of setting* a value returns the dictEntry structure to the user, that will make* sure to fill the value field as he wishes.** This function is also directly exposed to user API to be called* mainly in order to store non-pointers inside the hash value, example:** entry = dictAddRaw(dict,mykey);* if (entry != NULL) dictSetSignedIntegerVal(entry,1000);** Return values:** If key already exists NULL is returned.* If key was added, the hash entry is returned to be manipulated by the caller.*/ dictEntry *dictAddRaw(dict *d, void *key) {int index;dictEntry *entry;dictht *ht;if (dictIsRehashing(d)) _dictRehashStep(d);/* Get the index of the new element, or -1 if* the element already exists. */if ((index = _dictKeyIndex(d, key)) == -1)return NULL;/* Allocate the memory and store the new entry */ht = dictIsRehashing(d) ? &d->ht[1] : &d->ht[0];entry = zmalloc(sizeof(*entry));entry->next = ht->table[index];ht->table[index] = entry;ht->used++;/* Set the hash entry fields. */dictSetKey(d, entry, key);return entry; }/* Add an element, discarding the old if the key already exists.* Return 1 if the key was added from scratch, 0 if there was already an* element with such key and dictReplace() just performed a value update* operation. */ int dictReplace(dict *d, void *key, void *val) {dictEntry *entry, auxentry;/* Try to add the element. If the key* does not exists dictAdd will suceed. */if (dictAdd(d, key, val) == DICT_OK)return 1;/* It already exists, get the entry */entry = dictFind(d, key);/* Set the new value and free the old one. Note that it is important* to do that in this order, as the value may just be exactly the same* as the previous one. In this context, think to reference counting,* you want to increment (set), and then decrement (free), and not the* reverse. */auxentry = *entry;dictSetVal(d, entry, val);dictFreeVal(d, &auxentry);return 0; }/* dictReplaceRaw() is simply a version of dictAddRaw() that always* returns the hash entry of the specified key, even if the key already* exists and can't be added (in that case the entry of the already* existing key is returned.)** See dictAddRaw() for more information. */ dictEntry *dictReplaceRaw(dict *d, void *key) {dictEntry *entry = dictFind(d,key);return entry ? entry : dictAddRaw(d,key); }/* Search and remove an element */ static int dictGenericDelete(dict *d, const void *key, int nofree) {unsigned int h, idx;dictEntry *he, *prevHe;int table;if (d->ht[0].size == 0) return DICT_ERR; /* d->ht[0].table is NULL */if (dictIsRehashing(d)) _dictRehashStep(d);h = dictHashKey(d, key);for (table = 0; table <= 1; table++) {idx = h & d->ht[table].sizemask;he = d->ht[table].table[idx];prevHe = NULL;while(he) {if (dictCompareKeys(d, key, he->key)) {/* Unlink the element from the list */if (prevHe)prevHe->next = he->next;elsed->ht[table].table[idx] = he->next;if (!nofree) {dictFreeKey(d, he);dictFreeVal(d, he);}zfree(he);d->ht[table].used--;return DICT_OK;}prevHe = he;he = he->next;}if (!dictIsRehashing(d)) break;}return DICT_ERR; /* not found */ }int dictDelete(dict *ht, const void *key) {return dictGenericDelete(ht,key,0); }int dictDeleteNoFree(dict *ht, const void *key) {return dictGenericDelete(ht,key,1); }/* Destroy an entire dictionary */ int _dictClear(dict *d, dictht *ht, void(callback)(void *)) {unsigned long i;/* Free all the elements */for (i = 0; i < ht->size && ht->used > 0; i++) {dictEntry *he, *nextHe;if (callback && (i & 65535) == 0) callback(d->privdata);if ((he = ht->table[i]) == NULL) continue;while(he) {nextHe = he->next;dictFreeKey(d, he);dictFreeVal(d, he);zfree(he);ht->used--;he = nextHe;}}/* Free the table and the allocated cache structure */zfree(ht->table);/* Re-initialize the table */_dictReset(ht);return DICT_OK; /* never fails */ }/* Clear & Release the hash table */ void dictRelease(dict *d) {_dictClear(d,&d->ht[0],NULL);_dictClear(d,&d->ht[1],NULL);zfree(d); }dictEntry *dictFind(dict *d, const void *key) {dictEntry *he;unsigned int h, idx, table;if (d->ht[0].size == 0) return NULL; /* We don't have a table at all */if (dictIsRehashing(d)) _dictRehashStep(d);h = dictHashKey(d, key);for (table = 0; table <= 1; table++) {idx = h & d->ht[table].sizemask;he = d->ht[table].table[idx];while(he) {if (dictCompareKeys(d, key, he->key))return he;he = he->next;}if (!dictIsRehashing(d)) return NULL;}return NULL; }void *dictFetchValue(dict *d, const void *key) {dictEntry *he;he = dictFind(d,key);return he ? dictGetVal(he) : NULL; }/* A fingerprint is a 64 bit number that represents the state of the dictionary* at a given time, it's just a few dict properties xored together.* When an unsafe iterator is initialized, we get the dict fingerprint, and check* the fingerprint again when the iterator is released.* If the two fingerprints are different it means that the user of the iterator* performed forbidden operations against the dictionary while iterating. */ long long dictFingerprint(dict *d) {long long integers[6], hash = 0;int j;integers[0] = (long) d->ht[0].table;integers[1] = d->ht[0].size;integers[2] = d->ht[0].used;integers[3] = (long) d->ht[1].table;integers[4] = d->ht[1].size;integers[5] = d->ht[1].used;/* We hash N integers by summing every successive integer with the integer* hashing of the previous sum. Basically:** Result = hash(hash(hash(int1)+int2)+int3) ...** This way the same set of integers in a different order will (likely) hash* to a different number. */for (j = 0; j < 6; j++) {hash += integers[j];/* For the hashing step we use Tomas Wang's 64 bit integer hash. */hash = (~hash) + (hash << 21); // hash = (hash << 21) - hash - 1;hash = hash ^ (hash >> 24);hash = (hash + (hash << 3)) + (hash << 8); // hash * 265hash = hash ^ (hash >> 14);hash = (hash + (hash << 2)) + (hash << 4); // hash * 21hash = hash ^ (hash >> 28);hash = hash + (hash << 31);}return hash; }dictIterator *dictGetIterator(dict *d) {dictIterator *iter = zmalloc(sizeof(*iter));iter->d = d;iter->table = 0;iter->index = -1;iter->safe = 0;iter->entry = NULL;iter->nextEntry = NULL;return iter; }dictIterator *dictGetSafeIterator(dict *d) {dictIterator *i = dictGetIterator(d);i->safe = 1;return i; }dictEntry *dictNext(dictIterator *iter) {while (1) {if (iter->entry == NULL) {dictht *ht = &iter->d->ht[iter->table];if (iter->index == -1 && iter->table == 0) {if (iter->safe)iter->d->iterators++;elseiter->fingerprint = dictFingerprint(iter->d);}iter->index++;if (iter->index >= (long) ht->size) {if (dictIsRehashing(iter->d) && iter->table == 0) {iter->table++;iter->index = 0;ht = &iter->d->ht[1];} else {break;}}iter->entry = ht->table[iter->index];} else {iter->entry = iter->nextEntry;}if (iter->entry) {/* We need to save the 'next' here, the iterator user* may delete the entry we are returning. */iter->nextEntry = iter->entry->next;return iter->entry;}}return NULL; }void dictReleaseIterator(dictIterator *iter) {if (!(iter->index == -1 && iter->table == 0)) {if (iter->safe)iter->d->iterators--;elseassert(iter->fingerprint == dictFingerprint(iter->d));}zfree(iter); }/* Return a random entry from the hash table. Useful to* implement randomized algorithms */ dictEntry *dictGetRandomKey(dict *d) {dictEntry *he, *orighe;unsigned int h;int listlen, listele;if (dictSize(d) == 0) return NULL;if (dictIsRehashing(d)) _dictRehashStep(d);if (dictIsRehashing(d)) {do {h = random() % (d->ht[0].size+d->ht[1].size);he = (h >= d->ht[0].size) ? d->ht[1].table[h - d->ht[0].size] :d->ht[0].table[h];} while(he == NULL);} else {do {h = random() & d->ht[0].sizemask;he = d->ht[0].table[h];} while(he == NULL);}/* Now we found a non empty bucket, but it is a linked* list and we need to get a random element from the list.* The only sane way to do so is counting the elements and* select a random index. */listlen = 0;orighe = he;while(he) {he = he->next;listlen++;}listele = random() % listlen;he = orighe;while(listele--) he = he->next;return he; }/* Function to reverse bits. Algorithm from:* http://graphics.stanford.edu/~seander/bithacks.html#ReverseParallel */ static unsigned long rev(unsigned long v) {unsigned long s = 8 * sizeof(v); // bit size; must be power of 2unsigned long mask = ~0;while ((s >>= 1) > 0) {mask ^= (mask << s);v = ((v >> s) & mask) | ((v << s) & ~mask);}return v; }/* dictScan() is used to iterate over the elements of a dictionary.** Iterating works the following way:** 1) Initially you call the function using a cursor (v) value of 0.* 2) The function performs one step of the iteration, and returns the* new cursor value you must use in the next call.* 3) When the returned cursor is 0, the iteration is complete.** The function guarantees all elements present in the* dictionary get returned between the start and end of the iteration.* However it is possible some elements get returned multiple times.** For every element returned, the callback argument 'fn' is* called with 'privdata' as first argument and the dictionary entry* 'de' as second argument.** HOW IT WORKS.** The iteration algorithm was designed by Pieter Noordhuis.* The main idea is to increment a cursor starting from the higher order* bits. That is, instead of incrementing the cursor normally, the bits* of the cursor are reversed, then the cursor is incremented, and finally* the bits are reversed again.** This strategy is needed because the hash table may be resized between* iteration calls.** dict.c hash tables are always power of two in size, and they* use chaining, so the position of an element in a given table is given* by computing the bitwise AND between Hash(key) and SIZE-1* (where SIZE-1 is always the mask that is equivalent to taking the rest* of the division between the Hash of the key and SIZE).** For example if the current hash table size is 16, the mask is* (in binary) 1111. The position of a key in the hash table will always be* the last four bits of the hash output, and so forth.** WHAT HAPPENS IF THE TABLE CHANGES IN SIZE?** If the hash table grows, elements can go anywhere in one multiple of* the old bucket: for example let's say we already iterated with* a 4 bit cursor 1100 (the mask is 1111 because hash table size = 16).** If the hash table will be resized to 64 elements, then the new mask will* be 111111. The new buckets you obtain by substituting in ??1100* with either 0 or 1 can be targeted only by keys we already visited* when scanning the bucket 1100 in the smaller hash table.** By iterating the higher bits first, because of the inverted counter, the* cursor does not need to restart if the table size gets bigger. It will* continue iterating using cursors without '1100' at the end, and also* without any other combination of the final 4 bits already explored.** Similarly when the table size shrinks over time, for example going from* 16 to 8, if a combination of the lower three bits (the mask for size 8* is 111) were already completely explored, it would not be visited again* because we are sure we tried, for example, both 0111 and 1111 (all the* variations of the higher bit) so we don't need to test it again.** WAIT... YOU HAVE *TWO* TABLES DURING REHASHING!** Yes, this is true, but we always iterate the smaller table first, then* we test all the expansions of the current cursor into the larger* table. For example if the current cursor is 101 and we also have a* larger table of size 16, we also test (0)101 and (1)101 inside the larger* table. This reduces the problem back to having only one table, where* the larger one, if it exists, is just an expansion of the smaller one.** LIMITATIONS** This iterator is completely stateless, and this is a huge advantage,* including no additional memory used.** The disadvantages resulting from this design are:** 1) It is possible we return elements more than once. However this is usually* easy to deal with in the application level.* 2) The iterator must return multiple elements per call, as it needs to always* return all the keys chained in a given bucket, and all the expansions, so* we are sure we don't miss keys moving during rehashing.* 3) The reverse cursor is somewhat hard to understand at first, but this* comment is supposed to help.*/ unsigned long dictScan(dict *d,unsigned long v,dictScanFunction *fn,void *privdata) {dictht *t0, *t1;const dictEntry *de;unsigned long m0, m1;if (dictSize(d) == 0) return 0;if (!dictIsRehashing(d)) {t0 = &(d->ht[0]);m0 = t0->sizemask;/* Emit entries at cursor */de = t0->table[v & m0];while (de) {fn(privdata, de);de = de->next;}} else {t0 = &d->ht[0];t1 = &d->ht[1];/* Make sure t0 is the smaller and t1 is the bigger table */if (t0->size > t1->size) {t0 = &d->ht[1];t1 = &d->ht[0];}m0 = t0->sizemask;m1 = t1->sizemask;/* Emit entries at cursor */de = t0->table[v & m0];while (de) {fn(privdata, de);de = de->next;}/* Iterate over indices in larger table that are the expansion* of the index pointed to by the cursor in the smaller table */do {/* Emit entries at cursor */de = t1->table[v & m1];while (de) {fn(privdata, de);de = de->next;}/* Increment bits not covered by the smaller mask */v = (((v | m0) + 1) & ~m0) | (v & m0);/* Continue while bits covered by mask difference is non-zero */} while (v & (m0 ^ m1));}/* Set unmasked bits so incrementing the reversed cursor* operates on the masked bits of the smaller table */v |= ~m0;/* Increment the reverse cursor */v = rev(v);v++;v = rev(v);return v; }/* ------------------------- private functions ------------------------------ *//* Expand the hash table if needed */ static int _dictExpandIfNeeded(dict *d) {/* Incremental rehashing already in progress. Return. */if (dictIsRehashing(d)) return DICT_OK;/* If the hash table is empty expand it to the initial size. */if (d->ht[0].size == 0) return dictExpand(d, DICT_HT_INITIAL_SIZE);/* If we reached the 1:1 ratio, and we are allowed to resize the hash* table (global setting) or we should avoid it but the ratio between* elements/buckets is over the "safe" threshold, we resize doubling* the number of buckets. */if (d->ht[0].used >= d->ht[0].size &&(dict_can_resize ||d->ht[0].used/d->ht[0].size > dict_force_resize_ratio)){return dictExpand(d, d->ht[0].used*2);}return DICT_OK; }/* Our hash table capability is a power of two */ static unsigned long _dictNextPower(unsigned long size) {unsigned long i = DICT_HT_INITIAL_SIZE;if (size >= LONG_MAX) return LONG_MAX;while(1) {if (i >= size)return i;i *= 2;} }/* Returns the index of a free slot that can be populated with* a hash entry for the given 'key'.* If the key already exists, -1 is returned.** Note that if we are in the process of rehashing the hash table, the* index is always returned in the context of the second (new) hash table. */ static int _dictKeyIndex(dict *d, const void *key) {unsigned int h, idx, table;dictEntry *he;/* Expand the hash table if needed */if (_dictExpandIfNeeded(d) == DICT_ERR)return -1;/* Compute the key hash value */h = dictHashKey(d, key);for (table = 0; table <= 1; table++) {idx = h & d->ht[table].sizemask;/* Search if this slot does not already contain the given key */he = d->ht[table].table[idx];while(he) {if (dictCompareKeys(d, key, he->key))return -1;he = he->next;}if (!dictIsRehashing(d)) break;}return idx; }void dictEmpty(dict *d, void(callback)(void*)) {_dictClear(d,&d->ht[0],callback);_dictClear(d,&d->ht[1],callback);d->rehashidx = -1;d->iterators = 0; }void dictEnableResize(void) {dict_can_resize = 1; }void dictDisableResize(void) {dict_can_resize = 0; }#if 0/* The following is code that we don't use for Redis currently, but that is part of the library. *//* ----------------------- Debugging ------------------------*/#define DICT_STATS_VECTLEN 50 static void _dictPrintStatsHt(dictht *ht) {unsigned long i, slots = 0, chainlen, maxchainlen = 0;unsigned long totchainlen = 0;unsigned long clvector[DICT_STATS_VECTLEN];if (ht->used == 0) {printf("No stats available for empty dictionaries\n");return;}for (i = 0; i < DICT_STATS_VECTLEN; i++) clvector[i] = 0;for (i = 0; i < ht->size; i++) {dictEntry *he;if (ht->table[i] == NULL) {clvector[0]++;continue;}slots++;/* For each hash entry on this slot... */chainlen = 0;he = ht->table[i];while(he) {chainlen++;he = he->next;}clvector[(chainlen < DICT_STATS_VECTLEN) ? chainlen : (DICT_STATS_VECTLEN-1)]++;if (chainlen > maxchainlen) maxchainlen = chainlen;totchainlen += chainlen;}printf("Hash table stats:\n");printf(" table size: %ld\n", ht->size);printf(" number of elements: %ld\n", ht->used);printf(" different slots: %ld\n", slots);printf(" max chain length: %ld\n", maxchainlen);printf(" avg chain length (counted): %.02f\n", (float)totchainlen/slots);printf(" avg chain length (computed): %.02f\n", (float)ht->used/slots);printf(" Chain length distribution:\n");for (i = 0; i < DICT_STATS_VECTLEN-1; i++) {if (clvector[i] == 0) continue;printf(" %s%ld: %ld (%.02f%%)\n",(i == DICT_STATS_VECTLEN-1)?">= ":"", i, clvector[i], ((float)clvector[i]/ht->size)*100);} }void dictPrintStats(dict *d) {_dictPrintStatsHt(&d->ht[0]);if (dictIsRehashing(d)) {printf("-- Rehashing into ht[1]:\n");_dictPrintStatsHt(&d->ht[1]);} }/* ----------------------- StringCopy Hash Table Type ------------------------*/static unsigned int _dictStringCopyHTHashFunction(const void *key) {return dictGenHashFunction(key, strlen(key)); }static void *_dictStringDup(void *privdata, const void *key) {int len = strlen(key);char *copy = zmalloc(len+1);DICT_NOTUSED(privdata);memcpy(copy, key, len);copy[len] = '\0';return copy; }static int _dictStringCopyHTKeyCompare(void *privdata, const void *key1,const void *key2) {DICT_NOTUSED(privdata);return strcmp(key1, key2) == 0; }static void _dictStringDestructor(void *privdata, void *key) {DICT_NOTUSED(privdata);zfree(key); }dictType dictTypeHeapStringCopyKey = {_dictStringCopyHTHashFunction, /* hash function */_dictStringDup, /* key dup */NULL, /* val dup */_dictStringCopyHTKeyCompare, /* key compare */_dictStringDestructor, /* key destructor */NULL /* val destructor */ };/* This is like StringCopy but does not auto-duplicate the key.* It's used for intepreter's shared strings. */ dictType dictTypeHeapStrings = {_dictStringCopyHTHashFunction, /* hash function */NULL, /* key dup */NULL, /* val dup */_dictStringCopyHTKeyCompare, /* key compare */_dictStringDestructor, /* key destructor */NULL /* val destructor */ };/* This is like StringCopy but also automatically handle dynamic* allocated C strings as values. */ dictType dictTypeHeapStringCopyKeyValue = {_dictStringCopyHTHashFunction, /* hash function */_dictStringDup, /* key dup */_dictStringDup, /* val dup */_dictStringCopyHTKeyCompare, /* key compare */_dictStringDestructor, /* key destructor */_dictStringDestructor, /* val destructor */ }; #endif

創(chuàng)作挑戰(zhàn)賽新人創(chuàng)作獎(jiǎng)勵(lì)來(lái)咯,堅(jiān)持創(chuàng)作打卡瓜分現(xiàn)金大獎(jiǎng)

總結(jié)

以上是生活随笔為你收集整理的redis 源码 dict.c 实现的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。

如果覺(jué)得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。

亚洲免费在线观看视频 | 国内视频在线 | 激情欧美一区二区三区 | 日韩视频一区二区在线观看 | 欧美日韩国产综合一区二区 | 欧美精品在线观看一区 | 色综合久久精品 | 黄色小说免费观看 | 国产精久久久久久妇女av | 免费在线观看视频a | 日本性xxxxx| 久久福利综合 | 在线黄色国产 | 中文字幕资源网 | 欧美一级免费高清 | 亚洲欧美日韩中文在线 | 91精品国产乱码 | 五月综合激情婷婷 | 日韩精品一区二区三区高清免费 | 婷婷五月在线视频 | 久久久综合 | 久久久综合电影 | 亚洲精品99久久久久中文字幕 | 久久黄色片子 | 日韩精品久久一区二区三区 | 玖玖在线视频观看 | 天天操天天谢 | 欧美国产日韩一区 | 丁香婷婷激情国产高清秒播 | 狠狠色婷婷丁香六月 | 国产三级精品在线 | 青草视频在线看 | 日狠狠| 日韩av进入 | 91观看视频| 天天搞夜夜骑 | 精品久久一二三区 | 精品欧美在线视频 | 国产一区视频导航 | 成人免费91 | 久久免费黄色大片 | 奇米影视777影音先锋 | 久久久久成人精品 | 一区二区精品视频 | 久久综合之合合综合久久 | 国产资源免费 | 亚洲国产精品传媒在线观看 | 久在线观看| 亚洲资源在线观看 | 又黄又爽又刺激的视频 | 久久国产精品99国产 | 久久久久综合精品福利啪啪 | 92av视频 | av成人在线电影 | 99看视频在线观看 | 99精品在线免费 | 日日夜夜狠狠操 | 人人干天天射 | 久草热视频 | 国产精品一区二区久久国产 | 精品日本视频 | 久久久久国产成人免费精品免费 | 黄a在线观看 | 一区二区三区在线电影 | 亚洲天天做 | 中文字幕网址 | 精品久久久国产 | 免费十分钟 | av在线免费网| 国产欧美日韩一区 | 欧美激情视频一区二区三区 | 国产亚洲精品久久久久久大师 | 青青河边草免费观看 | 干狠狠| 国产区精品在线观看 | 久草在线观 | 激情欧美一区二区三区免费看 | 久久久国产精品亚洲一区 | 国产精品 中文在线 | 欧美极品一区二区三区 | 黄色毛片视频免费 | 国产色一区 | 久久天天躁夜夜躁狠狠躁2022 | 麻豆激情电影 | 国产无套视频 | 成人aⅴ视频 | 日日干天天 | 欧美性色综合网 | 欧美日韩精品免费观看视频 | av免费网站观看 | 蜜臀久久99精品久久久酒店新书 | 亚洲精品视频二区 | 久久爱资源网 | 久久不射电影院 | 欧美一级电影免费观看 | a视频在线看 | 激情久久久 | 日韩亚洲在线视频 | 综合国产在线 | 欧美人人爱 | 五月综合在线观看 | 久久精品8 | av电影中文字幕在线观看 | 久草在线视频首页 | 国产人成在线视频 | www.成人精品 | 99久久爱 | 日本三级中文字幕在线观看 | 精品国产乱码一区二区三区在线 | 欧美精品一区二区在线播放 | 免费看黄网站在线 | 热99在线| 国产一区二区精品久久91 | 最新av中文字幕 | 精品国产一区二区三区不卡 | 日本丰满少妇免费一区 | 免费看污网站 | 片网站 | 久久久精品国产一区二区 | 欧美夫妻生活视频 | 日韩免费成人av | 国产资源免费在线观看 | www.夜夜骑.com | 日本黄色一级电影 | 天天色综合天天 | www一起操| 国产超碰在线观看 | 96超碰在线 | av在线进入 | 黄色精品一区 | 色狠狠综合天天综合综合 | 在线视频欧美亚洲 | 久热超碰 | 日本字幕网 | 在线中文字幕网站 | 中文乱码视频在线观看 | 免费av小说| 国产免费人人看 | 色综合国产 | 性色av免费在线观看 | 久久经典国产视频 | 中文字幕免费在线看 | 久久99欧美| 久久久wwww| 久久久久久久久久久久久久电影 | 国产自在线观看 | 成人黄色在线观看视频 | 国产一区欧美一区 | 人人干干人人 | 麻豆国产精品一区二区三区 | 日韩在线视 | 国产精品av在线免费观看 | 国产剧情av在线播放 | 国产精品久久久久婷婷二区次 | 精品99在线视频 | 91麻豆精品国产自产在线游戏 | 日本韩国欧美在线观看 | 亚洲精品乱码久久久久v最新版 | 三级在线视频观看 | 色综合色综合久久综合频道88 | 国产精品久久久久久69 | 成人app在线免费观看 | 天天色欧美 | 超碰在线日本 | 黄色一级大片在线免费看国产一 | 国产在线观看二区 | 久久精品一区二区三区国产主播 | 一区二区精品视频 | 九九九热精品免费视频观看 | 久久96 | 欧美天天综合网 | 97超视频免费观看 | 久久精品国产99国产 | 狠狠精品| 精品国产色 | 丁香花在线观看免费完整版视频 | 99riav1国产精品视频 | 国产精品毛片久久久久久 | 久久久成人精品 | 欧美另类xxxx | 久草香蕉在线视频 | 国产激情久久久 | 在线免费观看成人 | 亚洲成人精品国产 | 欧美一级视频在线观看 | 久久免费精品视频 | 在线亚洲高清视频 | 免费网站看av片 | 国产一级免费电影 | 91丨九色丨丝袜 | 视频在线一区 | 欧美成人在线免费观看 | 九九在线精品视频 | 五月天伊人 | 香蕉视频久久 | 国产精品一区二区中文字幕 | 99视频播放| 色香蕉在线视频 | 欧美成人按摩 | 成年人黄色av | 国产成年免费视频 | 麻豆影音先锋 | 69夜色精品国产69乱 | 欧美日韩视频一区二区三区 | 国内丰满少妇猛烈精品播 | 久久精品久久久久 | 亚洲精品在线视频网站 | 日韩免费 | 久久成人资源 | 日韩色爱| 在线 成人 | 亚洲最新视频在线播放 | 天天艹天天爽 | 日韩电影一区二区三区 | 亚洲男男gⅴgay双龙 | 91九色porn在线资源 | 久久久91精品国产一区二区三区 | 中文字幕av有码 | a久久久久| 欧美在线free | 久久优 | 亚洲国产日韩欧美在线 | 色综合久久综合 | 日本中文字幕一二区观 | 久久久免费看片 | 久久伊人精品天天 | 深夜免费福利视频 | 97精品国产97久久久久久春色 | 天堂v中文| 久久精品国产精品亚洲 | 亚洲欧洲精品在线 | 一本一本久久a久久精品综合 | 国产色婷婷精品综合在线手机播放 | 色婷婷在线视频 | 国产视频第二页 | 亚洲五月激情 | 成年人网站免费在线观看 | 国产直播av | 国产成人一级电影 | 久久国产精品99久久久久久进口 | 亚洲理论视频 | 国产午夜在线观看 | 一区二区观看 | 三级av在线免费观看 | 超碰在线9 | 亚洲国产激情 | 欧美成人影音 | 久久99精品国产麻豆宅宅 | 国产精品视频在线观看 | 毛片网站免费 | 精品国产成人av在线免 | 91久久精品日日躁夜夜躁国产 | 综合网欧美 | 丁香六月天婷婷 | 日韩欧美网址 | 成人免费网站视频 | 久草免费新视频 | 日本福利视频在线 | 欧美日韩中文字幕在线视频 | 91人人爽人人爽人人精88v | 中文字幕123区 | 91在线精品播放 | 九九九九九九精品任你躁 | 五月婷婷在线观看 | 国产xxxxx在线观看 | 日本中文字幕在线看 | 国产资源免费在线观看 | 国产韩国精品一区二区三区 | 一级欧美日韩 | 91天天操| 久久夜av | 91精品亚洲影视在线观看 | 国产一二区视频 | 日韩精品久久久免费观看夜色 | 在线观看黄色av | 亚洲精品美女在线观看播放 | 色小说av| 色婷婷国产精品一区在线观看 | 97福利| 欧美日韩一区二区在线观看 | 精品国产乱码久久久久久浪潮 | 欧美性色综合网站 | 黄色录像av | 91爱爱视频 | 日本天天操 | 日本资源中文字幕在线 | 久久精品视频播放 | 国产亚洲精品久久久久久无几年桃 | 91爱爱网址| 国产精品久久电影观看 | 四虎最新入口 | 美女精品 | 日韩毛片在线免费观看 | 国产剧情一区在线 | 超碰在线最新网址 | 欧美日韩视频在线 | 日韩电影在线一区 | 国产精品久久久久免费观看 | 黄色一级大片在线免费看国产一 | av综合av | 在线免费黄网站 | 国产精品网红直播 | 国产精品原创视频 | 久久国产精品久久精品国产演员表 | 一级黄色a视频 | 国产精品久久久久久影院 | 一级黄色免费 | 成 人 黄 色 视频 免费观看 | 成人精品在线 | 国产精品久久久久久久久久 | 久久久久久久久久久黄色 | 色片网站在线观看 | 成人免费在线视频观看 | 一级片视频免费观看 | 亚洲一区二区三区四区精品 | 91九色蝌蚪| 日韩一区二区免费视频 | 中文字幕日韩av | 国产欧美精品xxxx另类 | 亚洲香蕉在线观看 | 四虎永久网站 | 国产午夜麻豆影院在线观看 | 久久久久国产一区二区三区四区 | 国产精品va最新国产精品视频 | 国产一区二区三区高清播放 | 在线电影 一区 | 人人插人人澡 | 在线视频精品播放 | 在线视频你懂得 | 婷婷久久久 | 97在线观看视频 | 在线久久 | 久草资源免费 | 黄色小网站在线 | 国产精品免费麻豆入口 | 国产小视频在线 | 日韩理论电影网 | 91av资源在线 | 日韩精品2区 | 成人久久久久久久久久 | 精品美女在线视频 | 亚洲精品美女久久17c | 国产黄色精品在线观看 | 免费福利在线 | 国产五月 | 久艹视频在线免费观看 | 91色视频| 欧美一区二区在线 | 一区二区三区在线视频111 | 狠狠狠色狠狠色综合 | 中文字幕在线高清 | 亚洲激情一区二区三区 | 国产欧美最新羞羞视频在线观看 | 久久久久欧美精品999 | 亚洲视频第一页 | 久99久在线视频 | 久久人人爽人人爽人人片 | 国产99久久99热这里精品5 | 日本午夜在线亚洲.国产 | 天干啦夜天干天干在线线 | 伊人国产女 | 久久久久久网址 | 国产一级精品在线观看 | 九九热有精品 | 在线视频app | 日韩高清在线看 | 免费看黄在线看 | 香蕉一区 | 国产精品12 | 在线性视频日韩欧美 | 91观看视频 | 精品国产1区2区3区 国产欧美精品在线观看 | 久久不射网站 | 日韩女同一区二区三区在线观看 | 色五月色开心色婷婷色丁香 | 国产精品高清免费在线观看 | 天天草天天| 日韩 在线| 成人免费网站视频 | 奇米影视四色8888 | 免费视频xnxx com | 麻豆影视在线播放 | 999一区二区三区 | 久久久亚洲麻豆日韩精品一区三区 | 91九色网站 | 日本公妇在线观看 | 国产精品不卡一区 | 国产精品自在线拍国产 | 中文在线免费看视频 | a在线播放 | 国产一区二区三区网站 | 国产亚洲情侣一区二区无 | 亚洲一区二区三区在线看 | 亚洲国内精品视频 | 手机色在线 | 久久综合桃花 | 国产在线国产 | 麻豆 91 在线 | 色综合久久中文综合久久牛 | 日韩电影精品一区 | 久久久久免费看 | 国产精品美乳一区二区免费 | 国产精品乱码久久 | 久久久国产精品成人免费 | 欧美一区二区精品在线 | 日韩综合视频在线观看 | 91麻豆精品国产自产在线游戏 | 色成人亚洲 | 九九视频这里只有精品 | 国产粉嫩在线 | 日韩成片 | 久久久久久久久久网 | 久久嗨 | 国产精品毛片一区 | 欧美韩国在线 | a爱爱视频 | 在线色亚洲 | 丝袜网站在线观看 | 人人爽人人爽人人爽人人爽 | bbbbb女女女女女bbbbb国产 | 中文字幕国语官网在线视频 | 在线电影 一区 | 天天草综合 | 亚州人成在线播放 | 制服丝袜天堂 | 天天射天天爱天天干 | 色综合久久久久网 | 日韩欧美一级二级 | 国产vs久久 | 午夜黄色一级片 | 黄色一级在线免费观看 | 国产婷婷一区二区 | 久久国产免费看 | 免费a v观看 | 免费a级黄色毛片 | 国产精品视频你懂的 | 夜夜狠狠 | 欧美一区二区在线 | 超碰在线免费97 | 欧美少妇xx | 成人免费在线网 | 91视频中文字幕 | 久久在线免费观看 | 久久天天躁狠狠躁夜夜不卡公司 | 精品日韩视频 | 人人添人人澡 | 在线成人免费电影 | 久久一区二区三区超碰国产精品 | 最新av免费在线观看 | 欧美精品久久久久久久久久 | 国产淫a| 四虎国产永久在线精品 | 蜜桃视频在线观看一区 | 欧美一级片免费观看 | 国产成人精品女人久久久 | 久草在线在线精品观看 | 精品女同一区二区三区在线观看 | 久久tv视频 | 欧美一区二区三区免费观看 | 久久精品国产成人 | 国产亲近乱来精品 | 尤物九九久久国产精品的分类 | 精品免费99久久 | 亚洲欧洲国产精品 | 国产精品不卡 | 成人a在线观看高清电影 | 日本中文乱码卡一卡二新区 | 91人人视频在线观看 | 天天射综合网视频 | 中文字幕在线观看一区二区三区 | 欧美专区国产专区 | 久久免费看a级毛毛片 | 日韩欧美电影在线观看 | 中文字幕乱码视频 | 久久国产精品一国产精品 | 亚洲色图激情文学 | 久久久久久久久久久久久9999 | 97超碰免费在线 | 精品二区视频 | 美女黄频在线观看 | 91av影视 | 中文字幕在线观看不卡 | 欧美疯狂性受xxxxx另类 | 99久久久国产精品免费99 | 免费亚洲电影 | 在线国产一区二区 | 女人18毛片a级毛片一区二区 | 2019免费中文字幕 | 手机在线看a | 奇米网444 | 婷婷av网 | 欧美日韩一区二区三区在线观看视频 | 成人影片免费 | 久久精品国产免费看久久精品 | 蜜桃视频在线观看一区 | 亚洲国产精品女人久久久 | 成人av电影免费在线播放 | 免费h在线观看 | 国产精品麻豆果冻传媒在线播放 | 精品欧美在线视频 | 久久久麻豆视频 | 国产97色| 在线观看91视频 | 27xxoo无遮挡动态视频 | 国产永久网站 | 欧洲一区二区在线观看 | 美女久久精品 | 亚洲三级在线播放 | 天天天天色射综合 | 中文十次啦 | 91高清免费看 | 欧美日韩精品在线观看视频 | 我要色综合天天 | 国产区精品在线观看 | 五月婷婷在线观看视频 | 亚洲最大免费成人网 | av大全在线免费观看 | 日本韩国精品一区二区在线观看 | 992tv在线观看 | 91一区二区三区久久久久国产乱 | 在线观看黄色免费视频 | 2024av在线播放| 成人黄色大片在线免费观看 | 国产精品免费视频一区二区 | 98涩涩国产露脸精品国产网 | www.成人精品| 天天操天天操天天操天天操天天操 | 色婷婷久久久综合中文字幕 | 超碰在线网 | 91探花国产综合在线精品 | 色综合久久五月 | a级国产乱理伦片在线观看 亚洲3级 | 国产第一页精品 | 国产黄色在线观看 | 久久久久久高潮国产精品视 | 国产高清第一页 | 国产精品自产拍在线观看中文 | 三级av免费看 | 亚洲免费在线观看视频 | 中文字幕一区二区在线播放 | 久久久久久久久黄色 | 久在线| 国产免费专区 | 久久精品视频国产 | 美国三级黄色大片 | 深爱综合网 | 亚洲第一久久久 | 中文资源在线播放 | 91麻豆免费看 | 亚洲另类视频 | 啪一啪在线 | 国产精品 视频 | 97精品在线 | 99久久精品国产免费看不卡 | 91精品伦理 | 国产精品观看在线亚洲人成网 | 国产在线视频一区 | 97色涩| 高清一区二区三区 | 99久久99久久精品国产片果冰 | 日韩网站在线看片你懂的 | 在线观看av片 | 二区三区视频 | 日本xxxxav | 97av视频在线观看 | 久久色网站 | 久久一区二区三区国产精品 | 国产精品久久久区三区天天噜 | 激情网综合 | 久久99久久99精品免观看粉嫩 | 99视频这里有精品 | 最新成人av | 中文字幕在线播放av | 五月天婷婷在线播放 | 好看av在线 | 国产资源在线视频 | 午夜精品一二三区 | 亚州视频在线 | 日本久久精品 | 91九色网站 | 国产精品成人一区二区 | 久久99精品久久只有精品 | 久久草网 | 久久久男人的天堂 | 中文字幕在线播出 | 国产专区精品 | 午夜av免费| 综合久久婷婷 | 五月天婷婷视频 | 欧美日韩大片在线观看 | www.91国产| 色中文字幕在线观看 | 成人国产精品av | 亚洲激情在线观看 | 九九国产视频 | 日韩av中文字幕在线免费观看 | 三级黄色理论片 | 国产精品成人一区二区三区吃奶 | 精品视频免费 | 伊人影院在线观看 | 久久伊人操 | 天天想夜夜操 | 97操操操 | 天天操夜夜看 | 国产精品爽爽久久久久久蜜臀 | 激情综合色播五月 | 国产一级片观看 | 夜夜躁日日躁狠狠躁 | 97av影院 | 久久久精品免费观看 | 天天爱综合| 亚洲黄色在线免费观看 | 国产 日韩 在线 亚洲 字幕 中文 | 亚洲国产日韩精品 | 天天综合精品 | 日本精品视频在线 | 久草免费在线视频 | 亚洲精品玖玖玖av在线看 | 亚洲爱爱视频 | 欧美精品免费在线 | a天堂一码二码专区 | 高清中文字幕 | 日韩免费电影 | 五月天.com| 国产成人综合在线观看 | japanese黑人亚洲人4k | 果冻av在线| 天天草天天摸 | 国产精品一区一区三区 | 人人干人人草 | 青春草国产视频 | av中文字幕在线播放 | 深爱婷婷久久综合 | 三级视频日韩 | 人人爽人人片 | 成年人在线| 99久久99久久精品免费 | 日韩精品中文字幕在线播放 | 久亚洲 | 免费在线观看黄网站 | 国产最新在线 | 人人添人人澡人人澡人人人爽 | 国产一区二区三区午夜 | 久久亚洲欧美 | 国产在线p | 五月婷婷av| 狠狠色噜噜狠狠 | 99色在线视频 | 亚洲 欧美 日韩 综合 | 99亚洲国产 | 91精品夜夜| 免费av一级电影 | 久久免费视频4 | 特级西西www44高清大胆图片 | 91精品网站| 福利片视频区 | 欧美怡红院 | 亚洲aⅴ乱码精品成人区 | 久草在线视频网站 | 国产精久久 | 久久久久二区 | 久久久久久毛片精品免费不卡 | 成人国产精品久久久 | 久久久国产精品免费 | 91精品系列 | 97国产大学生情侣白嫩酒店 | 天天干天天操人体 | 精品国产区在线 | 激情开心 | 亚洲国产精品久久 | 亚洲国产mv| 欧美一区成人 | 中文字幕影视 | 日韩字幕 | 日韩精品一区二区在线观看视频 | 久久综合精品一区 | 成人免费在线视频 | 香蕉精品视频在线观看 | 午夜国产福利在线 | 黄色免费在线看 | 亚洲精品乱码白浆高清久久久久久 | 天天操天天摸天天干 | 91日韩精品一区 | 日本一区二区免费在线观看 | 青青久草在线视频 | 亚洲欧美综合精品久久成人 | 久久se视频 | 亚洲国产精久久久久久久 | 久久国产精品99久久久久久丝袜 | 美女视频免费精品 | 亚洲丁香日韩 | 亚洲丝袜一区二区 | 久久人人爽人人爽人人片av免费 | 日韩高清久久 | 国产精品成人品 | 成人午夜电影在线 | 日韩成人免费在线观看 | 少妇精69xxtheporn | 亚洲国产视频a | 一本—道久久a久久精品蜜桃 | 中文在线免费视频 | 亚洲 综合 专区 | 国产中文字幕视频在线观看 | 人人舔人人 | 99久久精品免费看国产免费软件 | 一区二区激情 | 国产在线播放一区二区三区 | 97视频成人| 国产在线国产 | 91在线小视频 | 精品在线你懂的 | 国精产品999国精产品视频 | 丁香六月激情 | 96精品视频 | 日本3级在线观看 | 日日干天天操 | 色婷婷av国产精品 | 久久国产精品久久精品 | 99热最新精品 | 中文字幕在线日本 | 国产在线不卡一区 | 亚洲精品理论片 | wwwwww黄 | 91桃色视频| 亚洲欧洲视频 | 中文字幕a∨在线乱码免费看 | 久久精品一区 | 伊人久久影视 | 天天视频亚洲 | 有码视频在线观看 | 欧美综合久久 | 久久av免费电影 | 香蕉视频在线网站 | 亚洲午夜精品久久久久久久久久久久 | 日韩av伦理片 | 国产成人久久精品77777综合 | 亚洲国产美女久久久久 | 久久久久久看片 | 成人午夜电影网 | 美女视频黄的免费的 | 日韩精品一区二区在线观看 | 黄色免费网战 | 日本韩国欧美在线观看 | 欧美性免费 | 亚州国产视频 | 一级黄色免费 | 九九久久影视 | 中文字幕av有码 | 激情视频免费观看 | 国产专区日韩专区 | 亚洲精品在线视频播放 | 中文有码在线视频 | 手机在线欧美 | 一区 在线 影院 | 精品国产欧美一区二区三区不卡 | 精品国产视频一区 | 精品国产一区二区三区不卡 | 国产精品国产三级国产 | 男女激情免费网站 | 五月天色婷婷丁香 | 日韩精品一区二区三区免费观看 | 日本免费一二三区 | 亚洲不卡123 | 最近中文字幕免费av | 黄色小说免费在线观看 | 片网站| 色吧av色av| 黄色免费网 | 黄色一级免费网站 | 五月婷在线 | 日本精品小视频 | 欧美精品成人在线 | 国产精品久久久网站 | 亚洲成av人片在线观看香蕉 | 国产伦理精品一区二区 | 免费三级影片 | 久久不卡国产精品一区二区 | 国内精自线一二区永久 | 国产精品久久久久久妇 | 亚洲视频axxx | 中文字幕免费播放 | 欧美精品久久久久久久免费 | 亚洲狠狠丁香婷婷综合久久久 | 99久久毛片 | 国产精品一区二区久久久 | 九九九九九精品 | 五月婷在线 | 99c视频在线 | 久草在线视频新 | 婷婷综合视频 | 天天草天天色 | 精品欧美小视频在线观看 | 五月婷婷丁香激情 | 免费观看黄色12片一级视频 | 国产精品三级视频 | 久久夜色精品国产欧美乱极品 | aaaaaa毛片| 夜夜躁狠狠躁日日躁视频黑人 | 亚洲一区二区精品视频 | 国产一区二区三区免费观看视频 | www日韩在线 | 日韩电影一区二区在线观看 | 国产精品久久久久久久久久久久午 | 香蕉91视频 | 精品国产一区二区三区久久久久久 | 免费视频成人 | 五月婷婷深开心 | 国产成人免费在线 | 91福利社在线观看 | 国产精品片 | 亚洲va欧洲va国产va不卡 | 五月婷婷激情六月 | 玖玖国产精品视频 | 午夜一级免费电影 | 国产成人精品一区二区在线 | 九九视频这里只有精品 | 亚洲在线精品 | 亚洲激情在线播放 | 日黄网站 | 免费视频黄色 | 久久久久女人精品毛片 | 久久污视频 | 国产无套一区二区三区久久 | 国产手机在线播放 | 黄av资源| 天天拍天天草 | 激情欧美一区二区免费视频 | 色av男人的天堂免费在线 | 日韩有码中文字幕在线 | 婷婷免费在线视频 | 少妇bbw撒尿 | 新版资源中文在线观看 | 精品色综合| 日韩高清一区二区 | 色五月成人 | 麻豆高清免费国产一区 | 精品99在线观看 | 91视频在线| 中文av影院| 国产精品18videosex性欧美 | 国产成人综合在线观看 | 在线激情网 | 夜夜夜夜夜夜操 | 久久国产91 | 亚洲免费视频在线观看 | www亚洲精品 | 亚洲va韩国va欧美va精四季 | 麻豆传媒视频在线免费观看 | 五月天激情综合 | 日韩精品视频在线免费观看 | 日韩精品视频免费专区在线播放 | 成人午夜在线电影 | 五月综合激情网 | 午夜少妇一区二区三区 | 草久久久久久久 | 久久99国产精品 | 欧美一区二区三区免费看 | 在线观看视频免费大全 | av高清网站在线观看 | 最新国产在线视频 | 国产精品高清在线观看 | 久久高清精品 | 精品亚洲欧美无人区乱码 | 欧美一区二区三区激情视频 | 久久av免费电影 | 婷婷视频| 日韩午夜在线播放 | 久久免费资源 | 五月婷视频| 麻豆视频免费播放 | 国产视频欧美视频 | 91免费观看视频网站 | 免费三级黄| 久久精品一区二区三区国产主播 | 99久久国产免费看 | 免费麻豆视频 | 中文字幕在线观看的网站 | 日韩激情视频在线观看 | 97免费中文视频在线观看 | 久久一区二区三区四区 | 久精品视频免费观看2 | 日本精品视频在线观看 | 99久久精品免费 | 综合网欧美 | 亚洲黄色一级视频 | 久久精品免费观看 | 色先锋资源网 | 狠狠的干狠狠的操 | 天堂av在线中文在线 | 日韩伦理片hd | 综合久久综合久久 | 久久久久久久久影视 | 日韩电影在线一区二区 | 欧美精品亚洲二区 | 欧美一级日韩免费不卡 | 麻豆免费观看视频 | 成人黄色小视频 | 国产又粗又硬又爽视频 | 亚洲3级 | 亚洲欧美少妇 | 欧美一区二区在线免费观看 | 一级免费片 | 日韩欧美高清在线 | 欧美肥妇free| 国产精国产精品 | 国产精品视频99 | 在线视频日韩精品 | 不卡的av电影 | 99精品视频观看 | 在线观看黄色av | 精品久操 | 毛片一级免费一级 | 欧美日韩在线第一页 | av在线收看 | 亚洲综合激情 | 在线色吧 | 91精品婷婷国产综合久久蝌蚪 | 久久成人午夜 | 91av视频网 | 成人免费观看视频大全 | 亚洲乱码国产乱码精品天美传媒 | 黄色国产大片 | 在线精品视频免费播放 | 97人人视频 | 婷婷丁香激情五月 | 亚洲视频综合 | 在线观看日韩专区 | 日韩欧美国产免费播放 | 久久久免费 | 国产va精品免费观看 | 国内精品在线看 | 免费在线色 | 婷婷激情影院 | 中文字幕色网站 | 亚洲在线综合 | 亚洲一区 av| 国产亚洲精品v | 激情视频一区二区 | 久久一本综合 | 黄色免费观看网址 | 久草a视频 | 色午夜 | 国产亚洲久一区二区 | 美女视频黄是免费的 | 久久avav| 在线视频观看亚洲 | 久久黄色免费观看 | www一起操| 国产精品久久久久久一区二区三区 | 国产麻豆果冻传媒在线观看 | www.天天色| 亚洲视频456| 日韩大陆欧美高清视频区 | 欧产日产国产69 | 成人免费观看视频大全 | 日韩欧美一区二区三区视频 | 2021国产精品视频 | 成人av资源在线 | 久久高清av | 久草电影在线观看 | 2019av在线视频 | 久一在线 | av网站在线观看免费 | 久久狠狠一本精品综合网 | 二区在线播放 | 久久久久久电影 | 三级a毛片 | 天天躁天天操 | 日本在线观看中文字幕无线观看 | 在线观看黄色免费视频 | 456成人精品影院 | 国产成人黄色片 | 国产精品日韩欧美一区二区 | 国产一级免费电影 | 中文字幕在线观看第三页 | 亚洲欧洲久久久 | 91精品视频一区二区三区 | 国产99久久久精品 | 久久精品在线免费观看 | 91精品国产一区二区三区 | 夜夜夜夜夜夜操 | 69av在线播放 | 久久亚洲私人国产精品va | 超碰av在线 | 亚洲视频在线播放 | 国产免费一区二区三区网站免费 | 国产精品成人免费精品自在线观看 | 男女免费视频观看 | 国产亚洲一级高清 | 久久女同性恋中文字幕 | 免费在线观看日韩视频 | av观看久久久 | 国产精品久久久毛片 | 国产对白av | 国产美女被啪进深处喷白浆视频 | 久久久久国产一区二区三区四区 | 久久国产精品一区二区三区四区 | 成年人黄色在线观看 |