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

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结论

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

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

  1. 【数据结构课程设计报告】电话号码查询系统(Java实现)

    数据结构课程设计报告 电话号码查询系统 数据结构课程设计报告 一.需求分析 二.系统功能划分及设计 1.存储结构设计 2.系统的功能架构设计 3.模块设计 3.代码实现 一.需求分析 问题描述:路径规 ...

  2. 数据结构迷宫代码_数据结构课程设计——迷宫求解(二)

    前言 接上文的介绍,本文将主要介绍如何生成随机迷宫,在网上找到的资源也比较多,这里我选取了随机 Prim 算法生成迷宫,选择这个算法的理由如下: 算法思想简单,易于实现 生成的迷宫比较自然,不会出现明 ...

  3. 2017大二学年 数据结构课程设计-校园十大优秀青年评比

    南 通 大 学   数据结构课程设计报告   姓    名:   班    级: 物联网162 学    号:   指导老师: 杭月琴 选   

  4. c语言数据结构课程设计停车场管理系统,数据结构课程设计报告停车场管理系统...

    <数据结构课程设计报告停车场管理系统>由会员分享,可在线阅读,更多相关<数据结构课程设计报告停车场管理系统(8页珍藏版)>请在人人文库网上搜索. 1.数据结构课程设计报告系 别 ...

  5. 大学数据结构课程设计题目

    数据结构课程设计题目 1.         飞机订票系统(限1 人完成) 任务:通过此系统可以实现如下功能: 录入: 可以录入航班情况(数据可以存储在一个数据文件中,数据结构.具体数据自定) 查询: ...

  6. c语言二叉树族谱管理系统,数据结构课程设计报告(用二叉树实现家谱管理系统).doc...

    数据结构课程设计 题目:用二叉树实现家谱管理系统 姓名:郭志超 学号:031010151554042 完成日期:2005.7.3 一.需求分析 ??建立输入文件以存放最初家谱中各成员的信息. ??成员 ...

  7. C/C++数据结构课程设计安排

    C/C++数据结构课程设计安排 数据结构课程设计安排 课程设计学时:32学时 课程设计目的:综合应用数据结构课程中所学的数据结构:线性表.栈.队列.数组.广义表.树.二叉树.图.查找表中的一种或多种数 ...

  8. 数据结构课程设计 ——考试报名系统

    数据结构课程设计 --考试报名系统 一.项目功能要求 完成对考生信息的建立,查找,插入,修改,删除等功能.其中考生信息包括准考证号,姓名,性别,年龄和报考类别等信息.项目在设计时应首先确定系统的数据结 ...

  9. “数据结构”课程设计题目

    "数据结构"课程设计题目 1.城市链表 [问题描述] 将若干城市的信息,存入一个带头结点的单链表.结点中的城市信息包括:城市名,城市的位置坐标.要求能够利用城市名和位置坐标进行有关 ...

最新文章

  1. Mac OS X10.11(OS X EI Capitan)安装程序下载
  2. 通过Windbg查看DataTable的值
  3. php 对象转换成数组,PHP把对象转换为数组的问题
  4. LeetCode 108. Convert Sorted Array to Binary Search Tree
  5. 零基础学前端可行吗?要如何学习呢?
  6. 如何用程序实现对IE中scripts的控制(禁止和允许)
  7. 显式强制类型转换static_cast, dynamic_cast, const_cast, reinterpret_cast
  8. thing php官网,Thinkphp5企业官网,php后台管理框架
  9. cookie控制窗口打开打开
  10. openstack: No valid host was found. There are not enough hosts available
  11. mysql innodb count 优化_MySQL · 引擎特性 · InnoDB COUNT(*) 优化(?)
  12. matplotlib学习日记(八)----完善统计图
  13. linux下载git并为git配置连接ssh
  14. Seat分布式事务学习
  15. Android常用逆向工具+单机游戏破解
  16. 7-2 哈利·波特的考试 (25 分)(解释的挺详细了吧)
  17. 计算机信息管理专业 英文,计算机信息管理专业英文简历范文
  18. mysql 3389_SQL语句直接开启3389
  19. 计算机IPv4升级到IPv6的技术,IPv4到IPv6的变化
  20. python实现根据文件名自动分类转移至不同的文件夹

热门文章

  1. 08 Spring框架 AOP (一)
  2. NOIP 2012 Day2
  3. 函数 (四) 迭代器和生成器
  4. 搭建mysql集群,使用Percona XtraDB Cluster搭建
  5. Topcoder SRM 648 (div.2)
  6. 面试题-ASP 与 ASP.Net的区别?
  7. 在asp.net中调用process.start执行程序
  8. 关于 SENDKEYS 的代码
  9. svn安装教程 mysql_CentOS6.4 下安装SVN的详细教程(超详细)
  10. linux下I2C驱动发送IO时序,Linux I2C 驱动阅读的碰到的一些网上没有提到的东西