精品国产人成在线_亚洲高清无码在线观看_国产在线视频国产永久2021_国产AV综合第一页一个的一区免费影院黑人_最近中文字幕MV高清在线视频

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

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

3天內不再提示

uthash簡介及使用說明

FPGA之家 ? 來源:嵌入式與Linux那些事 ? 作者:嵌入式那些事 ? 2022-08-27 18:04 ? 次閱讀

  • 1. uthash簡介

  • 2. uthash的使用

    • 2.1 定義結構體

    • 2.2 添加

    • 2.3 查找

    • 2.4 替換

    • 2.5 刪除

    • 2.6 循環(huán)刪除

    • 2.7 刪除哈希表所有元素

    • 2.8 計算哈希表元素個數(shù)

    • 2.9 遍歷哈希表中的所有項目

    • 2.10 排序哈希表

    • 2.11 完整代碼

  • 3. 鍵值的各種類型舉例

    • 3.1 整型鍵值

    • 3.2 字符串鍵值

    • 3.3 指針鍵值

    • 3.4 結構體鍵值

  • 4. 常用宏參考

1. uthash簡介

??由于C語言本身不存在哈希,但是當需要使用哈希表的時候自己構建哈希會異常復雜。因此,我們可以調用開源的第三方頭文件,這只是一個頭文件:uthash.h。我們需要做的就是將頭文件復制到項目中,然后:#include "uthash.h"。由于uthash僅是頭文件,因此沒有可鏈接的庫代碼。

??使用uthash添加,查找和刪除通常是常數(shù)時間的操作,此哈希的目標是簡約高效,大約有1000行代碼。

??uthash還包括三個額外的頭文件,主要提供鏈表,動態(tài)數(shù)組和字符串。utlist.h為C結構提供了鏈接列表宏。utarray.h使用宏實現(xiàn)動態(tài)數(shù)組。utstring.h實現(xiàn)基本的動態(tài)字符串。

??github下載鏈接:https://github.com/troydhanson/uthash

2. uthash的使用

2.1 定義結構體

??這里我們將id作為一個索引值,也就是鍵值,將name作為value。

#include"uthash.h"
structmy_struct{
intid;/*鍵值*/
charname[10];
UT_hash_handlehh;/*使能哈希表*/
};
structmy_struct*users=NULL;//*聲明哈希為NULL指針*/

??注意:一定要包含UT_hash_handle hh;hh不需要初始化。它可以命名為任何名稱,但是我們一般都命名為hh。

2.2 添加

??HASH_ADD_INT表示添加的鍵值為int類型。

??HASH_ADD_STR表示添加的鍵值為字符串類型。

??HASH_ADD_PTR表示添加的鍵值為指針類型。

??HASH_ADD表示添加的鍵值可以是任意類型。

voidadd_user(intuser_id,char*name){
structmy_struct*s;
HASH_FIND_INT(users,&user_id,s);/*重復性檢查,當把兩個相同key值的結構體添加到哈希表中時會報錯*/
if(s==NULL){
s=(structmy_struct*)malloc(sizeof*s);///*只有在哈希中不存在ID的情況下,我們才創(chuàng)建該項目并將其添加。否則,我們只修改已經(jīng)存在的結構。*/
s->id=user_id;
HASH_ADD_INT(users,id,s);
}
strcpy(s->name,name);
}

??HASH_ADD_INT函數(shù)中,第一個參數(shù)users是哈希表,第二個參數(shù)id是鍵字段的名稱。最后一個參數(shù)s是指向要添加的結構的指針。

2.3 查找

structmy_struct*find_user(intuser_id){
structmy_struct*s;
s=(structmy_struct*)malloc(sizeof*s);
HASH_FIND_INT(users,&user_id,s);/*s:返回值*/
returns;
}

??在上述代碼中,第一個參數(shù)users是哈希表,第二個參數(shù)是user_id的地址一定要傳遞地址)。最后s是輸出變量。當可以在哈希表中找到相應鍵值時,s返回給定鍵的結構,當找不到時s返回NULL。

2.4 替換

??HASH_REPLACE宏等效于HASH_ADD宏,HASH_REPLACE會嘗試查找和刪除項目外。如果找到并刪除了一個項目,它還將返回該項目的指針作為輸出參數(shù)。

voidreplace_user(HashHead*head,HashNode*newNode){
HashNode*oldNode=find_user(*head,newNode->id);
if(oldNode)
HASH_REPLACE_INT(*head,id,newNode,oldNode);
}

2.5 刪除

??要從哈希表中刪除結構,必須具有指向它的指針。(如果只有鍵,請先執(zhí)行HASH_FIND以獲取結構指針)。

voiddelete_user(structmy_struct*user){
HASH_DEL(users,user);/*user:將要刪除的結構體指針*/
free(user);
}

??同樣,這里users是哈希表,user是指向我們要從哈希中刪除的結構的指針。

??刪除結構只是將其從哈希表中刪除,并非free 。何時釋放結構的選擇完全取決于自己;uthash永遠不會主動釋放結構。

2.6 循環(huán)刪除

??HASH_ITER是一個宏定義,程序執(zhí)行時被替換為一個循環(huán)。

voiddelete_all(){
structmy_struct*current_user,*tmp;

HASH_ITER(hh,users,current_user,tmp){
HASH_DEL(users,current_user);
free(current_user);
}
}

2.7 刪除哈希表所有元素

??如果只想刪除所有項目,但不釋放它們或進行每個元素的清理,則可以通過一次操作更有效地做到這一點:

HASH_CLEAR(hh,users);

??之后,列表頭(此處為users)將設置為NULL。

2.8 計算哈希表元素個數(shù)

unsignedintnum_users;
num_users=HASH_COUNT(users);
printf("thereare%uusers
",num_users);

??當users為NULL時,HASH_COUNT會返回0。

2.9 遍歷哈希表中的所有項目

voidprint_users(){
structmy_struct*s;
for(s=users;s!=NULL;s=s->hh.next){
printf("userid%d:name%s
",s->id,s->name);
}
}

??還有一個hh.prev指針,可用于從任何已知項開始向后迭代哈希。

??由于hh.prev和hh.next字段的緣故,可以在哈希中向前和向后迭代。可以通過遍歷這些指針來訪問哈希中的所有項目,因此哈希也是雙鏈表

2.10 排序哈希表

HASH_SORT(users,name_sort);

??第二個參數(shù)是指向比較函數(shù)的指針。它必須接受兩個指針參數(shù)(要比較的項目),并且如果第一個項目分別在第二個項目之前,等于或之后排序,則必須返回小于零,零或大于零的int。 (這與標準C庫中的strcmp或qsort使用的方法相同)。

intsort_function(void*a,void*b){
/*將a與b比較*/
if(areturn(int)-1;
if(a==b)return(int)0;
if(a>b)return(int)1;

}

??name_sort和id_sort的兩個排序函數(shù)示例。

intname_sort(structmy_struct*a,structmy_struct*b){
returnstrcmp(a->name,b->name);
}

intid_sort(structmy_struct*a,structmy_struct*b){
return(a->id-b->id);
}

voidsort_by_name(){
HASH_SORT(users,name_sort);
}

voidsort_by_id(){
HASH_SORT(users,id_sort);
}

2.11 完整代碼

/*
*@Description:UTHASH的使用
*@Version:V1.0
*@Autor:公眾號【嵌入式Linux那些事】
*@Date:2020-2-22112
*@LastEditors:公眾號【嵌入式與Linux那些事】
*@LastEditTime:2020-2-22246
*/
#include/*gets*/
#include/*atoi,malloc*/
#include/*strcpy*/
#include"uthash.h"

structmy_struct{
intid;/*鍵值*/
charname[10];
UT_hash_handlehh;/*使能結構體*/
};

structmy_struct*users=NULL;

voidadd_user(intuser_id,char*name){
structmy_struct*s;

HASH_FIND_INT(users,&user_id,s);
if(s==NULL){
s=(structmy_struct*)malloc(sizeof*s);
s->id=user_id;
HASH_ADD_INT(users,id,s);
}
strcpy(s->name,name);
}

structmy_struct*find_user(intuser_id){
structmy_struct*s;
s=(structmy_struct*)malloc(sizeof*s);
HASH_FIND_INT(users,&user_id,s);
returns;
}

voiddelete_user(structmy_struct*user){
HASH_DEL(users,user);
free(user);
}

voiddelete_all(){
structmy_struct*current_user,*tmp;

HASH_ITER(hh,users,current_user,tmp){
HASH_DEL(users,current_user);
free(current_user);
}
}

voidprint_users(){
structmy_struct*s;

for(s=users;s!=NULL;s=(structmy_struct*)(s->hh.next)){
printf("userid%d:name%s
",s->id,s->name);
}
}

intname_sort(structmy_struct*a,structmy_struct*b){
returnstrcmp(a->name,b->name);
}

intid_sort(structmy_struct*a,structmy_struct*b){
return(a->id-b->id);
}

voidsort_by_name(){
HASH_SORT(users,name_sort);
}

voidsort_by_id(){
HASH_SORT(users,id_sort);
}

intmain(intargc,char*argv[]){
charin[10];
intid=1,running=1;
structmy_struct*s;
unsignednum_users;

while(running){
printf("1.adduser
");
printf("2.add/renameuserbyid
");
printf("3.finduser
");
printf("4.deleteuser
");
printf("5.deleteallusers
");
printf("6.sortitemsbyname
");
printf("7.sortitemsbyid
");
printf("8.printusers
");
printf("9.countusers
");
printf("10.quit
");
gets(in);
switch(atoi(in)){
case1:
printf("name?
");
add_user(id++,gets(in));
break;
case2:
printf("id?
");
gets(in);id=atoi(in);
printf("name?
");
add_user(id,gets(in));
break;
case3:
printf("id?
");
s=find_user(atoi(gets(in)));
printf("user:%s
",s?s->name:"unknown");
break;
case4:
printf("id?
");
s=find_user(atoi(gets(in)));
if(s)delete_user(s);
elseprintf("idunknown
");
break;
case5:
delete_all();
break;
case6:
sort_by_name();
break;
case7:
sort_by_id();
break;
case8:
print_users();
break;
case9:
num_users=HASH_COUNT(users);
printf("thereare%uusers
",num_users);
break;
case10:
running=0;
break;
}
}

delete_all();
return0;
}

3. 鍵值的各種類型舉例

3.1 整型鍵值

??當鍵值為整型時,可以使用HASH_ADD_INT和HASH_FIND_INT。(對于所有類型的鍵,其他操作(例如HASH_DELETE和)HASH_SORT都是相同的)。

3.2 字符串鍵值

??當鍵值為字符串時,具體要使用那個函數(shù)取決于結構體中的鍵值為字符串數(shù)組還是字符串指針。 這一點很重要。當結構體中的鍵值為字符串數(shù)組時,使用HASH_ADD_STR。鍵值為字符串指針時使用HASH_ADD_KEYPTR。接下來給出兩個例子參考。

??當結構體中的鍵值為字符串數(shù)組時

#include/*strcpy*/
#include/*malloc*/
#include/*printf*/
#include"uthash.h"

structmy_struct{
charname[10];
intid;
UT_hash_handlehh;
};


intmain(intargc,char*argv[]){
constchar*names[]={"joe","bob","betty",NULL};
structmy_struct*s,*tmp,*users=NULL;

for(inti=0;names[i];++i){
s=(structmy_struct*)malloc(sizeof*s);
strcpy(s->name,names[i]);
s->id=i;
HASH_ADD_STR(users,name,s);
}

HASH_FIND_STR(users,"betty",s);
if(s)printf("betty'sidis%d
",s->id);


HASH_ITER(hh,users,s,tmp){
HASH_DEL(users,s);
free(s);
}
return0;
}

??當結構體中的鍵值為字符串指針時

#include/*strcpy*/
#include/*malloc*/
#include/*printf*/
#include"uthash.h"

structmy_struct{
constchar*name;
intid;
UT_hash_handlehh;
};


intmain(intargc,char*argv[]){
constchar*names[]={"joe","bob","betty",NULL};
structmy_struct*s,*tmp,*users=NULL;

for(inti=0;names[i];++i){
s=(structmy_struct*)malloc(sizeof*s);
s->name=names[i];
s->id=i;
HASH_ADD_KEYPTR(hh,users,s->name,strlen(s->name),s);
}

HASH_FIND_STR(users,"betty",s);
if(s)printf("betty'sidis%d
",s->id);

HASH_ITER(hh,users,s,tmp){
HASH_DEL(users,s);
free(s);
}
return0;
}

3.3 指針鍵值

#include
#include
#include"uthash.h"

typedefstruct{
void*key;
inti;
UT_hash_handlehh;
}el_t;

el_t*hash=NULL;
char*someaddr=NULL;

intmain(){
el_t*d;
el_t*e=(el_t*)malloc(sizeof*e);
if(!e)return-1;
e->key=(void*)someaddr;
e->i=1;
HASH_ADD_PTR(hash,key,e);
HASH_FIND_PTR(hash,&someaddr,d);
if(d)printf("found
");

/*releasememory*/
HASH_DEL(hash,e);
free(e);
return0;
}

3.4 結構體鍵值

??在將項目添加到哈希或查找項目之前,必須將結構體鍵值中的元素清零。

#include
#include
#include"uthash.h"

typedefstruct{
chara;
intb;
}record_key_t;

typedefstruct{
record_key_tkey;
UT_hash_handlehh;
}record_t;

intmain(intargc,char*argv[]){
record_tl,*p,*r,*tmp,*records=NULL;

r=(record_t*)malloc(sizeof*r);

memset(r,0,sizeof*r);/*結構體鍵值清零*/
r->key.a='a';
r->key.b=1;
HASH_ADD(hh,records,key,sizeof(record_key_t),r);

memset(&l,0,sizeof(record_t));
l.key.a='a';
l.key.b=1;
HASH_FIND(hh,records,&l.key,sizeof(record_key_t),p);

if(p)printf("found%c%d
",p->key.a,p->key.b);

HASH_ITER(hh,records,p,tmp){
HASH_DEL(records,p);
free(p);
}
return0;
}

4. 常用宏參考

4.1 類型宏

HASH_ADD_INT(head,keyfield_name,item_ptr)

HASH_REPLACE_INT(head,keyfiled_name,item_ptr,replaced_item_ptr)

HASH_FIND_INT(head,key_ptr,item_ptr)

HASH_ADD_STR(head,keyfield_name,item_ptr)

HASH_REPLACE_STR(head,keyfield_name,item_ptr,replaced_item_ptr)

HASH_FIND_STR(head,key_ptr,item_ptr)

HASH_ADD_PTR(head,keyfield_name,item_ptr)

HASH_REPLACE_PTR(head,keyfield_name,item_ptr,replaced_item_ptr)

HASH_FIND_PTR(head,key_ptr,item_ptr)

HASH_DEL(head,item_ptr)

HASH_SORT(head,cmp)

HASH_COUNT(head)

4.2 通用宏

HASH_ADD(hh_name,head,keyfield_name,key_len,item_ptr)

HASH_ADD_BYHASHVALUE(hh_name,head,keyfield_name,key_len,hashv,item_ptr)

HASH_ADD_KEYPTR(hh_name,head,key_ptr,key_len,item_ptr)

HASH_ADD_KEYPTR_BYHASHVALUE(hh_name,head,key_ptr,key_len,hashv,item_ptr)

HASH_ADD_INORDER(hh_name,head,keyfield_name,key_len,item_ptr,cmp)

HASH_ADD_BYHASHVALUE_INORDER(hh_name,head,keyfield_name,key_len,hashv,item_ptr,cmp)

HASH_ADD_KEYPTR_INORDER(hh_name,head,key_ptr,key_len,item_ptr,cmp)

HASH_ADD_KEYPTR_BYHASHVALUE_INORDER(hh_name,head,key_ptr,key_len,hashv,item_ptr,cmp)

HASH_REPLACE(hh_name,head,keyfield_name,key_len,item_ptr,replaced_item_ptr)

HASH_REPLACE_BYHASHVALUE(hh_name,head,keyfield_name,key_len,hashv,item_ptr,replaced_item_ptr)

HASH_REPLACE_INORDER(hh_name,head,keyfield_name,key_len,item_ptr,replaced_item_ptr,cmp)

HASH_REPLACE_BYHASHVALUE_INORDER(hh_name,head,keyfield_name,key_len,hashv,item_ptr,replaced_item_ptr,cmp)

HASH_FIND(hh_name,head,key_ptr,key_len,item_ptr)

HASH_FIND_BYHASHVALUE(hh_name,head,key_ptr,key_len,hashv,item_ptr)

HASH_DELETE(hh_name,head,item_ptr)

HASH_VALUE(key_ptr,key_len,hashv)

HASH_SRT(hh_name,head,cmp)

HASH_CNT(hh_name,head)

HASH_CLEAR(hh_name,head)

HASH_SELECT(dst_hh_name,dst_head,src_hh_name,src_head,condition)

HASH_ITER(hh_name,head,item_ptr,tmp_item_ptr)

HASH_OVERHEAD(hh_name,head)

4.4 參數(shù)說明

??hh_name:UT_hash_handle結構中字段的 名稱。俗稱 hh。

??head:結構指針變量,用作哈希的“頭”。如此命名是因為它最初指向添加到哈希中的第一項。

??keyfield_name:結構中鍵字段的名稱。(對于多字段鍵,這是鍵的第一個字段)。

??key_len:鍵字段的長度(以字節(jié)為單位)。例如,對于整數(shù)鍵,它是sizeof(int),而對于字符串鍵,它是strlen(key)。

??key_ptr:對于HASH_FIND,這是指向要在哈希中查找的鍵的指針(由于它是指針,因此不能在此處直接傳遞文字值)。對于 HASH_ADD_KEYPTR,這是要添加的項的鍵的地址。

??hashv:提供的鍵的哈希值。這是BYHASHVALUE宏的輸入?yún)?shù)。如果要重復查找相同的鍵,則重用緩存的哈希值可以優(yōu)化性能。

??item_ptr:指向要添加,刪除,替換或查找的結構的指針,或迭代期間的當前指針。這是HASH_ADD, HASH_DELETE和HASH_REPLACE宏的輸入?yún)?shù),輸出參數(shù)為HASH_FIND 和HASH_ITER。(當HASH_ITER用于迭代時,tmp_item_ptr 是與item_ptr內部使用的類型相同的另一個變量)。

??replace_item_ptr:用于HASH_REPLACE宏。這是一個輸出參數(shù),設置為指向替換的項目(如果沒有替換的項目,則設置為NULL)。

??cmp:指向比較函數(shù)的指針,該函數(shù)接受兩個參數(shù)(指向要比較的項目的指針),并返回一個int值,該值指定第一個項目應在第二個項目之前,等于還是之后排序(如strcmp)。

??condition:接受單個參數(shù)的函數(shù)或宏(指向結構的空指針,需要將其強制轉換為適當?shù)慕Y構類型)。如果應“選擇”結構以將其添加到目標哈希中,則函數(shù)或宏的值應為非零值。

審核編輯:湯梓紅


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

    關注

    180

    文章

    7598

    瀏覽量

    136210
  • 代碼
    +關注

    關注

    30

    文章

    4750

    瀏覽量

    68357
  • 哈希表
    +關注

    關注

    0

    文章

    9

    瀏覽量

    4834

原文標題:你知道uthash嗎?

文章出處:【微信號:zhuyandz,微信公眾號:FPGA之家】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    PCBNavigator的使用說明

    PCBNavigator的使用說明 
    發(fā)表于 05-11 20:46

    iccavr使用說明

    iccavr使用說明
    發(fā)表于 04-06 11:46

    PCBNavigator的使用說明

    PCBNavigator的使用說明
    發(fā)表于 08-20 16:02

    NI6008使用說明

    NI6008的使用說明
    發(fā)表于 08-31 16:11

    常用的API及使用說明

    為了方便用戶使用,這里列出了常用的API,并給出了相關的使用說明
    發(fā)表于 03-30 06:20

    步進電機模塊使用說明

    FPGA控制_步進電機模塊使用說明今天給大俠帶來步進電機模塊使用說明,話不多說,上貨。一、步進電機簡介步進電機是將電脈沖信號轉變?yōu)榻俏灰苹蚓€位移的開環(huán)控制電機,是現(xiàn)代數(shù)字程序控制系統(tǒng)中的主要執(zhí)行元件
    發(fā)表于 07-07 07:57

    PCA9685使用說明

    舵機驅動板,網(wǎng)上很多資源,但是基于STM32F103的能用代碼實在太少。具體使用說明我覺得這兩個鏈接寫的夠詳細了,附上鏈接1.PCA9685使用說明12.PCA9685使用說明2文字說明
    發(fā)表于 08-11 06:03

    諾基亞Nokia 770手機使用說明

    諾基亞Nokia 770手機使用說明簡介 注意事項 操作說明 設置 .........
    發(fā)表于 09-01 09:57 ?5次下載

    飛利浦X510手機使用說明

    飛利浦X510手機使用說明簡介 使用說明 設置 ...............
    發(fā)表于 09-02 11:44 ?8次下載

    步步高i518手機使用說明

    步步高i518手機使用說明簡介 使用說明 ..............
    發(fā)表于 09-03 16:24 ?61次下載

    步步高K12手機使用說明

    步步高K12手機使用說明簡介 使用說明 操作說明 ...........
    發(fā)表于 09-03 16:25 ?12次下載

    步步高K118手機使用說明

    步步高K118手機使用說明簡介 安全事項 使用說明 ............
    發(fā)表于 09-03 16:25 ?32次下載

    三星SGH-E418手機使用說明

    三星SGH-E418手機使用說明簡介 使用說明 操作 ..........
    發(fā)表于 09-07 12:07 ?10次下載

    C語言uthash簡介的使用

    參考 4.1 類型宏 4.2 通用宏 4.4 參數(shù)說明 1. uthash簡介??由于C語言本身不存在哈希,但是當需要使用哈希表的時候自己構建哈希會異常復雜。因此,我們可以調用開源的第三方
    的頭像 發(fā)表于 03-22 10:44 ?8410次閱讀

    經(jīng)緯儀簡介使用說明

    經(jīng)緯儀簡介使用說明
    發(fā)表于 01-14 11:01 ?2次下載