数据结构-散列表

散列技术是在记录的存储位置和它的关键字之间建立一个确定的对应关系f,使得每个关键字对应一个存储位置。

采用自定义结构体存储用户的信息包括姓名、电话、住址。用户的电话号码作为该用户信息的关键字。代码如下:

#ifndef HASH_TABLE_H_
#define HASH_TABLE_H_

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

#define HASH_SIZE 12
#define NAME_SIZE 16
#define ADDRESS_SIZE 16

typedef struct{
	unsigned int tel;
	char name[NAME_SIZE];
	char address[ADDRESS_SIZE];
}person;

typedef struct{
	person elem[NAME_SIZE];
	int count;
}Hash,*Hash_Code;
 

void InitHashTable(Hash_Code h);
void SearchHash(Hash_Code h);
void Add_Person(Hash_Code h);
void Show_Person(Hash_Code h);
void Delet_Pereson(Hash_Code h);

#endif

1:初始化散列表:

//初始化散列表
void InitHashTable(Hash_Code h){
	h->count = 0;//用户个数初始化为0
	int i = 0;
	for (i = 0; i < HASH_SIZE; i++){
		h->elem[i].tel = 0;//把散列表中的用户电话均初始化为0
	}
}

2:添加用户,采用除模取余法为每个关键字找到对应的存储位置,当发生冲突时采用线性探测法为关键字继续寻找对应存储位置。代码如下:

//添加用户
void Add_Person(Hash_Code h){
	if (h->count == 12){
		printf("散列表已满,无法添加\n");
		return;
	}
	person p;
	printf("请输入用户电话号码:\n");
	scanf("%u", &p.tel);
	printf("请输入用户姓名:\n");
	scanf(" %s", p.name);
	printf("请输入用户住址:\n");
	scanf("%s", p.address);
	int addr = p.tel % HASH_SIZE;//除模取余法求存储位置
	while (h->elem[addr].tel != 0){//出现冲突
		addr = (addr + 1) % HASH_SIZE;//线性探测存储位置
	}
	if (h->elem[addr].tel == 0){
		h->elem[addr] = p;//添加联系人
		h->count++;
	}
}

2:查找用户,根据用户输入的电话号码继续采用除模取余法得到存储位置,访问该存储位置的用户信息的电话号码若不相同采用线性探测法为关键字继续寻找对应存储位置,当散列表中每个存储位置都访问后无与用户相匹配的信息则判断无该用户。代码如下:

//查找用户
void SearchHash(Hash_Code h){
	if (h->count == 0){
		printf("表为空,无法查找!\n");
		return;
	}
	printf("请输入要查找用户的电话号码:\n");
	unsigned int tele;
	scanf("%u", &tele);
	int addr = tele%HASH_SIZE;//除模取余法求存储位置
	int temp = addr;
	while (h->elem[addr].tel != tele){//出现冲突
		addr = (addr + 1) % HASH_SIZE;//线性探测存储位置
		if (addr == temp){//当遍历到起始存储位置,则表示整个散列表已全部遍历过
			printf("不存在该用户!\n");
			return;
		}
	}
	if (h->elem[addr].tel == tele){
		printf("用户姓名:%s\n", h->elem[addr].name);
		printf("用户电话:%u\n", h->elem[addr].tel);
		printf("用户住址:%s\n", h->elem[addr].address);
	}
}

3:显示全部用户信息:

//显示全部用户信息
void Show_Person(Hash_Code h){
	if (h->count == 0){
		printf("散列表为空!\n");
		return;
	}
	printf("共有%d个用户,用户信息如下:\n", h->count);
	int i = 0;
	for (i = 0; i < HASH_SIZE; i++){
		if (h->elem[i].tel != 0){
			printf("用户姓名:%s  ", h->elem[i].name);
			printf("用户电话:%u  ", h->elem[i].tel);
			printf("用户住址:%s  ", h->elem[i].address);
			printf("\n");
		}
	}
}

4:删除用户:

//删除用户
void Delet_Pereson(Hash_Code h){
    if (h->count == 0){
        printf("表为空!\n");
        return;
    }
    printf("请输入要删除用户的电话号码:\n");
    unsigned int tele;
    scanf("%u", &tele);
    int addr = tele%HASH_SIZE;//除模取余法求存储位置
    int temp = addr;
    while (h->elem[addr].tel != tele){//出现冲突
        addr = (addr + 1) % HASH_SIZE;//线性探测存储位置
        if (addr == temp){//当遍历到起始存储位置,则表示整个散列表已全部遍历过
            printf("不存在该用户!\n");
            return;
        }
    }
    if (h->elem[addr].tel == tele){
        h->elem[addr].tel = 0;
        h->count--;
    }
}

程序执行开始先将散列表中每个元素的关键字初识化为零,将添加联系人和查找联系人功能封装成函数体,每个函数体对应一个选项,将所有功能以索引的形式呈现给用户,当用户选择某一索引时,就执行当前索引对应的函数体。代码如下:

#define _CRT_SECURE_NO_WARNINGS 1
#include "Hash_Table.h"
void Menue(){
	printf("______________\n");
	printf("1:添加用户\n");
	printf("2:查找用户\n");
	printf("3:显示全部用户信息\n");
	printf("4:删除用户\n");
	printf("0:退出\n");
}
int main(){
	Hash h;
	int num;
	InitHashTable(&h);
	int select = 0;
	int quit = 0;
	while (!quit){
		Menue();
		scanf("%d", &select);
		switch (select){
		case 1:
			Add_Person(&h);
			break;
		case 2:
			SearchHash(&h);
			break;
		case 3:
			Show_Person(&h);
			break;
		case 4:
			Delet_Pereson(&h);
			break;
		case 0:
			quit = 1;
			break;
		default:
			break;
		}
	}
	system("pause");
	return 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、付费专栏及课程。

余额充值