数据结构课程设计------c实现散列表(二次探测再哈希)电话簿(文件存储)

题目二 :散列表的设计与实现

2.1问题描述

设计散列表实现电话号码查找系统,使得平均查找长度不超过2

基本要求
(1)设每个记录有下列数据项:电话号码、用户名、地址;
(2)从键盘输入各记录,以电话号码为关键字建立散列表;
(3)采用一定的方法解决冲突;
(4)查找并显示给定电话号码的记录;

2.2.算法设计与分析

2.2.1设计思路分析
基本思路
1构建哈希表(需要构造一个哈希函数,确定一个合适的处理冲突的方法)
2按照电话号码查找并输出
3将哈希表按序存进文件中
基本思想
1 哈希表是根据关键码而直接进行访问的数据结构,也就是说,它通过把关键码值映射到表中一个位置来访问记录,以加快查找的速度。这个映射函数叫做散列函数,存放记录的数组叫做散列表
2给定表M,存在函数f(key),对任意给定的关键子值key,代入函数后能得到包含该关键字的记录在表中的地址,则称表M为哈希表,函数f(key)为哈希函数。
3 二次探测再散列法:若发生冲突,则按照+1²,-1²,+2²,-2²……方式进行探测再散列的方法。

2.2.2设计程序流程图
在这里插入图片描述
进入程序,进行功能选择,选择1进行存入电话号码,构建哈希表,如果存在冲突解决冲突再录入,不存在直接录入信息。选择0查找电话号码,怎么解决冲突,就按相应的方法查找。所有功能执行完后返回,选择0退出程序,程序结束。
2.2.3数据结构定义

typedef struct record				//定义一条记录的结构体
{char Number[20];				char Name[20];char Address[20];
}Record;typedef struct Hash					//定义一个散列表
{Record *data;					//存访多条记录的数组int cnt;						int size;						//数组大小
}*HashTable, HashElem;

2.2.4算法的时间复杂度分析
整个程序的时间复杂度看哈希表在解决冲突时的时间复杂度
解决冲突
最差的时间复杂度O(n),当所有数据被填满时
最好的时间复杂度O(1)没有冲突的时候.

2.3源程序清单

hash.h文件

#define _CRT_SECURE_NO_WARNINGS
#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>#define MAXSIZE 50using namespace std;				typedef struct record				//定义一条记录的结构体
{char Number[20];				char Name[20];char Address[20];
}Record;typedef struct Hash					//定义一个散列表
{Record *data;					//存访多条记录的数组int cnt;						int size;						//数组大小
}*HashTable, HashElem;void savenumb(HashTable numbertable, Record record[]);		//存访函数
void findnumb(HashTable numbertable);						//查询函数

function.cpp文件

#define _CRT_SECURE_NO_WARNINGS
#include"hash.h"int Czy = 1;
//哈希函数,将电话号码每一位求和 
int GetHashKey(char ar[])
{int len = strlen(ar);						//计算电话号码的长度int key = 0;for (int i = 0; i<len; i++){					key += ar[i] - '0';						//key=总和,数字字符减'0'就是数字}return key%MAXSIZE;//必须取模,否则下标越界		//返回得到的地址
}//冲突处理,二次探测再散列 
int HandleCollision(HashTable table, int key)
{Czy = 1; //从2,3,4,5,....... while (1){Czy++; //从2,3,4,5,....... if (Czy % 2 == 0) {										//偶数和偶数下一个数奇数除二的值相等if (table->data[(key + (Czy / 2)*(Czy / 2)) % MAXSIZE].Name[0] == 0)	//这个位置上没有数据	return (key + (Czy / 2)*(Czy / 2)) % MAXSIZE;						//返回这个位置上的地址}else if (Czy % 2 != 0) {if ((key - (Czy / 2)*(Czy / 2))<0) continue;                             //由于是减法,要注意负数不能取模 if (table->data[(key - (Czy / 2)*(Czy / 2)) % MAXSIZE].Name[0] == 0)	//如果这个位置上没有数据return (key - (Czy / 2)*(Czy / 2)) % MAXSIZE;}}//return -1;
}//构建哈希表 
void CreateHashTable(HashTable &table, Record *record, int n)
{int key;for (int i = 0; i<n; i++){key = GetHashKey(record[i].Number);				//接受每个电话返回来的地址if (table->data[key].Name[0] != 0)				//当这个地址里的名字不为空时,也就是有冲突了key = HandleCollision(table, key);			//进行冲突处理  传冲突的地址//如果不冲突,则进行赋值strcpy(table->data[key].Number, record[i].Number);	strcpy(table->data[key].Name, record[i].Name);strcpy(table->data[key].Address, record[i].Address);}
}//按照电话号码寻找 
int SerchKey(HashTable table, char PhoneNumber[])
{int key = GetHashKey(PhoneNumber);						//得到该元素的最初地址位置if (strcmp(table->data[key].Number, PhoneNumber)){		//如果该位置上的数与给定的数不相等,则考虑冲突寻址for (Czy = 1; Czy < MAXSIZE; Czy++){if (Czy % 2 == 0) {//strcmp(str1,str2)//当str1的字典序大于str2时返回一个一个正数 小于负数,等于0if (!strcmp(PhoneNumber, table->data[(key + (Czy / 2)*(Czy / 2)) % MAXSIZE].Number)){//如果相等则地址就等于冲突处理后的值key = (key + (Czy / 2)*(Czy / 2)) % MAXSIZE;break;}}else if (Czy % 2 != 0) {if ((key - (Czy / 2)*(Czy / 2)) < 0) continue;//由于是减法,要注意负数不能取模			//如果取差后为负数,则不做处理,Czy+1//不是负数,判断值是否与当前地址值相等if (!strcmp(PhoneNumber, table->data[(key - (Czy / 2)*(Czy / 2)) % MAXSIZE].Number)){key = (key - (Czy / 2)*(Czy / 2)) % MAXSIZE;break;}}}}return key;
}
//将哈希表存入文件中 void GoToFile(HashTable table)
{FILE *fp = fopen("Output.txt", "w");				//打开文件Output.txtfor (int i = 0; i <= MAXSIZE; i++)					//遍历整个散列表if (table->data[i].Name[0] != 0)					//如果名字不为空,则写入文件fprintf(fp, "%s %s %s\n", table->data[i].Name, table->data[i].Number, table->data[i].Address);//printf("%s %s %s\n",table->data[i].Name,table->data[i].Number,table->data[i].Address);	fclose(fp);											//关闭文件
}void findnumb(HashTable numbertable){				//查询函数int key = 0;//输入并寻找PhoneNumber(必须存在表中) char PhoneNumber[20];printf("请输入你想找的电话:\n");cin >> PhoneNumber;cout << "给定电话号码为:" << endl << PhoneNumber << endl;key= SerchKey(numbertable, PhoneNumber);if (!strcmp(numbertable->data[key].Number, PhoneNumber)){printf("找到的电话信息为:\n");}else{printf("\n没有该电话号码\n");return;}cout << numbertable->data[key].Name << " " << numbertable->data[key].Number << " " << numbertable->data[key].Address << endl;}void savenumb(HashTable numbertable,Record record[]){			//保存函数int k;//输入数据 组数及 各个数据 //freopen("Data.txt","r",stdin);printf("你想要存入几个人的电话\n");cin >> k;while(k > 10){printf("一次至多只能存放10人\n");printf("请重新输入:\n");cin >> k;}for (int i = 0; i < k; i++){printf("请输入电话\n");cin >> record[i].Number;printf("请输入姓名\n");cin >> record[i].Name;printf("请输入住址\n");cin >> record[i].Address;}//创建哈希表 CreateHashTable(numbertable, record, k);//存入文件中 GoToFile(numbertable);printf("保存成功\n");}

Main.cpp文件

#include"hash.h"
void menu2(){printf("****************************\n");printf("*欢迎使用电话簿(散列表版本)*\n");printf("****** 1存入电话号码 *******\n");printf("****** 2查找电话号码 *******\n");printf("****** 0 退出程序    *******\n");printf("****************************\n");
}void menu(){//定义及初始化 Record record[50];			//定义结构体数组,可以存放50个元素的信息HashElem table;				//散列表结构体变量HashTable numbertable;		//散列表结构体指针numbertable = &table;		//给结构体指针赋初值numbertable->data = (Record*)malloc(sizeof(record[0])*MAXSIZE);memset(numbertable->data, 0, sizeof(record[0])*MAXSIZE);numbertable->size = MAXSIZE;numbertable->cnt = 0;int input;do{menu2();printf("请输入你想要执行的操作\n");scanf("%d", &input);switch (input){case 1:savenumb(numbertable, record);break;case 2:findnumb(numbertable);break;case 0:printf("再见\n");break;default:printf("你的输入有误请重新输入\n");break;}} while (input);}int main(){menu();system("pause");return 0;
}

2.4执行结果

1存入电话号码

在这里插入图片描述

2查找电话号码

在这里插入图片描述
0退出程序
在这里插入图片描述

2.5存在问题分析

  1. 电话簿中无法分辨电话相同的情况。
    2 电话簿没有一个图形化界面,用户需要手动输入相关命令,无法实现鼠标点击

2.6结论

基本实现了电话簿的添加和查找功能

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/383345.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

科技论文----论搜索引擎现状及发展趋势

搜索引擎现状及发展趋势 【摘要】 随着最近10年中国互联网的快速发展菜互联网已经彻底改变了人们的生活方式&#xff0c;而在互联网的发展过程中。搜索引擎发挥了巨大的推动作用。本文对搜索引擎的发展历史采用的技术&#xff0c;发展现状出现的问题以及未来发展方向进行了综述…

数据结构课程设计------扫雷游戏(升级版,可展开)

本程序由团队中的一个人所写&#xff0c;本人看懂并写下此文章 题目&#xff1a;扫雷 3.1问题描述 扫雷游戏 [基本要求] &#xff08;1&#xff09;完成棋盘的初始化并在标准显示器中显示 &#xff08;2&#xff09;通过输入行列值确定用户输入 &#xff08;3&#xff09;游…

C语言的编译链接过程的介绍

发布时间: 2012-11-08 10:17 作者: 未知 来源: 51Testing软件测试网采编 字体: 小 中 大 | 上一篇 下一篇 | 打印 | 我要投稿 | 推荐标签&#xff1a; DotNet 软件开发 | 感言十年 C语言的编译链接过程要把我们编写的一个c程序&#xff08;源代码&#x…

技术与技巧札记

Linux常用命令及技巧&#xff1a; &#xff08;1&#xff09;cat /proc/version 查看当前内核的版本 (2) 挂载nfs文件夹&#xff1a;需要先确认在&#xff0f;etc&#xff0f;exports文件&#xff0c;可以用于开发板挂载的文件夹 mount -o nolock 10.0.22.30:/root/sharednfs …

从0开始python学习-35.allure报告企业定制

目录 1. 搭建allure环境 2. 生成报告 3. logo定制 4. 企业级报告内容或层级定制 5. allure局域网查看 1. 搭建allure环境 1.1 JDK&#xff0c;使用PyCharm 找到pycharm安装目录找到java.exe记下jbr目录的完整路径&#xff0c;eg: C:\Program Files\JetBrains\PyCharm Com…

系统架构札记

什么是高内聚、低耦合&#xff1f; 起因&#xff1a;模块独立性指每个模块只完成系统要求的独立子功能&#xff0c;并且与其他模块的联系最少且接口简单&#xff0c;两个定性的度量标准――耦合性和内聚性。 耦合性也称块间联系。指软件系统结构中各模块间相互联系紧密程度的一…

c++中的继承--1(引出,继承方式,继承的对象模型)

继承的引出 概念&#xff1a; 继承(inheritance)机制是面向对象程序设计使代码可以复用的最重要的手段&#xff0c;它允许程序员在保持原有类特 性的基础上进行扩展&#xff0c;增加功能&#xff0c;这样产生新的类&#xff0c;称派生类。继承呈现了面向对象程序设计的层次结构…

c++中的多态---1(多态概念,静态联编和动态联编,多态原理解析,重载,重写,重定义的对比)

多态的基本概念 多态是面向对象设计语言数据抽象和继承之外的第三个基本特征多态性(polymorphism)提供接口与具体实现之间的另一层隔膜&#xff0c;从而将“what”和“how”分离开来&#xff0c;多态性改善了代码的可读和组织性&#xff0c;同时也使创建的程序具有可扩展性&am…

Ubuntu下各种服务搭建及操作技巧

Ubuntu下搭建TFTP 1、安装软件包 sudo apt-get install tftpd tftp xinetd 2、建立配置文件 在/etc/xinetd.d/下建立一个配置文件tftp sudo vi /etc/xinetd.d/tftp 内容如下 service tftp { socket_type dgram protocol udp wait yes user root …

使用Automake和Autoconf生成Makefile

automake 所产生的 Makefile 除了可以做到程序的自动编译和链接 外&#xff0c;还可以用来生成各种文档&#xff08;如manual page、info文件&#xff09;&#xff0c;可以将源代码文件包装起来以供发布。所以程序源代码所存放的目录 结构最好符合GNU的标准惯例。下面以hello.…

c++中多态---3(虚析构和纯虚析构,向上类型转化和向下类型转化)

虚析构和纯虚析构 虚析构virtual ~类名(){}类内声明&#xff0c;类内实现解决问题&#xff1a;通过父类指针指向子类对象释放时候不干净的问题 纯虚析构 写法 virtual ~类名(){}0; 类内声明 类外实现 如果出现了纯虚析构函数&#xff0c;这个类也算是抽象类&#xff0c;不可…

嵌入式开发硬件知识札记

三态逻辑 1. 概念 三态指其输出既可以是一般二值逻辑电路&#xff0c;即正常的高电平&#xff08;逻辑1&#xff09;或低电平&#xff08;逻辑0&#xff09;&#xff0c;又可以保持特有的高阻抗状态。高阻态相当于隔断状态&#xff08;电阻很大&#xff0c;相当于开路&#xff…

《凡人修仙传》中打斗场景(c++多态实现)

我们 要实现打斗场景&#xff0c;第一&#xff0c;我们需要有打斗的双方&#xff0c;一个是英雄&#xff0c;一个是怪物&#xff0c;他们都有自己的属性&#xff0c;比如攻击&#xff0c;防御&#xff0c;血量。其次我们的英雄还会有武器。武器上有一些加成属性&#xff0c;可以…

c++中的文件读写的操作

写文件 ofstreamopen指定打开方式isopen判断是否打开成功ifs<<“数据”ofs.close&#xff08;&#xff09; 读文件 ifstream ifs 指定打开方式ios::in isopen判断是否打开成功 读取有三种方式 #include<iostream>using namespace std;//文件读写头文件#incl…

c++的STL--1概念通述

STL的概念 什么是STL? STL(standard template libaray-标准模板库)&#xff1a;是C标准库的重要组成部分&#xff0c;不仅是一个可复用的组件库&#xff0c;而且 是一个包罗数据结构与算法的软件框架。 STL从广义上分为&#xff1a;容器(container)&#xff0c;算法(algorit…

c++的vector容器

vector容器概念 vector是表示可变大小数组的序列容器。就像数组一样&#xff0c;vector也采用的连续存储空间来存储元素。也就是意味着可以采用下标对vector的元素 进行访问&#xff0c;和数组一样高效。但是又不像数组&#xff0c;它的大小是可以动态改变的&#xff0c;而且它…

新一代数据库技术

新一代非关系型数据库有以下5个主要类型&#xff1a; 面向文件存储&#xff1a;适用于存储海量文件&#xff0c;代表产品MongoDb 列存储(wide column store/column-family)数据库&#xff1a;快速查找相关数据&#xff0c;相关数据被放在同一列中&#xff0c;代表产品Cassandra…

c++中stack容器

Stack 简介 stack 是堆栈容器&#xff0c;是一种“先进后出”的容器。stack 是简单地装饰 deque 容器而成为另外的一种容器。#include stack没有迭代器 Stack所有元素的进出都必须符合“先进后出”的条件&#xff0c;只有stack顶端的元素&#xff0c;才有机会被外界取用&am…

c++中的queue容器

queue容器 队列是一种容器适配器&#xff0c;专门用于在FIFO上下文(先进先出)中操作&#xff0c;其中从容器一端插入元素&#xff0c;另一端 提取元素。 队列作为容器适配器实现&#xff0c;容器适配器即将特定容器类封装作为其底层容器类&#xff0c;queue提供一组特定的 成员…

c++中list容器

list概念 list是可以在常数范围内在任意位置进行插入和删除的序列式容器&#xff0c;并且该容器可以前后双向迭代。list的底层是双向链表结构&#xff0c;双向链表中每个元素存储在互不相关的独立节点中&#xff0c;在节点中通过指针指向 其前一个元素和后一个元素。list与for…