在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

周立功來講解哈希表的實現

UtFs_Zlgmcu7890 ? 來源:互聯網 ? 作者:佚名 ? 2017-09-30 06:02 ? 次閱讀

近日周立功教授公開了數年的心血之作《程序設計與數據結構》,電子版已無償性分享到電子工程師與高校群體下載,經周立功教授授權,特對本書內容進行連載。

>>>>1.1.1 哈希表的實現

1. 初始化

hash_db_init()接口用于哈希表實例的初始化,在定義哈希表結構體類型時,哈希表數組大小、記錄長度、關鍵字長度和哈希函數都需要由用戶根據實際情況確定,其函數原型定義如下(hash_db.h):

int hash_db_init (

hash_db_t *p_hash, //指向哈希表實例的指針

unsigned int size, //哈希表大小

unsigned int key_len, //關鍵字長度

unsigned int value_len, //記錄長度

hash_func_t pfn_hash); //哈希函數

在這里以學生記錄為例創建一個大小為250組的哈希表

hash_db_t hash_students;

hash_db_init(

&hash_students,

250, //大小為250

6, //關鍵字長度為6字節

sizeof(student_t), //記錄的長度

(hash_func_t)db_id_to_idx); //哈希函數

在初始化函數的實現中,需要按照size指定的大小分配內存,用于存儲哈希表的各個表項鏈表頭),接著需要完成各個鏈表頭和結構體成員的初始化初始化函數的實現范例詳見程序清單3.63。

程序清單3.63初始化函數范例程序

1 int hash_db_init (hash_db_t *p_hash, unsigned int size, unsigned int key_len,

2 unsigned int value_len, hash_func_t pfn_hash)

3 {

4 int i;

5 if ((p_hash == NULL) || (pfn_hash == NULL)){

6 return NULL;

7 }

8 p_hash -> p_head = (slist_head_t *)malloc(size * sizeof(slist_head_t));

9 if (p_hash -> p_head == NULL) {

10 return -1;

11 }

12 for (i = 0; i < size; i++){

13 slist_init(&p_hash -> p_head[i]);

14 }

15 p_hash -> size = size;

16 p_hash -> key_len = key_len;

17 p_hash -> value_len = value_len;

18 p_hash -> pfn_hash = pfn_hash;

19 return 0;

20 }

2. 添加記錄

hash_db_add()接口用于向已經初始化的哈希表中添加一條記錄,添加一條記錄時,需要指定關鍵字信息和記錄值信息,其函數原型定義(hash_db.h):

int hash_db_add (hash_db_t *p_hash, void *key, const void *value);

其中,p_hash為指向哈希表實例的指針,key為指向關鍵字的指針,value為指向記錄值的指針。特別地,由于在添加記錄時,程序不會修改key和value指針所指向的值,因此,指針都加了const修飾符。以添加一條學生記錄為例,使用范例如下:

student_t stu = {

"zhangsan",

'M',

173.3,

60

};

unsigned char id[6] = {0x20, 0x14, 0x44, 0x70, 0x02, 0x39};

hash_db_add(&hash_students, id, &stu);

在添加記錄函數的實現中,首先需要使用哈希函數找到關鍵字對應的記錄在哈希表中的索引,以確定該條記錄所在鏈表的表頭,然后分配一個存儲記錄的結點空間,將關鍵字、記錄等信息存儲在該空間中,然后將結點添加到對應鏈表的頭部(由于記錄在鏈表中的具體位置不重要,因此直接添加在鏈表頭部,效率更高)。函數實現的范例詳見程序清單3.64

程序清單3.64添加記錄函數范例程序

1 int hash_db_add (hash_db_t *p_hash, const void *key, const void *value)

2 {

3 int idx = p_hash -> pfn_hash(key); //使用哈希函數通過關鍵字得到哈希值

4 //分配內存,存儲鏈表結點+關鍵字+記錄

5 char *p_mem = (char *)malloc(sizeof(slist_node_t) + p_hash -> key_len + p_hash -> value_len);

6 if (p_mem == NULL) {

7 return -1;

8 }

9 memcpy(p_mem + sizeof(slist_node_t), key, p_hash -> key_len); //存儲關鍵字

10 memcpy(p_mem + sizeof(slist_node_t) + p_hash->key_len, value, p_hash->value_len); //存儲記錄

11 return slist_add_head(&p_hash -> p_head[idx], (slist_node_t *)p_mem); //將結點加入鏈表

12 }

程序分配了一個結點的空間,該結點的空間需要存儲一個slist_node_t類型鏈表結點,便于添加結點到鏈表中,存儲長度為p_hash->key_len的關鍵字,存儲長度為p_hash->value_len的記錄值,詳見圖3.26,其內存的大小為

sizeof(slist_node_t) + p_hash -> key_len + p_hash -> value_len

圖3.26 結點存儲空間

由于結點空間的首部用于存儲結點slist_node_t的值以組織鏈表。因此需要將結點添加到鏈表中時,直接將p_mem轉換為slist_node_t*類型使用即可,通用鏈式哈希表的結構示意圖詳見圖3.27

圖3.27 通用的鏈式哈希表結構示意圖

圖3.25中管理學生記錄的鏈式哈希表結構示意圖對比發現,它們表達的含義是完全一致的,僅僅是具體類型變為了更加通用的void *類型。

3. 查找記錄

hash_db_search()接口通過關鍵字查找與之對應的記錄,查找記錄時,需要指定關鍵字信息,同時還需要使用一個指向記錄的指針獲取查找到的記錄值,其函數原型(hash_db.h)如下:

int hash_db_search(hash_db_t *p_hash,const void *key, void *value);

雖然參數與添加記錄是完全一樣的,但value表示的含義卻不一樣,此處的value是輸出參數,用于得到查找到的記錄值。而添加記錄函數中的value是輸入參數,提供需要存儲的記錄值。由于此處的value指向指向的值是需要被改變的(改變為查找到的記錄值),因此,其不能增加const修飾符。以查找ID為201444700239的學生記錄為例,使用范例如下:

student_t stu;

unsigned char id[6] = {0x20, 0x14, 0x44, 0x70, 0x02, 0x39};

if (hash_db_search(&hash_students, id, &stu) == 0) {

//查找到該學號的學生記錄

} else {

//查找失敗,未找到該學號的學生記錄

}

在該函數的實現中,首先需要使用哈希函數找到關鍵字對應的記錄在哈希表中的索引,以確定該條記錄所在鏈表的表頭,然后遍歷鏈表的各個結點,將提供的關鍵字與結點中存儲的關鍵字比對,直到找到關鍵字完全一致的記錄(查找成功)或鏈表遍歷結束(查找失敗)。找到該記錄對應的結點后,將結點中存儲的value值拷貝到參數value指針指向的空間中即可。函數實現的范例詳見程序清單3.65。

程序清單3.65查找記錄函數范例程序

1 //尋找結點的上下文(僅內部使用)

2 struct _node_find_ctx {

3 void *key; //查找關鍵字

4 unsigned int key_len; //關鍵字長度

5 slist_node_t *p_result; //用于存儲查找到的結點

6 };

7

8 //遍歷鏈表的回調函數,查找指定結點

9 static int __hash_db_node_find (void *p_arg, slist_node_t *p_node)

10 {

11 struct _node_find_ctx *p_info = (struct _node_find_ctx *)p_arg; //用戶參數為尋找結點的上下文

12 char *p_mem = (char *)p_node + sizeof(slist_node_t); //關鍵字存儲在結點之后

13

14 if (memcmp(p_mem, p_info->key, p_info->key_len) == 0) {

15 p_info->p_result = p_node;

16 return -1; //找到該結點,終止遍歷

17 }

18 return 0;

19 }

20

21 int hash_db_search(hash_db_t *p_hash, const void *key, void *value)

22 {

23 int idx = p_hash->pfn_hash(key); //得到關鍵字對應的哈希表的索引

24 struct _node_find_ctx info = {key, p_hash->key_len, NULL}; //設置遍歷鏈表的上下文信息

25 slist_foreach(&p_hash->p_head[idx], __hash_db_node_find, &info); //遍歷,尋找關鍵字對應結點

26

27 if (info.p_result != NULL) { //找到對應結點, 將存儲的記錄值拷貝到用戶提供的空間中

28 memcpy(value, (char *)info.p_result+sizeof(slist_node_t)+p_hash->key_len+p_hash->value_len);

29 return 0;

30 }

31 return -1;

32 }

程序中,由于查找結點時需要遍歷鏈表,關鍵字比對的操作需要在遍歷函數的回調函數中完成,因此,需要將用戶查找記錄使用的關鍵字信息(關鍵字及其長度)提供給回調函數,同時,當查找到記錄時,需要將查找到的結點反饋給調用遍歷函數的主程序。為此,定義了一個內部使用的用于尋找一個結點的上下文結構體:

struct _node_find_ctx {

const void *key; //查找關鍵字

unsigned int key_len; //關鍵字長度

slist_node_t *p_result; //用于存儲查找到的結點

};

調用遍歷函數時,需要提供一個設置好關鍵字信息的結構體作為回調函數的用戶參數。遍歷函數結束時,可以通過該結構體中的p_result成員獲取遍歷結果。

4. 刪除記錄

該接口用于刪除指定關鍵字對應的記錄,可以定義其函數名為:hash_db_del()。刪除記錄時,需要指定關鍵字信息。可以定義函數的原型為:

int hash_db_del(hash_db_t *p_hash, const void *key);

以刪除學號為201444700239的學生記錄為例,使用范例如下:

unsigned char id[6] = {0x20, 0x14, 0x44, 0x70, 0x02, 0x39};

hash_db_del(&hash_students, id);

在該函數的實現中,絕大部分操作與查找記錄是相同的,唯一的不同是,當找到關鍵字對應的結點時,不再需要將記錄值提取出來,直接將該結點刪除即可。函數實現的范例詳見程序清單3.66。

程序清單3.66刪除記錄函數范例程序

1 int hash_db_del (hash_db_t *p_hash, const void *key)

2 {

3 int idx = p_hash->pfn_hash(key); //得到關鍵字對應的哈希表的索引

4 struct _node_find_ctx info = {key, p_hash->key_len, NULL}; //設置遍歷鏈表的上下文信息

5 slist_foreach(&p_hash->p_head[idx], __hash_db_node_find, &info); //遍歷,尋找關鍵字對應結點

6 if (info.p_result != NULL) {

7 slist_del(&p_hash->p_head[idx], info.p_result); //從鏈表中刪除該結點

8 free(info.p_result); //釋放結點空間

9 return 0;

10 }

11 return -1;

12 }

5. 解初始化

對應于哈希表的初始化,用于當不再使用哈希表時,釋放相關的空間。可以定義其函數名為:hash_db_deinit()。需要通過參數指定需要解初始化的哈希表實例,可以定義函數的原型為(hash_db.h):

int hash_db_deinit (hash_db_t *p_hash);

如不再使用學生信息管理系統,則需使用解初始化函數釋放哈希表的相關資源,使用范例如下:

hash_db_deinit(&hash_students);

在該函數的實現中,需要釋放程序中分配的所有空間,主要包括添加記錄時分配的結點空間,鏈表頭結點數組空間。函數實現詳見程序清單3.67。

程序清單3.67解初始化函數范例程序

1 int hash_db_deinit (hash_db_t *p_hash)

2 {

3 int i;

4 slist_node_t *p_node;

5 for (i = 0; i < p_hash->size; i++) { //釋放哈希表中各個表項中存儲的所有結點

6

7 while (slist_begin_get(&p_hash->p_head[i]) != slist_end_get(&p_hash->p_head[i])) {

8 p_node = slist_begin_get(&p_hash->p_head[i]);

9 slist_del(&p_hash->p_head[i], p_node); //刪除第一個結點

10 free(p_node);

11 }

12 }

13 free(p_hash->p_head); //釋放鏈表頭結點數組空間

15 return 0;

16 }

為便于查閱,如程序清單3.29所示展示了hash_db.h文件的內容。

程序清單3.68 hash_db.h文件內容

1 #pragma once;

2 #include "slist.h"

3

4 typedef unsigned int (*hash_func_t) (const void *key); //哈希函數類型,返回值為整數,參數為關鍵字

5 struct _hash_db{

6 slist_head_t *p_head; //指向數組首地址

7 unsigned int size; //數組成員數

8 unsigned int value_len; //一條記錄的長度

9 unsigned int key_len; //關鍵字的長度

10 hash_func_t pfn_hash; //哈希函數

11 };

12 typedef struct _hash_db *hash_db_t; //指向哈希表對象的指針類型

13

14 int hash_db_init (hash_db_t *p_hash, // 哈希表初始化

15 unsigned int size,

16 unsigned int key_len,

17 unsigned int value_len,

18 hash_func_t pfn_hash);

19

20 int hash_db_add (hash_db_t *p_hash, const void *key,const void *value); //添加記錄

21 int hash_db_del (hash_db_t *p_hash, const void *key); //刪除記錄

22 int hash_db_search(hash_db_t *p_hash, const void *key, void *value); // 查找記錄

23 int hash_db_deinit (hash_db_t *p_hash); //解初始化

以使用該鏈式哈希表管理系統來管理學生記錄為例綜合范例程序詳見程序清單3.30。

程序清單3.69哈希表綜合范例程序

1 #include

2 #include

3 #include "hash_db.h"

4

5 typedef struct _student{

6 char name[10]; //姓名

7 char sex; //性別

8 float height, weight; //身高、體重

9 } student_t;

10

11 int db_id_to_idx (unsigned char id[6]) //通過ID得到數組索引

12 {

13 int i;

14 int sum = 0;

15 for (i = 0; i < 6; i++){

16 sum += id[0];

17 }

18 return sum % 250;

19 }

20

21 int student_info_generate (unsigned char *p_id, student_t *p_student) //隨機產生一條學生記錄

22 {

23 int i;

24 for (i = 0; i < 6; i++) {?????????????????????????? ?? //?隨機產生一個學號

25 p_id[i] = rand();

26 }

27 for (i = 0; i < 9; i++) {????????????????????????? ???? //?隨機名字,由 'a' ~ 'z' 組成

28 p_student->name[i] = (rand() % ('z' - 'a')) + 'a';

29 }

30 p_student->name[i]= '\0'; //字符串結束符

31 p_student->sex = (rand() & 0x01) ? 'F' : 'M'; //隨機性別

32 p_student->height = (float)rand() / rand();

33 p_student->weight = (float)rand() / rand();

34 return 0;

35 }

36

37 int main ()

38 {

39 student_t stu;

40 unsigned char id[6];

41 int i;

42 hash_db_t hash_students;

43

44 hash_db_init(&hash_students, 250, 6, sizeof(student_t), (hash_func_t)db_id_to_idx);

45

46 for (i = 0; i < 100; i++) {???????????? ?????????? //?添加100個學生的信息

47 student_info_generate(id, &stu); //設置學生的信息,當前一隨機數作為測試

48 if (hash_db_search(&hash_students, id, &stu) == 0) { //查找到已經存在該ID的學生記錄

49 printf("該ID的記錄已經存在!\n");

50 continue;

51 }

52 printf("增加記錄:ID : %02x%02x%02x%02x%02x%02x",id[0],id[1],id[2],id[3],id[4],id[5]);

53 printf("信息: %s %c %.2f %.2f\n", stu.name, stu.sex, stu.height, stu.weight);

54 if (hash_db_add(&hash_students, id, &stu) != 0) {

55 printf("添加失敗");

56 }

57 }

58

59 printf("查找ID為:%02x%02x%02x%02x%02x%02x的信息\n",id[0],id[1],id[2],id[3],id[4],id[5]);

60 if (hash_db_search(&hash_students, id, &stu) == 0) {

61 printf("學生信息: %s %c %.2f %.2f\n", stu.name, stu.sex, stu.height, stu.weight);

62 } else {

63 printf("未找到該ID的記錄!\r\n");

64 }

65 hash_db_deinit(&hash_students);

66 return 0;

67 }

在這里,首先創建了一個哈希表,然后向其中添加了100個學生信息(隨機數的方式產生的),接著查找了ID對應的學生信息(這里的ID沒有特別設置,即查找最后添加的學生記錄),最后釋放哈希表。

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 周立功
    +關注

    關注

    38

    文章

    130

    瀏覽量

    37656
  • 大數據
    +關注

    關注

    64

    文章

    8893

    瀏覽量

    137471
  • 哈希表
    +關注

    關注

    0

    文章

    9

    瀏覽量

    4852

原文標題:周立功:哈希表的實現,干貨!

文章出處:【微信號:Zlgmcu7890,微信公眾號:周立功單片機】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    立功科技與求遠電子推出全新藍牙汽車數字鑰匙方案

    藍牙6.0更新帶來了高精度的CS(Channel Sounding)測距功能,為當前的藍牙定位市場注入一股強勁動力。立功科技·求遠電子推出CS+多節點監聽的全新藍牙汽車數字鑰匙方案,讓絲滑的無感解鎖刷新您的出行體驗。
    的頭像 發表于 12-10 16:39 ?495次閱讀
    <b class='flag-5'>立功</b>科技與求遠電子推出全新藍牙汽車數字鑰匙方案

    芯片封裝工藝詳細講解

    芯片封裝工藝詳細講解
    發表于 11-29 14:02 ?1次下載

    立功科技汽車防盜方案介紹

    汽車防盜產品多種多樣,目前主流的汽車防盜系統之一是PEPS(Passive Entry Passive Start)系統,車主無需掏出鑰匙就能直接進入車輛和啟動引擎,極大方便了車主的操作。同時,防盜芯片種類繁多,應用場景也多樣化。本文將從不同的應用場景出發,介紹立功科技的幾種汽車防盜方案。
    的頭像 發表于 11-25 15:44 ?357次閱讀
    <b class='flag-5'>立功</b>科技汽車防盜方案介紹

    華納云:Chord算法如何管理節點間的聯系?

    Chord算法是一種分布式哈希(DHT)協議,它通過構建一個環狀結構來管理節點間的聯系。以下是Chord算法如何管理節點間聯系的具體方式: 環狀結構: Chord算法將所有節點和鍵映射到一個環狀
    發表于 11-08 16:03

    SATA主機協議的物理層的實現過程

    這里講解SATA主機協議的物理層的實現過程。
    的頭像 發表于 10-22 15:17 ?313次閱讀
    SATA主機協議的物理層的<b class='flag-5'>實現</b>過程

    干貨篇:低功耗4G模組Air780E的串口通信

    ? 今天我們來講解低功耗4G模組Air780E的串口通信的基本用法,小伙伴們,學起來吧!
    的頭像 發表于 10-05 14:38 ?571次閱讀
    干貨篇:低功耗4G模組Air780E的串口通信

    電感技術的講解

    詳細講解電感的原理及計算
    的頭像 發表于 09-06 02:07 ?2207次閱讀
    電感技術的<b class='flag-5'>講解</b>

    立功科技ISD智能交互車燈技術方案

    隨著智能汽車的快速發展,車燈產業正在經歷從功能車燈向智能車燈轉型發展,ISD智能交互車燈憑借成熟的產業鏈以及不斷升級的技術方案,正逐步成為市場主流。本文為大家介紹立功科技ISD智能交互車燈技術方案。
    的頭像 發表于 07-18 14:26 ?1093次閱讀
    <b class='flag-5'>立功</b>科技ISD智能交互車燈技術方案

    中國電子黨組副書記、總經理李立功調研CET中電技術馬來西亞數據中心項目

    為進一步落實中央企業“走出去”戰略,助力構建“雙循環”新發展格局,服務共建“一帶一路”高質量發展,6月22日,中國電子黨組副書記、總經理李立功率隊赴馬來西亞項目施工現場調研。調研期間,李立功通過實地
    的頭像 發表于 07-06 08:36 ?549次閱讀
    中國電子黨組副書記、總經理李<b class='flag-5'>立功</b>調研CET中電技術馬來西亞數據中心項目

    foxbot基本操作與應用講解

    電子發燒友網站提供《foxbot基本操作與應用講解.pptx》資料免費下載
    發表于 05-11 09:34 ?1次下載

    鴻蒙ArkTS開始實例:【canvas實現簽名板功能】

    使用ArkTS中的canvas實現簽名板的功能,canvas畫布大家都很熟悉,我們會用它經常實現一些畫板或者圖表、表格之類的功能。canvas簽名板是我在開發APP過程中實現的一個功能,開發過程中也是遇到比較多的問題。我會按照以
    的頭像 發表于 04-08 10:10 ?954次閱讀
    鴻蒙ArkTS開始實例:【canvas<b class='flag-5'>實現</b>簽名板功能】

    樂華工業顯示器日常應該如何維護

    今天我來講解一下工業顯示器的日常使用和維護,良好的維護可以使工業顯示器的使用壽命延長和穩定可靠的運行,因此工業顯示器日常的維護就顯得非常重要。那么下面我就來講解一下它的幾種維護方式:
    的頭像 發表于 03-28 10:55 ?316次閱讀
    樂華工業顯示器日常應該如何維護

    毫厘中的絢爛綻放,先楫攜手立功科技發布HPM6800數字儀表方案

    機界面應用平臺。廣州立功科技股份有限公司(立功科技,GZLG)基于先楫高性能HPM6800MCU搭載AWTKGUI組件開發的全新汽車液晶儀表解決方案,使用RTOS系統滿足開機
    的頭像 發表于 03-14 08:16 ?589次閱讀
    毫厘中的絢爛綻放,先楫攜手<b class='flag-5'>立功</b>科技發布HPM6800數字儀表方案

    2024年第9理想汽車銷量0.62萬輛

    2024年第9(2.26-3.3),理想汽車銷量0.62萬輛,位居新勢力品牌銷量第二。
    的頭像 發表于 03-05 18:25 ?1566次閱讀
    2024年第9<b class='flag-5'>周</b>理想汽車<b class='flag-5'>周</b>銷量0.62萬輛

    開關電源電感電壓波形過沖和下沖原理

    以一個同步降壓電路例子來講解。
    的頭像 發表于 01-22 14:06 ?2643次閱讀
    開關電源電感電壓波形過沖和下沖原理
    主站蜘蛛池模板: 国产巨大bbbb俄罗斯| 天天操天天舔天天射| 一区二区三区四区免费视频| 国产成人精品视频一区二区不卡| 色网站免费看| 99热都是精品| 国内精品久久久久影院男同志| 天天插天天舔| 美女喷白浆| 天天干天天色天天干| 成人av.com| 久久婷婷色一区二区三区| 日韩一级欧美一级一级国产| 亚洲精品电影天堂网| 69堂在线观看国产成人| 亚洲激情五月| 色射色| 日本在线一级| 色天使色护士 在线视频观看| 日韩毛片免费在线观看| 久青草视频在线播放| 七月婷婷在线视频综合| 特级毛片免费视频| 五月综合激情久久婷婷| 亚洲国内精品自在线影视| 色在线视频观看| 噜噜色噜噜| 久久国产高清视频| 精品一区二区三区免费爱| 操你啦在线视频| 美女操出水| 午夜看片在线| 午夜看片影院在线观看| 黄网观看| 2021国产成人午夜精品| 午夜黄色网址| 欧美同性精品xxxx| 中文字幕第二区| 欧美性xxxxxbbbbbb精品| 国产黄色a三级三级三级| 中国业余老太性视频|