C语言数据结构哈希表详解

 更新时间:2022年02月26日 15:17:55   作者:橙子@C  
哈希表是一种根据关键码去寻找值的数据映射结构,该结构通过把关键码映射的位置去寻找存放值的地方,说起来可能感觉有点复杂,我想我举个例子你就会明白了,最典型的的例子就是字典

/*
 * 程序名:hash.c,此程序演示哈希表的实现,数据元素单链表带头结点。
 * 
*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
 
// 哈希表中数据元素的结构体。
typedef struct Element
{
  unsigned int key; // 关键字。
  int value;        // 数据元素其它数据项,可以是任意数据类型。
  // char value[1001];        // 数据元素其它数据项,可以是任意数据类型。
}Element;
 
// 数据元素单链表。
typedef struct Node
{
  Element elem;      // 数据元素。
  struct Node *next; // next指针。
}Node;
 
// 哈希表
typedef struct HashTable
{
  struct Node *head;  // 数据元素存储基址,动态分配数组。
  int tablesize;      // 哈希表当前大小,即表长。
  int count;          // 哈希表中数据元素的个数。
}HashTable;
 
// 初始化哈希表,tablesize为哈希表的表长,返回哈希表的地址。
HashTable *InitHashTable(const unsigned int tablesize)
{
  // 分配哈希表。
  HashTable *hh=(HashTable *)malloc(sizeof(HashTable));
 
  hh->tablesize=tablesize;  // 哈希表长。
 
  // 分配和初始化数据元素单链表的头结点。
  hh->head=(Node *)malloc((hh->tablesize)*sizeof(Node));
  memset(hh->head,0,(hh->tablesize)*sizeof(Node));
 
  hh->count=0;  // 哈希表中数据元素个数置为0。
 
  return hh;
}
 
// 哈希函数。
unsigned int Hash(HashTable *hh,unsigned int key)
{
  return key%hh->tablesize;  // 对表长取余。
}
 
// 在哈希表中查找关键字,成功返回单链表结点的地址,失败返回空。
Node *LookUp(HashTable *hh,unsigned int key)
{
  int ii;
 
  ii=Hash(hh,key);  // 获取关键key的哈希地址。
 
  Node *pp=hh->head[ii].next;
 
  // 遍历单链表。
  while( (pp!=NULL) && (pp->elem.key!=key) )
  {
    pp=pp->next;
  }
 
  return pp;
}
 
// 从哈希表中删除关键及其数据,成功返回1,如果关键字不存在返回0。
int Delete(HashTable *hh,unsigned int key)
{
  int ii;
 
  ii=Hash(hh,key);  // 获取关键key的哈希地址。
 
  Node *pp=&hh->head[ii];
 
  // 遍历单链表,pp指针停留在待删除关键key的前一结点。
  while( (pp->next!=NULL) && (pp->next->elem.key!=key) )
  {
    pp=pp->next;
  }
 
  if (pp->next==NULL) return 0;  // 查找失败。
 
  Node *tmp=pp->next;        // tmp为将要删除的结点。
  pp->next=pp->next->next;   // 写成p->next=tmp->next更简洁。
 
  free(tmp);     // 释放结点。
 
  hh->count--;   // 表中元素个数减1。
 
  return 1;
}
 
// 向哈希表中插入数据元素,成功返回1,如果数据元素关键字已存在,返回0。
int Insert(HashTable *hh,Element *ee)
{
  // 查找关键字是否已存在,如果存在,插入失败。
  Node *pp=LookUp(hh,ee->key);
 
  if (pp!=NULL) { printf("关键字%d已存在。\n",ee->key); return 0; }
  
  Node *qq=(Node *)malloc(sizeof(Node));
 
  memcpy(&qq->elem,ee,sizeof(Element));
 
  // 用头插法插入新数据元素。
  int ii=Hash(hh,ee->key);
  qq->next=hh->head[ii].next;
  hh->head[ii].next=qq;
  
  hh->count++;   // 表中元素个数加1。
 
  return 1;
}
 
// 销毁哈希表
void FreeHashTable(HashTable *hh)
{
  int ii;
 
  Node *pp,*qq;
 
  // 释放全部的单链表。
  for(ii=0;ii<hh->tablesize;ii++)
  {
    pp=hh->head[ii].next;
    while(pp)
    {
      qq=pp->next;
      free(pp);
      pp=qq;
    }
  }
 
  // 释放全部单链表的头结点数组。
  free(hh->head);
 
  free(hh);  // 释放哈希表。
}
 
// 打印哈希表。
void PrintTable(HashTable *hh)
{
  int ii; 
 
  for (ii=0;ii<hh->tablesize;ii++)
  {
    Node *pp=hh->head[ii].next;
    while (pp)
    {
      printf("[%d-%d] ",pp->elem.key,pp->elem.value);
      // printf("[%d-%s] ",pp->elem.key,pp->elem.value);
      pp=pp->next;
    }
 
    printf("^\n");
  }
 
  printf("\n");
}
 
int main()
{
  // 初始化哈希表。
  HashTable *hh=InitHashTable(10);
 
  Element ee;
 
  // 插入数据元素,关键字从10到20。
  ee.key=10; ee.value=110; Insert(hh,&ee);
  ee.key=11; ee.value=111; Insert(hh,&ee);
  ee.key=12; ee.value=112; Insert(hh,&ee);
  ee.key=13; ee.value=113; Insert(hh,&ee);
  ee.key=14; ee.value=114; Insert(hh,&ee);
  ee.key=15; ee.value=115; Insert(hh,&ee);
  ee.key=16; ee.value=116; Insert(hh,&ee);
  ee.key=17; ee.value=117; Insert(hh,&ee);
  ee.key=18; ee.value=118; Insert(hh,&ee);
  ee.key=19; ee.value=119; Insert(hh,&ee);
 
  // 插入数据元素,关键字从20到30。
  ee.key=20; ee.value=120; Insert(hh,&ee);
  ee.key=21; ee.value=121; Insert(hh,&ee);
  ee.key=22; ee.value=122; Insert(hh,&ee);
  ee.key=23; ee.value=123; Insert(hh,&ee);
  ee.key=24; ee.value=124; Insert(hh,&ee);
  ee.key=25; ee.value=125; Insert(hh,&ee);
  ee.key=26; ee.value=126; Insert(hh,&ee);
  ee.key=27; ee.value=127; Insert(hh,&ee);
  ee.key=28; ee.value=128; Insert(hh,&ee);
  ee.key=29; ee.value=129; Insert(hh,&ee);
 
  // 插入数据元素,关键字从30到32。
  ee.key=30; ee.value=130; Insert(hh,&ee);
  ee.key=31; ee.value=131; Insert(hh,&ee);
  ee.key=32; ee.value=132; Insert(hh,&ee);
 
  printf("count=%d\n",hh->count);
  PrintTable(hh);    // 打印哈希表 
 
  Delete(hh,12);     // 删除哈希表中关键字为12的数据元素。
 
  printf("count=%d\n",hh->count);
  PrintTable(hh);    // 打印哈希表 
 
  // 在哈希表中查找关键字18。
  Node *pp=LookUp(hh,18);
  if (pp==0) printf("LookUp(18) failed.\n");
  else printf("key=18,value=%d.\n",pp->elem.value);  
 
  // ee.key=10; strcpy(ee.value,"<no>00010<no/><name>西施</name><yz>绝世美人</yz>"); Insert(hh,&ee);
  // PrintTable(hh);    // 打印哈希表 
 
  FreeHashTable(hh);  // 销毁哈希表 
 
  return 0;
}

到此这篇关于C语言数据结构哈希表详解的文章就介绍到这了,更多相关C语言 哈希表内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • C++文件IO流及stringstream流读写文件和字符串操作详解

    C++文件IO流及stringstream流读写文件和字符串操作详解

    本文详细介绍C++中的文件IO流和stringstream流的使用方法,包括文件的打开、读写操作,以及字符串的输入输出、转换等操作。同时提供实用的示例代码和技巧,帮助读者更好地掌握这两种流的使用
    2023-04-04
  • c++ 单线程实现同时监听多个端口

    c++ 单线程实现同时监听多个端口

    这篇文章主要介绍了c++ 单线程实现同时监听多个端口的方法,帮助大家更好的理解和学习使用c++,感兴趣的朋友可以了解下
    2021-03-03
  • C++中实现保存数据到CSV文件

    C++中实现保存数据到CSV文件

    这篇文章主要介绍了C++中实现保存数据到CSV文件方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-08-08
  • 基于C/C++将派生类赋值给基类的超详细讲解

    基于C/C++将派生类赋值给基类的超详细讲解

    类其实也是一种数据类型,也可以发生数据类型转换,下面这篇文章主要给大家介绍了关于基于C/C++将派生类赋值给基类的超详细讲解,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2023-06-06
  • C++中关于多态实现和使用方法

    C++中关于多态实现和使用方法

    这篇文章主要介绍了C++中关于多态实现和使用方法,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-07-07
  • 详解C语言中return与exit的区别

    详解C语言中return与exit的区别

    这篇文章主要介绍了详解C语言中return与exit的区别的相关资料,希望通过本文能帮助到大家,让大家理解这部分内容,需要的朋友可以参考下
    2017-10-10
  • C++ 面试题翻译电话号码实例代码

    C++ 面试题翻译电话号码实例代码

    这篇文章主要介绍了C++ 面试题翻译电话号码实例代码的相关资料,需要的朋友可以参考下
    2017-04-04
  • 显示任何进程加载的DLL文件的代码

    显示任何进程加载的DLL文件的代码

    c语言实现的显示任何进程加载的DLL,方便开发软件的朋友
    2013-05-05
  • C++实现简单的生产者-消费者队列详解

    C++实现简单的生产者-消费者队列详解

    这篇文章主要为大家详细介绍了如何利用C++实现一个简单的生产者-消费者队列,文中的示例代码讲解详细,感兴趣的小伙伴可以了解一下
    2023-04-04
  • C语言线性表顺序表示及实现

    C语言线性表顺序表示及实现

    这篇文章主要介绍了C语言线性表顺序表示及实现,线性表是最常用且最简单的一种数据结构。简而言之,一个线性表是n个数据元素的有限序列
    2022-07-07

最新评论