dict.c: introduce dictUnlink().

Notes by @antirez:

This patch was picked from a larger commit by Oran and adapted to change
the API a bit. The basic idea is to avoid double lookups when there is
to use the value of the deleted entry.

BEFORE:

    entry = dictFind( ... ); /* 1st lookup. */
    /* Do somethjing with the entry. */
    dictDelete(...);         /* 2nd lookup. */

AFTER:

    entry = dictUnlink( ... ); /* 1st lookup. */
    /* Do somethjing with the entry. */
    dictFreeUnlinkedEntry(entry); /* No lookups!. */
This commit is contained in:
oranagra 2016-05-09 18:01:09 +03:00 committed by antirez
parent 8c84c962cf
commit afcbcc0e58
3 changed files with 45 additions and 11 deletions

View File

@ -407,14 +407,15 @@ dictEntry *dictReplaceRaw(dict *d, void *key) {
return entry ? entry : existing; return entry ? entry : existing;
} }
/* Search and remove an element */ /* Search and remove an element. This is an helper function for
static int dictGenericDelete(dict *d, const void *key, int nofree) * dictDelete() and dictUnlink(), please check the top comment
{ * of those functions. */
static dictEntry *dictGenericDelete(dict *d, const void *key, int nofree) {
unsigned int h, idx; unsigned int h, idx;
dictEntry *he, *prevHe; dictEntry *he, *prevHe;
int table; int table;
if (d->ht[0].size == 0) return DICT_ERR; /* d->ht[0].table is NULL */ if (d->ht[0].used == 0) return NULL;
if (dictIsRehashing(d)) _dictRehashStep(d); if (dictIsRehashing(d)) _dictRehashStep(d);
h = dictHashKey(d, key); h = dictHashKey(d, key);
@ -432,27 +433,58 @@ static int dictGenericDelete(dict *d, const void *key, int nofree)
if (!nofree) { if (!nofree) {
dictFreeKey(d, he); dictFreeKey(d, he);
dictFreeVal(d, he); dictFreeVal(d, he);
zfree(he);
} }
zfree(he);
d->ht[table].used--; d->ht[table].used--;
return DICT_OK; return he;
} }
prevHe = he; prevHe = he;
he = he->next; he = he->next;
} }
if (!dictIsRehashing(d)) break; if (!dictIsRehashing(d)) break;
} }
return DICT_ERR; /* not found */ return NULL; /* not found */
} }
/* Remove an element, returning DICT_OK on success or DICT_ERR if the
* element was not found. */
int dictDelete(dict *ht, const void *key) { int dictDelete(dict *ht, const void *key) {
return dictGenericDelete(ht,key,0); return dictGenericDelete(ht,key,0) ? DICT_OK : DICT_ERR;
} }
int dictDeleteNoFree(dict *ht, const void *key) { /* Remove an element from the table, but without actually releasing
* the key, value and dictionary entry. The dictionary entry is returned
* if the element was found (and unlinked from the table), and the user
* should later call `dictFreeUnlinkedEntry()` with it in order to release it.
* Otherwise if the key is not found, NULL is returned.
*
* This function is useful when we want to remove something from the hash
* table but want to use its value before actually deleting the entry.
* Without this function the pattern would require two lookups:
*
* entry = dictFind(...);
* // Do something with entry
* dictDelete(dictionary,entry);
*
* Thanks to this function it is possible to avoid this, and use
* instead:
*
* entry = dictUnlink(dictionary,entry);
* // Do something with entry
* dictFreeUnlinkedEntry(entry); // <- This does not need to lookup again.
*/
dictEntry *dictUnlink(dict *ht, const void *key) {
return dictGenericDelete(ht,key,1); return dictGenericDelete(ht,key,1);
} }
/* You need to call this function to really free the entry after a call
* to dictUnlink(). */
void dictFreeUnlinkedEntry(dict *d, dictEntry *he) {
dictFreeKey(d, he);
dictFreeVal(d, he);
zfree(he);
}
/* Destroy an entire dictionary */ /* Destroy an entire dictionary */
int _dictClear(dict *d, dictht *ht, void(callback)(void *)) { int _dictClear(dict *d, dictht *ht, void(callback)(void *)) {
unsigned long i; unsigned long i;

View File

@ -154,7 +154,8 @@ dictEntry *dictAddRaw(dict *d, void *key, dictEntry **existing);
int dictReplace(dict *d, void *key, void *val); int dictReplace(dict *d, void *key, void *val);
dictEntry *dictReplaceRaw(dict *d, void *key); dictEntry *dictReplaceRaw(dict *d, void *key);
int dictDelete(dict *d, const void *key); int dictDelete(dict *d, const void *key);
int dictDeleteNoFree(dict *d, const void *key); dictEntry *dictUnlink(dict *ht, const void *key);
void dictFreeUnlinkedEntry(dict *d, dictEntry *he);
void dictRelease(dict *d); void dictRelease(dict *d);
dictEntry * dictFind(dict *d, const void *key); dictEntry * dictFind(dict *d, const void *key);
void *dictFetchValue(dict *d, const void *key); void *dictFetchValue(dict *d, const void *key);

View File

@ -3085,7 +3085,8 @@ int moduleUnload(sds name) {
/* Remove from list of modules. */ /* Remove from list of modules. */
serverLog(LL_NOTICE,"Module %s unloaded",module->name); serverLog(LL_NOTICE,"Module %s unloaded",module->name);
dictDeleteNoFree(modules,module->name); dictDelete(modules,module->name);
module->name = NULL; /* The name was already freed by dictDelete(). */
moduleFreeModuleStructure(module); moduleFreeModuleStructure(module);
return REDISMODULE_OK; return REDISMODULE_OK;