哈希表的c语言实现

1.哈希表和数组

        我们先来看这样一个问题

        问题:

        把乱序5个自然数 1 2 3 4 5存入到数组中,查询自然数5并返回它的下标

        一个简单的思路就是申请一个数组,然后遍历数组查询值为5的下标并返回,c语言的代码实现如下

#include "stdio.h"
#include "stdlib.h"
int main() {
    int arr1[5] = {4, 3, 2, 1, 5};
    for (int i = 0; i < 5; ++i) {
        if (arr1[i] == 5) {
            printf("index is %d", i);
            break;
        }
    }
}

        很明显,上述代码的时间复杂度是o(n),有没有更快的方式呢,就是给出一个数,通过计算就能得到它的下标在哪?

        我们可以设计这种计算,输入的是数字,输出的是下标,这个函数叫做哈希函数,每个数字和英文字母都应对应的ASCII码,我们用 ASCII % 5的方式来作为它的下标,哈希函数就是: 输入值的ASCII % 5,c语言的代码实现如下

#include <stdio.h>
#include <stdlib.h>

typedef struct Dict {
    int size;
    int *value;
}Dict;

// 哈希函数,返回下标值
int hash(int key) {
    return key % 5;
}

Dict* initDict() {
    Dict* dict = (Dict*) malloc(sizeof(Dict));
    dict->value = (int*) malloc(sizeof(int) * 5);
    dict->size = 0;
    // 将节点初始化为0
    for (int i = 0; i < 5; ++i) {
        dict->value[i] = 0;
    }
    return dict;
}

void destroyDict(Dict* dict) {
    free(dict);
}

void put(Dict* dict, int value) {
    int index = hash(value);
    dict->value[index] = value;
}

int main() {
    int arr1[5] = {4, 3, 2, 1, 5};
    Dict* d1 = initDict();
    for (int i = 0; i < 5; ++i) {
        put(d1, arr1[i]);
    }
    int index = hash(5);
    printf("%d", d1->value[index]);
    destroyDict(d1);
    return 0;
}

        哈希表的存储也是通过数组来实现的,但是不同与数组按照顺序一个个放入数据,哈希表会通过哈希函数来计算下标,将值放入对应的位置,同样的,取值时,只需要根据哈希函数进行计算,就能得到数组的下标值,所以查找的时间复杂度是o(1)

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是一个简单的哈希表C语言实现代码,使用链地址法解决哈希冲突: ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #define HASHSIZE 12 #define NULLKEY -32768 typedef struct { int *elem; // 数据元素存储基址,动态分配数组 int count; // 当前数据元素个数 } HashTable; int InitHashTable(HashTable *H) { int i; H->count = HASHSIZE; H->elem = (int *)malloc(HASHSIZE * sizeof(int)); for (i = 0; i < HASHSIZE; i++) { H->elem[i] = NULLKEY; } return 1; } int Hash(int key) { return key % HASHSIZE; // 散列函数采用除留余数法 } void InsertHash(HashTable *H, int key) { int addr = Hash(key); // 求哈希地址 while (H->elem[addr] != NULLKEY) { // 如果不为空,则冲突 addr = (addr + 1) % HASHSIZE; // 开放定址法的线性探测 } H->elem[addr] = key; // 直到有空位后插入关键字 } int SearchHash(HashTable H, int key) { int addr = Hash(key); // 求哈希地址 while (H.elem[addr] != key) { // 如果不相等,则冲突 addr = (addr + 1) % HASHSIZE; // 开放定址法的线性探测 if (H.elem[addr] == NULLKEY || addr == Hash(key)) { // 如果循环回到原点或者遇到空位,则说明关键字不存在 return -1; } } return addr; // 找到关键字,返回其地址 } int main() { HashTable H; int i, key, result; InitHashTable(&H); printf("请输入%d个数:\n", HASHSIZE); for (i = 0; i < HASHSIZE; i++) { scanf("%d", &key); InsertHash(&H, key); } printf("哈希表中的数据为:\n"); for (i = 0; i < HASHSIZE; i++) { printf("%d ", H.elem[i]); } printf("\n请输入要查找的数:\n"); scanf("%d", &key); result = SearchHash(H, key); if (result == -1) { printf("没有找到%d\n", key); } else { printf("%d的位置是%d\n", key, result); } return 0; } ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值