哈希桶 entry_哈希表代码实现

9d46f2cf7d87944ef87ef6efc8eff172.png

哈希表的主要用处:快速的数据存储和查找。例如,在我们日常上网过程中搜索某条信息的时候,信息的存储方式决定了查找该信息的速度,哈希表结合了数组的便捷访问和链表的便捷查找和删除的特点。

实现方式:通过哈希函数获取哈希表的地址,遇到冲突的情况下采用拉链法解决冲突。

时间复杂度O(1)

#include <stdio.h>

#include <stdlib.h>

#include <string.h>

/*定义哈希表数组的长度*/

#define BUCKETCOUNT 16

/*哈希表结点数据结构定义*/

struct hashEntry

{

char *key;

char *value;

struct hashEntry *next;

};

typedef struct hashEntry entry;

struct hashTable

{

entry bucket[BUCKETCOUNT];/*先默认定义16个桶*/

};

typedef struct hashTable table;

/*初始化哈希表*/

void initHashTable(table *t)

{

int i;

if(!t)

return;

for(i = 0; i < BUCKETCOUNT; ++i)

{

t->bucket[i].key = NULL;

t->bucket[i].next = NULL;

t->bucket[i].value = NULL;

}

}

/*散列函数*/

int keyToIndex(const char *key)

{

int index, len, i;

if(key == NULL)

return -1;

len = strlen(key);

index = (int)key[0];

for(i = 1; i < len; ++i)

{

index *= 1103515245 + (int)key[i];

}

index >>= 27;

index &= (BUCKETCOUNT - 1);

return index;

}

/*在堆上分配足以保存str的内存,并拷贝str的内容到新分配的位置*/

char *strDup(const char *str)

{

int len;

char *ret;

if(!str)

return NULL;

len = strlen(str);

ret = (char *)malloc(sizeof(len + 1));

if(!ret)

return NULL;

memcpy(ret, str, len);

ret[len] = '0';

return ret;

}

/*插入数据到hash表中*/

int insertEntry(table *t, const char *key, const char *value)

{

int index, vlen1, vlen2;

entry *e, *ep;

if(t == NULL || key == NULL || value == NULL)

return -1;

index = keyToIndex(key);

if(!t->bucket[index].key)

{

t->bucket[index].key = strDup(key);

t->bucket[index].value = strDup(value);

}

else

{

e = ep = &(t->bucket[index]);

/*先从已经存在的找*/

while(e)

{

/*找到key所在的位置,替换相应的值*/

if(strcmp(e->key, key) == 0)

{

vlen1 = strlen(value);

vlen2 = strlen(e->value);

if(vlen1 > vlen2)

{

free(e->value);

e->value = (char *)malloc(sizeof(vlen1 + 1));

}

memcpy(e->value, value, vlen1 + 1);

return index;/*插入完成*/

}

ep = e;

e = e->next;

}

/*没有在当前桶中找到,创建条目加入*/

e = (entry *)malloc(sizeof(entry));

e->key = strDup(key);

e->value = strDup(value);

e->next = NULL;

ep->next = NULL;

}

return index;

}

/*找到哈希表中key对应的entry,找到之后返回对应的entry,并将其删除*/

/*没找到返回NULL*/

entry *removeEntry(table *t, char *key)

{

int index;

entry *e, *ep;

if(!t || !key)

return NULL;

index = keyToIndex(key);

e = &(t->bucket[index]);

while(e)

{

if(strcmp(e->key, key) == 0)/*如果是桶的第一个*/

{

if(e == &(t->bucket[index]))

{

ep = e->next;

if(ep)

{

entry *tmp = e;

e = ep;

ep = tmp;

ep->next = NULL;

}

else/*这个桶只有一个元素*/

{

//ep = (entry*)malloc(sizeof(entry));

ep = e;

e->key = e->value = NULL;

e->next = NULL;

}

return ep;

}

}

else/*如果不是木桶第一个元素*/

{

//找到它的前一个

ep = t->bucket[index].next;

while((ep->key != key) && ep)

{

ep = ep->next;

}

if(!ep)

return NULL;

return ep;

}

e = e->next;

}

return NULL;

}

/*打印hash表*/

void printTable(table *t)

{

int i;

entry *e;

if(!t)

return;

for(i = 0; i < BUCKETCOUNT; i++)

{

printf("nbucket[%d]:n", i);

e = &(t->bucket[i]);

while(e)

{

printf("t%st = t%sn",e->key, e->value);

e = e->next;

}

}

}

/*释放哈希表*/

void freeHashTable(table* t)

{

int i;

entry* e,*ep;

if (t == NULL)return;

for (i = 0; i<BUCKETCOUNT; ++i) {

e = &(t->bucket[i]);

while (e->next != NULL) {

ep = e->next;

e->next = ep->next;

free(ep->key);

free(ep->value);

free(ep);

}

}

}

int main(void)

{

table t;

initHashTable(&t);

insertEntry(&t , "电脑型号" , "华硕 X550JK 笔记本电脑");

insertEntry(&t , "操作系统" , "Windows 8.1 64位 (DirectX 11)");

insertEntry(&t , "处理器" , "英特尔 Core i7 - 4710HQ @ 2.50GHz 四核");

insertEntry(&t , "主板" , "华硕 X550JK(英特尔 Haswell)");

insertEntry(&t , "内存" , "4 GB(Hynix / Hyundai)");

insertEntry(&t , "主硬盘" , "日立 HGST HTS541010A9E680(1 TB / 5400 转 / 分)");

insertEntry(&t , "显卡" , "NVIDIA GeForce GTX 850M (2 GB / 华硕)");

insertEntry(&t , "显示器" , "奇美 CMN15C4(15.3 英寸)");

insertEntry(&t , "光驱" , "松下 DVD - RAM UJ8E2 S DVD刻录机");

insertEntry(&t , "声卡" , "Conexant SmartAudio HD @ 英特尔 Lynx Point 高保真音频");

insertEntry(&t , "网卡" , "瑞昱 RTL8168 / 8111 / 8112 Gigabit Ethernet Controller / 华硕");

insertEntry(&t , "主板型号" , "华硕 X550JK");

insertEntry(&t , "芯片组" , "英特尔 Haswell");

insertEntry(&t , "BIOS" , "X550JK.301");

insertEntry(&t , "制造日期" , "06 / 26 / 2014");

insertEntry(&t , "主人" , "就是我");

insertEntry(&t , "价格" , "六十张红色毛主席");

insertEntry(&t , "主硬盘" , "换了个120G的固态");

entry *e = removeEntry(&t, "主板型号");

if(!e)

{

printf("不存在相关记录.n");

}

else

{

puts("释放记录结点的内存.");

free(e->key);

free(e->value);

free(e);

e = NULL;

}

printTable(&t);

freeHashTable(&t);

getchar();

return 0;

}

a8073046cddc1835fbef7bc1b4d44bbe.png

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/436339.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

串口握手协议的XON\XOFF原理

串口握手协议的XON\XOFF原理有人可能搞不清楚&#xff0c;看看下面这个例子你应该能有所感悟。 假设电脑A和电脑B通过RS232串口通信 A 和 B协商使用软件握手协议(dcb.fOutX TRUE), 定义XonChar \ XoffChar 比如XonChar 1, XoffChar 2. 现在, A向B发送字符2, B向A发送字符串&…

中秋望月

中秋望月——代腾飞 2007年9月23日 于成都皎皎夜空孤月轮且盼嫦娥来相会但见江水东流去只问佳人几时归 转载于:https://www.cnblogs.com/daitengfei/archive/2007/09/24/903765.html

【转】C++ 存储类

C 存储类 存储类定义 C 程序中变量/函数的范围&#xff08;可见性&#xff09;和生命周期。这些说明符放置在它们所修饰的类型之前。下面列出 C 程序中可用的存储类&#xff1a; autoregisterstaticexternmutable auto 存储类 auto 存储类是所有局部变量默认的存储类。 {in…

投标报价得分计算程序_什么是投标报价?怎么计算呢?

投标价格是投标文件中的实质性内容&#xff0c;也是评审因素中的核心内容。在招标采购活动中&#xff0c;由于投标人的疏忽&#xff0c;经常出现投标报价遗漏、书写错误、前后不一等问题。今天保标招标网小编就跟大家一起来看看什么是投标报价&#xff1f;招投标中复价是什么意…

设备I/O之OVERLAPPED

OVERLAPPED,顾名思义为重叠,乍一看会很奇怪,重叠?谁跟谁重叠?似乎在WIN32的Programming中没有这个概念呀?要讨论这个问题就要追溯到对设备I/O的访问中。 在WIN32中,用户不能象以前那样直接对硬件进行访问,使得这一层对开发者而言是个"黑盒",而提供了一组对应的API…

Question of the Day: Microsoft | Database, Multiple Questions in One

Question of the Day: Microsoft | Database, Multiple Questions in One from CareerCup by Jason Yip HR: SQL What is an index? Explain cluster index. What are the different types of joins. Explain. Explain normalizatiion. What is a foreign key? What is the d…

【转】在windows上安装VTK

看了很多教程&#xff0c;花了1天半的时间装上了&#xff0c;记录下。 前置条件&#xff1a;我安装了VS2015&#xff0c;用来编译工程。 参考资料 官方&#xff1a;http://www.vtk.org/Wiki/VTK/Building 安装&#xff1a;http://blog.csdn.net/Chinamming/article/details/1…

映射网络驱动器后为什么每次都要重新连接_如何映射网络驱动器?

有些设备的电脑需要处理一些共享资料或者需要一些外部资料才能生产运行&#xff0c;但是这些资料在本电脑没有&#xff0c;要去企业的公共盘&#xff08;共享磁盘&#xff09;里面获取&#xff0c;公共盘是网络上驱动器&#xff0c;这时候我们就要把公共盘映射到本地电脑。PCB厂…

解析 Linux 中的 VFS 文件系统机制

简介&#xff1a; 本文阐述 Linux 中的文件系统部分&#xff0c;源代码来自基于 IA32 的 2.4.20 内核。总体上说 Linux 下的文件系统主要可分为三大块&#xff1a;一是上层的文件系统的系统调用&#xff0c;二是虚拟文件系统 VFS(Virtual Filesystem Switch)&#xff0c;三是挂…

Js——elementFromPoint方法

src&#xff1a;http://www.aspxuexi.com/html/dhtml/2007-8-30/elementFromPoint.htm 语法&#xff1a; oElement document . elementFromPoint ( iX , iY ) 参数&#xff1a; iX :  必选项。整数(Integer)。单位&#xff1a;象素(Pixel)。定位横坐标偏移量。 iY :  必…

【转】C++中#include包含头文件带 .h 和不带 .h 的区别

C中#include包含头文件带 .h 和不带 .h 的区别&#xff1f; 如 #include <iostream> 和 #include <iostream.h> 包含的东西有哪些不同&#xff1f;之前在写C程序的时候只知道使用 #include <iostream> 的时候&#xff0c;使用函数前要用 using namespace st…

mysql 第二大的数据_MYSQL – 从大表中的第二行中选择数据

我有一个外部第三方程序实时将数据库导出到mysql,我想显示报告数据.所以,我无法改变结构,因为它是实时同步的.表结构是这样的ID | Date | Transaction-----------------------------12 | 2012-11-01 | 20012 | 2012-11-02 | 25012 | 2012-11-03 | 15012 | 2012-11-04 | 100012 …

串口通讯编程一日通1(整合资料) .

本文以VC为平台进行串口通信编程 串口通讯在VC下实现方式有很多&#xff0c;控件自然是最简单话的方式了&#xff0c;但由于控件只支持对话框程序&#xff0c;有些场合又不需要对话框&#xff0c;所以用Windows API实现是比较好的方式 串行通信的操作方式 1.同步方式 同步方…

去除分页预览

有时我们在打开一个Excel表的时候,会出现如下图样式: 此为作了"分页预览"设置,取消方法如下图: 选择"普通"即可转载于:https://www.cnblogs.com/wxbbk/archive/2007/10/07/915773.html

【转】C++指针详解!!!!

指针的概念 指针是一个特殊的变量&#xff0c;它里面存储的数值被解释成为内存里的一个地址。要搞清一个指针需要搞清指针的四方面的内容&#xff1a;指针的类型&#xff0c;指针所指向的类型&#xff0c;指针的值或者叫指针所指向的内存区&#xff0c;还有指针本身所占据的内…

mysql rename语法错误_为什么现在MySQL无法重命名数据库?

MySQL 之前提供了一个 rename database db_old to db_new 的命令来直接对数据库改名&#xff0c;可能由于实现的功能不完备(比如&#xff0c;这条命令可能是一个超大的事务&#xff0c;或者是由于之前的表很多还是 MyISAM 等)&#xff0c;后来的版本直接取消了这条命令。一、my…

串口通讯编程一日通2(Overlapped IO模型)

第一篇初步了解串口的大致运作&#xff0c;接下来我们看基本操作 先看串口操作的数据结构&#xff1a; 串口操作有几个比较重要的Struct 1.Overlapped I/O 异步I/O模型 异步I/O和同步I/O不同&#xff0c;同步I/O时&#xff0c;程序被挂起&#xff0c;一直到I/O处理完&#xff0…

【转】VTK + QT + VS 编译安装配置

VTK Vtk&#xff0c;&#xff08;visualization toolkit&#xff09;是一个开源的免费软件系统&#xff0c;主要用于三维计算机图形学、图像处理和可视化。Vtk是在面向对象原理的基础上设计和实现的&#xff0c;它的内核是用C构建的&#xff0c;包含有大约250,000行代码&#…

python手动回收内存哪家好_谈谈如何手动释放Python的内存

在上篇博客中&#xff0c;提到了对一个脚本进行的多次优化。当时以为已经优化得差不多了&#xff0c;但是当测试人员测试时&#xff0c;我才发现&#xff0c;踩到了Python的一个大坑。在上文的优化中&#xff0c;对每500个用户&#xff0c;会进行一些计算并记录结果在磁盘文件中…

串口通讯编程一日通3(COMMTIMEOUTS DCB整理)

上一篇看了Overlapped IO模型后&#xff0c;接下来看剩下两个重要结构&#xff1a; 2.COMMTIMEOUTS结构 超时设置 COMMTIMEOUTS&#xff1a;COMMTIMEOUTS主要用于串口超时参数设置。COMMTIMEOUTS结构如下&#xff1a; typedef struct _COMMTIMEOUTS { D…