⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 table.c

📁 一套接口
💻 C
字号:
static char rcsid[] = "$Id: table.c 6 2007-01-22 00:45:22Z drhanson $";#include <limits.h>#include <stddef.h>#include "mem.h"#include "assert.h"#include "table.h"#define T Table_Tstruct T {	int size;	int (*cmp)(const void *x, const void *y);	unsigned (*hash)(const void *key);	int length;	unsigned timestamp;	struct binding {		struct binding *link;		const void *key;		void *value;	} **buckets;};static int cmpatom(const void *x, const void *y) {	return x != y;}static unsigned hashatom(const void *key) {	return (unsigned long)key>>2;}T Table_new(int hint,	int cmp(const void *x, const void *y),	unsigned hash(const void *key)) {	T table;	int i;	static int primes[] = { 509, 509, 1021, 2053, 4093,		8191, 16381, 32771, 65521, INT_MAX };	assert(hint >= 0);	for (i = 1; primes[i] < hint; i++)		;	table = ALLOC(sizeof (*table) +		primes[i-1]*sizeof (table->buckets[0]));	table->size = primes[i-1];	table->cmp  = cmp  ?  cmp : cmpatom;	table->hash = hash ? hash : hashatom;	table->buckets = (struct binding **)(table + 1);	for (i = 0; i < table->size; i++)		table->buckets[i] = NULL;	table->length = 0;	table->timestamp = 0;	return table;}void *Table_get(T table, const void *key) {	int i;	struct binding *p;	assert(table);	assert(key);	i = (*table->hash)(key)%table->size;	for (p = table->buckets[i]; p; p = p->link)		if ((*table->cmp)(key, p->key) == 0)			break;	return p ? p->value : NULL;}void *Table_put(T table, const void *key, void *value) {	int i;	struct binding *p;	void *prev;	assert(table);	assert(key);	i = (*table->hash)(key)%table->size;	for (p = table->buckets[i]; p; p = p->link)		if ((*table->cmp)(key, p->key) == 0)			break;	if (p == NULL) {		NEW(p);		p->key = key;		p->link = table->buckets[i];		table->buckets[i] = p;		table->length++;		prev = NULL;	} else		prev = p->value;	p->value = value;	table->timestamp++;	return prev;}int Table_length(T table) {	assert(table);	return table->length;}void Table_map(T table,	void apply(const void *key, void **value, void *cl),	void *cl) {	int i;	unsigned stamp;	struct binding *p;	assert(table);	assert(apply);	stamp = table->timestamp;	for (i = 0; i < table->size; i++)		for (p = table->buckets[i]; p; p = p->link) {			apply(p->key, &p->value, cl);			assert(table->timestamp == stamp);		}}void *Table_remove(T table, const void *key) {	int i;	struct binding **pp;	assert(table);	assert(key);	table->timestamp++;	i = (*table->hash)(key)%table->size;	for (pp = &table->buckets[i]; *pp; pp = &(*pp)->link)		if ((*table->cmp)(key, (*pp)->key) == 0) {			struct binding *p = *pp;			void *value = p->value;			*pp = p->link;			FREE(p);			table->length--;			return value;		}	return NULL;}void **Table_toArray(T table, void *end) {	int i, j = 0;	void **array;	struct binding *p;	assert(table);	array = ALLOC((2*table->length + 1)*sizeof (*array));	for (i = 0; i < table->size; i++)		for (p = table->buckets[i]; p; p = p->link) {			array[j++] = (void *)p->key;			array[j++] = p->value;		}	array[j] = end;	return array;}void Table_free(T *table) {	assert(table && *table);	if ((*table)->length > 0) {		int i;		struct binding *p, *q;		for (i = 0; i < (*table)->size; i++)			for (p = (*table)->buckets[i]; p; p = q) {				q = p->link;				FREE(p);			}	}	FREE(*table);}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -