iOS标准库中常用数据结构和算法之哈希表

上一篇: iOS标准库中常用数据结构和算法之二叉排序树

?哈希表

系统提供一个全局的key为字符串的哈希表。并提供哈希表的创建、元素添加、元素查找、哈希表的销毁的能力。存储在哈希表中的元素是一个如下的标准结构:

//哈希表元素实体结构定义
typedef	struct entry {
	char *key;   //哈希表中的key,必须是字符串
	void	*data;  //哈希表中的值,是一个指针类型,其内容可以任意。
} ENTRY;
复制代码

一、哈希表的创建和销毁

功能:用于全局哈希表的创建和销毁操作。

头文件:#include <search.h>

平台:BSD Unix

函数签名

//创建一个哈希表
 int hcreate(size_t nel);
//销毁一个哈希表
 void hdestroy(void);
复制代码

参数

nel: [in]指定哈希表的初始容量尺寸,这个参数主要用于内存存储上的优化处理。

return:[out] 如果哈希表创建成功则返回0,否则返回非0。

描述

系统提供了一个全局的哈希表,因此这也是一个非常重要的缺点,因为我们无法知道其他函数是否也正在使用这个哈希表。因此在特定时刻只有一个哈希表是有效的。个人的感觉是这就是一个非常不合理的哈希表实现。

二、哈希表元素的添加和查询。

功能:用于哈希表元素的添加和查询。

头文件:#include <search.h>

平台:BSD Unix

函数签名


ENTRY * hsearch(ENTRY item, ACTION action);

复制代码

参数

item:[in] 要进行查询或者添加的条目,这是一个ENTRY类型的数据。如果我们只是查询则只需要设置ENTRY中的key部分的值,而如果是添加则需要设置完整的key和data的值。

action:[in]指定要对哈希表执行的动作,这个类型是一个ACTION类型的枚举值,其定义如下:

typedef	enum {
	FIND, ENTER
} ACTION;
复制代码

当值设置为FIND时则只进行查找处理。 当值设置为ENTER是就先进行查找,如果不存在时就进行添加处理。

return:[out] 返回查找或者添加时在哈希表中的实体元素的指针。如果没有查找到或者添加失败则返回NULL。我们不需要对返回的ENTRY指针进行内存释放处理,而是由系统来完成。

描述

对哈希表执行ENTER动作时,如果找到了则直接返回以前曾经插入到哈希表中的条目,如果没有找到则会在哈希表中创建一个新的条目,并返回新条目的指针。**这里需要注意的是在执行插入时要求ENTRY结构体中的key部分的内存必须要用malloc进行分配,因为哈希表在销毁时会对所有哈希表中的元素的key部分调用free处理。**也就是说对于哈希表的插入来说key的内存我们负责分配,而由系统负责销毁。这里也存在一个BUG就是当我们对一个在哈希表中已经存在的key再次调用hsearch时会返回对应的ENTRY指针。但是我们无法得知这个返回值到底是新创建的还是已经存在的。而我们的key又是通过malloc分配出来的内存数据,因此就无法确定我们是否需要去释放这部分已经分配出来的key的内存。

示例代码

void main()
{
   //创建
   if (hcreate(10) != 0)
    {
         //插入
        ENTRY ent;
        ent.key = malloc(4);    //对于插入处理必须用malloc进行内存分配,而且我们不需要去释放它。
        ent.data = (int*)10;   //这里值保存着整数类型。
        strcpy(ent.key, "Bob");
        ENTRY *p1 = hsearch(ent, ENTER);
        NSAssert(strcmp(p1->key, "Bob")==0, @"oops!");
        
        ent.key = malloc(6);
        ent.data = (int*)20;
        strcpy(ent.key, "Alice");
        ENTRY *p2 = hsearch(ent, ENTER);
        
       //查找
        ENTRY *p3 = hsearch(ent, FIND);
        NSAssert(p3 == p2);

        //销毁
        hdestroy();
    }
}
复制代码

由于这个哈希表的实现对插入重复元素时存在着BUG,以及又是全局唯一的,所以不建议使用它。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
#include #include #include using namespace std; #define NULL 0 unsigned int key; //用来输入/输出文件流类 unsigned int key2; //key和key2分别是用做了电话号码和姓名的关键字 int *p; struct node //新建节点(用户姓名、地址、电话号码、指向下一个结点的指针 ) { char name[8],address[20]; char num[11]; node * next; }; typedef node* pnode; typedef node* mingzi; //声明了名字和电话两个指针 node **phone; node **nam; node *a; void hash(char num[11]) //以电话号码为关键字建立哈希函数 { int i = 3; key=(int)num[2]; while(num[i]!=NULL) { key+=(int)num[i]; i++; } key=key%20; } void hash2(char name[8]) //姓名为关键字建立哈希函数 { int i = 1; key2=(int)name[0]; while(name[i]!=NULL) { key2+=(int)name[i]; i++; } key2=key2%20; } //强制类型转换,将用户名的每一个字母的ASCLL码值相加并且除以20后的余数 node* input() //输入节点信息 ,建立结点,并将结点的next指针指空 { node *temp; temp = new node; temp->next=NULL; cout<<"输入姓名:"<>temp->name; cout<<"输入地址:"<>temp->address; cout<<"输入电话:"<>temp->num; return temp; } //对于指针类型返回的是地址 int apend() //添加节点 { node *newphone; node *newname; newphone=input(); newname=newphone; newphone->next=NULL; newname->next=NULL; hash(newphone->num); //利用哈希函数计算出对应关键字的存储地址 hash2(newname->name); newphone->next = phone[key]->next; //利用电话号码为关键字插入 phone[key]->next=newphone; //是采用链地址法,拉链法处理冲突的散列表结构 newname->next = nam[key2]->next; //利用用户名为关键字插入 nam[key2]->next=newname; return 0; } void create() //新建节点 { int i; phone=new pnode[20]; //动态创建对象数组,C++课本P188页 for(i=0;inext=NULL; } } void create2() //新建节点 { int i; nam=new mingzi[20]; for(i=0;inext=NULL; } } void list() //显示列表 { int i; node *p; for(i=0;inext; while(p) { cout<name<<'_'<address<<'_'<num<next; } } } void list2() //显示列表 { int i; node *p; for(i=0;inext; while(p) { cout<name<<'_'<address<<'_'<num<next; } } } void find(char num[11]) //在以电话号码为关键字的哈希表查找用户信息 { hash(num); node *q=phone[key]->next; while(q!= NULL) { if(strcmp(num,q->num)==0) break; q=q->next; } if(q) cout<name<<"_" <address<<"_"<num<<endl; else cout<<"无此记录"<next; while(q!= NULL) { if(strcmp(name,q->name)==0) break; q=q->next; } if(q) cout<name<<"_" <address<<"_"<num<<endl; else cout<<"无此记录"<<endl; } void save() //保存用户信息 { int i; node *p; for(i=0;inext; while(p) { fstream iiout("out.txt", ios::out); //创建一个文件流对象:iiout iiout<name<<"_"<address<<"_"<num<next; } } } void menu() //菜单 { cout<<" 哈希表通讯录"<<endl; cout<<" 0.添加记录"<<endl; cout<<" 2.姓名散列"<<endl; cout<<" 3.查找记录"<<endl; cout<<" 4.号码散列"<<endl; cout<<" 5.清空记录"<<endl; cout<<" 6.保存记录"<<endl; cout<<" 7.退出系统"<>sel; if(sel==3) { cout<<"8姓名查询" <<endl;cout<<"9号码查询"<>b; if(b==9) {cout<<"请输入电话号码:"<>num; cout<<"输出查找的信息:"<<endl; find(num); } else {cout<<"请输入姓名:"<>name; cout<<"输出查找的信息:"<<endl; find2(name);}} if(sel==2) {cout<<"姓名散列结果:"<<endl; list2();} if(sel==0) {cout<<"请输入要添加的内容:"<<endl; apend();} if(sel==4) {cout<<"号码散列结果:"<<endl; list(); } if(sel==5) {cout<<"列表已清空:"<<endl; create();create2();} if(sel==6) { cout<<"通信录已保存:"<<endl; save();} if(sel==7) return 0; } return 0; }

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值