数据结构 ——哈希表

数据结构 ------哈希表

1、哈希表的概念

概念参考 算法数据结构基础------哈希表(Hash Table)

2、代码实现

下面是用数组实现哈希结构,开放地址法解决冲突的简单哈希表操作

c 复制代码
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

//数组实现哈希结构,开放地址法解决冲突
typedef struct pair
{
    int key;//构造一个键出来,充当哈希地址的求解 key,element(存放字符串)
    char element[20];//存放字符串
}data;

typedef struct hashTable
{
    data **table;//二级指针方便初始化
    int divisor;//除因子 H(key)=key%divisor
    int curSize;//当前表大小
}hash;
//创表
hash *createHashTable(int divisor)
{
    hash *ht = (hash *)malloc(sizeof(hash));
    if(ht == NULL)
       return NULL;
    ht->curSize = 0;
    ht->divisor = divisor;
    ht->table=(data **)malloc(sizeof(data *) *(ht->divisor));//分配divisor块4字节的内存放指针
    if(ht->table == NULL)
    {
        free(ht);//申请失败,返回前释放前面申请的内存
        return NULL;
    }
    //指针数组初始化1
    #if 0
    for (int i = 0; i < divisor; i++)
    {
        ht->table[i] = NULL;//每块指针指向空
    }
    #endif
    //指针数组初始化2,直接把指针数组的内存全部置零,即初始化为NULL
    memset(ht->table,0,sizeof(data *)*ht->divisor);
    return ht;
}
//找存储当前元素的地址
int search(hash *ht,int key) 
{
    int pos=key % ht->divisor;//不存在冲突,直接存放
    //开放地址法解决冲突
    int curPos=pos;
    do
    {
        //key相同,采用覆盖的数据方式
        if((ht->table[curPos]==NULL) || (ht->table[curPos]->key==key))
            return curPos;
        curPos=(curPos+1)%ht->divisor;
    } while (curPos!=pos);//从原位置的下一个位置开始找空位,若再次回到原位置,说明没有空间了
    return curPos;
}   
//插入元素 传参不一样
#if 0
void insert(hash *ht,data data)
{
    int pos=search(ht,data.key);//找存储当前元素的地址
    if(ht->table[pos]==NULL)
    {
        //当前位置为空,直接插入
        ht->table[pos]=malloc(sizeof(data));
        if(ht->table[pos]==NULL)
           return;
        memcpy(ht->table[pos],&data,sizeof(data));
        ht->curSize++;
    }
    else
    {
        //key相同,采用覆盖的数据方式
        if(ht->table[pos]->key==data.key)
        {
            strcpy(ht->table[pos]->element,data.element);
        }
        else 
        {
            //如果表满了,会返回原来的位置,但原来的位置与传进来的key会不匹配
            printf("hashTable is full\n");
            return;
        }
    }
}
#endif

void insert(hash *ht,data *d)
{
    int pos=search(ht,d->key);//找存储当前元素的地址
    if(ht->table[pos]==NULL)
    {
        //当前位置为空,直接插入
        ht->table[pos]=malloc(sizeof(data));
        if(ht->table[pos]==NULL)
           return;//内存申请失败
        memcpy(ht->table[pos],d,sizeof(data));
        ht->curSize++;
    }
    else
    {
        //key相同,采用覆盖的数据方式
        if(ht->table[pos]->key==d->key)
        {
            strcpy(ht->table[pos]->element,d->element);
        }
        else 
        {
            //如果表满了,会返回原来的位置,但原来的位置与传进来的key会不匹配
            printf("hashTable is full\n");
            return;
        }
    }
}
//遍历哈希表
void travel(hash *ht)
{
    for (int i = 0; i < ht->divisor; i++)
    {
        if(ht->table[i]!=NULL)
          printf("%d:%s\n",ht->table[i]->key,ht->table[i]->element);
        else
          printf("NULL\n");
    }
}
//销毁
void destroy(hash *ht)
{
    for (int i = 0; i < ht->divisor; i++)
    {
        if(ht->table[i]!=NULL)
        {
            free(ht->table[i]);
            ht->table[i]=NULL;
        }
    }
    free(ht->table);
}

int main()
{
    hash *ht=createHashTable(10);
    data arr[6]={1,"hello",1,"hellomy",3,"women",4,"lucky",5,"money",11,"world123"};
    for(int i=0;i<6;i++)
    {
        //arr[i]是具体元素,&arr[i]是才是地址,指针
        insert(ht,&arr[i]);
    }
    travel(ht);
    destroy(ht);
    return 0;
}

3、邻接表法实现哈希结构

相关推荐
CSharp精选营4 天前
关系型 vs 非关系型:从原理到选型,一文搞定数据库核心分类
数据结构·nosql·关系型数据库·非关系型数据库·技术选型
刘马想放假7 天前
Modbus 全栈技术解析:TCP、RTU、ASCII、RTU over TCP
数据结构·网络协议
北域码匠8 天前
冒泡排序太慢?鸡尾酒排序双向优化,原生 C# 零第三方库完整代码
数据结构·排序算法·泛型·c# 算法·鸡尾酒排序·原生 c# 开发·冒泡排序优化·嵌入式算法
Darling噜啦啦15 天前
列表转树算法深度解析:从 Map 到 Reduce 的两种实现,面试高频考点
数据结构·算法·面试
小小工匠16 天前
Redis - 事务机制:能实现 ACID 属性吗
数据结构·redis·性能优化·并发·持久化
玖玥拾16 天前
C/C++ 数据结构(七)栈、容器适配器
c语言·数据结构·c++··容器适配器
Qres82116 天前
算法复键——树状数组
数据结构·算法
牛油果子哥q16 天前
并查集(DSU)超精讲,路径压缩、按秩合并、万能模板、连通性判定、最小生成树与刷题实战全解
数据结构·c++·最小生成树·并查集
凌波粒16 天前
LeetCode--491.递增子序列(回溯算法)
数据结构·算法·leetcode
WL学习笔记16 天前
单项不带头不循环链表
数据结构·链表