当前位置: 首页 > news >正文

C语言实现通讯录

咱们手机上面还有教务系统上都可以存储信息,这些都是使用编程语言来实现的,那么今天,咱们今天就用C语言来实现通讯录。

一. 实验名称

通讯录

二. 实验目标

1.数据的储存

2.数据的增加

3.数据的删除

4.数据的修改

5.数据的展示

6.数据的保存

7.数据的排序

三. 实现的方法

首先,咱们需要一个头文件用来包括数据类型的定义,函数声明,以及所需要使用函数的头文件,然后还需要一个.c文件用来存放所有函数的定义,不然使得主函数所在的.c文件过于冗杂。

test.c(即主函数所在的文件)

#include "test.h"void menu()
{printf("*****************************\n");printf("****1.add       2.del    ****\n");printf("****3.search    4.modify ****\n");printf("****5.show      6.sort   ****\n");printf("****0.exit               ****\n");printf("*****************************\n");
}enum option
{EXIT,//0ADD,//1DEL,//2SEARCH,MODIFY,SHOW,SORT
};int main()
{int input = 0;//创建通讯录并将其初始化Contact con;//创建Init_Contact(&con);//将通讯录初始化do{menu();printf("请输入那要选择的操作:");scanf("%d", &input);switch (input){case ADD:Add_Contact(&con);break;case DEL:Del_Contact(&con);break;case SEARCH:Search_Contact(&con);break;case MODIFY:Modify_Contact(&con);break;case SHOW:Show_Contact(&con);break;case SORT:Sort_Contact(&con);break;case EXIT:Save_Contact(&con);Destroy_Contact(&con);printf("退出通讯录!\n");break;default:printf("输入错误,请重新选择!\n");break;}} while (input);return 0;
}

为什么上面采用了枚举类型(后续博客会讲到的),因为第一个枚举数默认为0,后面每一个元素都是往后加一,这个Contact con是一个结构体类型(后续博客会讲到的)我们会在头文件中定义。


test.h(即对于函数的声明,结构体定义)

#define _CRT_SECURE_NO_WARNINGS 1
#include <string.h>
#include <stdlib.h>
#include <assert.h>
#include <stdio.h>
#pragma once
#define NAME_MAX 20
#define SEX_MAX 5
#define TELE_MAX 11
#define NUMBER_MAX 8
#define SZ 3
#define ADD_SZ 2//定义一个结构体类型,存放通讯录人物的基本信息
typedef struct PeoInfo
{char name[NAME_MAX];int age;char sex[SEX_MAX];char tele[TELE_MAX];char number[NUMBER_MAX];
}PeoInfo;//定义一个结构体类型,存放通讯录的信息
typedef struct Contact
{PeoInfo* data;//结构体类型,用于存储通讯录人物信息int sz;//当前已经存放的信息的个数int capacity;//内存空间大小
}Contact;//函数声明(初始化通讯录)
void Init_Contact(Contact* pc);//函数声明(增加通讯录信息)
void Add_Contact(Contact* pc);//函数声明(删除通讯录信息)
void Del_Contact(Contact* pc);//函数声明(查找通讯录信息)
void Search_Contact(const Contact* pc);//函数声明(更改通讯录信息)
void Modify_Contact(Contact* pc);//函数声明(展示通讯录信息)
void Show_Contact(Contact* pc);//函数声明(排序通讯录)
void Sort_Contact(Contact* pc);//函数声明(存储通讯录)
void Save_Contact(Contact* pc);//函数声明(销毁通讯录)
void Destroy_Contact(Contact* pc);//函数声明(加载通讯录)
void Load_Contact(Contact* pc);

这里的结构体类型后续博客会详细的讲解。


contact.h(函数定义)

#include "test.h"
//信息初始化
void Init_Contact(Contact* pc)
{assert(pc);pc->sz = 0;PeoInfo* ptr = (PeoInfo*)calloc(SZ, sizeof(PeoInfo));if (ptr == NULL){perror("Init_Contact::calloc");return;}pc->data = ptr;pc->capacity = SZ;//加载文件到通讯录Load_Contact(pc);
}
//检查容量
void Check_Capacity(Contact* pc)
{if (pc->capacity == pc->sz)//此时容量与已存入数据相同,这时需要增容{PeoInfo* ptr = realloc(pc->data, ((pc->capacity) + ADD_SZ) * sizeof(PeoInfo));if (ptr == NULL){perror("Check_Capacity;;realloc");return;}pc->data = ptr;pc->capacity += ADD_SZ;printf("增容成功!\n");}
}
//增加信息
void Add_Contact(Contact* pc)
{assert(pc);//检查是否容量已满,已满则选择扩容Check_Capacity(pc);//增加一个新的信息printf("请输入名字:");scanf("%s", pc->data[pc->sz].name);printf("请输入年龄:");scanf("%d", &pc->data[pc->sz].age);printf("请输入性别:");scanf("%s", pc->data[pc->sz].sex);printf("请输入电话:");scanf("%s", pc->data[pc->sz].tele);printf("请输入学号:");scanf("%s", pc->data[pc->sz].number);//信息个数加一pc->sz++;
}
//查找删除人姓名
int Find_Name(Contact* pc, char name[])
{assert(pc);int i = 0;for (i = 0; i < pc->sz; i++){if (strcmp(pc->data[i].name, name[i]) == 0){return i;}}return -1;
}
//删除信息
void Del_Contact(Contact* pc)
{assert(pc);char name[NAME_MAX] = { 0 };if (pc->sz == 0){printf("通讯录为空,无法删除!\n");return;}//删除printf("请输入你要删除人的姓名:");scanf("%s", name);int ret = Find_Name(pc,name);if (-1 == ret){printf("你要删除的人不存在!\n");return;}int i = 0;for (i = ret; i < pc->sz-1; i++){pc->data[i] = pc->data[i + 1];}pc->sz--;printf("删除成功!\n");
}
//查找信息
void Search_Contact(const Contact* pc)
{assert(pc);char name[NAME_MAX] = { 0 };printf("请输入你查找人的姓名:");scanf("%s",name);int tmp = Find_Name(pc,name);if (-1 == tmp){printf("你要查找的人不存在!\n");return;}printf("%-20s\t%-4s\t%-5s\t%-11s\t%-8s\n", "名字", "年龄", "性别", "电话", "学号");printf("%-20s\t%-4d\t%-5s\t%-11s\t%-8s\n", pc->data [tmp].name ,pc->data[tmp].age,pc->data[tmp].sex, pc->data[tmp].tele, pc->data[tmp].number);
}
//更改信息
void Modify_Contact(Contact* pc)
{assert(pc);char name[NAME_MAX] = { 0 };printf("请输入你更改人的姓名:");scanf("%s", name);int pos = Find_Name(pc, name);if (-1 == pos){printf("你要更改的人不存在!\n");return;}//更改printf("请输入更改的名字:");scanf("%s", pc->data[pos].name);printf("请输入更改的年龄:");scanf("%d", &pc->data[pos].age);printf("请输入更改的性别:");scanf("%s", pc->data[pos].sex);printf("请输入更改的电话:");scanf("%s", pc->data[pos].tele);printf("请输入更改的学号:");scanf("%s", pc->data[pos].number);printf("修改完成!\n");
}
//展示信息
void Show_Contact(Contact* pc)
{assert(pc);int i = 0;printf("%-20s\t%-4s\t%-5s\t%-11s\t%-8s\n", "名字", "年龄", "性别", "电话", "学号");printf("%-20s\t%-4d\t%-5s\t%-11s\t%-8s\n", pc->data[i].name,pc->data[i].age,pc->data[i].sex,pc->data[i].tele,pc->data[i].number);
}
//比较名字大小
int cmp_name(const void* n1, const void* n2)
{return (strcmp(((PeoInfo*)n1)->name, ((PeoInfo*)n2)->name));
}
//排序通讯录
void Sort_Contact(Contact* pc)
{assert(pc);qsort(pc->data, pc->sz, sizeof(PeoInfo), cmp_name);printf("排序完成!\n");
}
//储存数据
void Save_Contact(Contact* pc)
{FILE* pf = fopen("contact.txt", "wb");if (NULL == pf){perror("Save_Contact");}else{int i = 0;for (i = 0; i < pc->sz; i++){fwrite(pc->data + i, sizeof(PeoInfo), 1, pf);}}fclose(pf);pf = NULL;printf("保存成功!\n");
}
//
void Load_Contact(Contact* pc)
{FILE* pf = fopen("contact.txt", "rb");if (pf == NULL){perror("Load_Contact");}else{PeoInfo tmp = { 0 };int i = 0;while (fread(&tmp, sizeof(PeoInfo), 1, pf)){Check_Capacity(pc);pc->data[i] = tmp;pc->sz++;i++;}fclose(pf);pf = NULL;}	
}
//销毁信息
void Destroy_Contact(Contact* pc)
{free(pc->data);pc->data = NULL;pc->sz = 0;pc->capacity = 0;pc = NULL;
}

这段代码很长,接下来我们将一一介绍:

assert函数,就是断言,用于判断传入指针是否为空指针,

calloc函数,为创建空间的函数,并会初始化,与melloc还是有略微差异的,

perror函数,查看函数错误的原因并打印出错误信息,

realloc函数,在原有空间上增加空间,

FILE文档类型,

fopen,以某种方式打开文档,

fclose,关闭文档,并且需要将其置位空指针,

free,释放空间函数创建的空间,也需要将其置位空,

这些就是我们所使用的数组,下面就要介绍我们是如何实现的了。

根据主函数我们所需要的功能有增加,删除,展示,排序,查找,展示,保存退出,首先我们需要一个通讯录,一个通讯录需要数据(data),容量(capacity)还有大小(size),并且需要将其初始化,当容量满了之后就需要扩容,增加就只需要让数据的容量内存都变大一个相应的值,非常容易实现,删除,更改和查找都需要先找到所需要元素的名字,找到直接删除,更改即可,展示通讯录使用遍历即可,排序我们则采用qsort快排来实现,最后存储文件我们在后续博客中会讲到,这就是通讯录的全部思路。

相关文章:

C语言实现通讯录

咱们手机上面还有教务系统上都可以存储信息&#xff0c;这些都是使用编程语言来实现的&#xff0c;那么今天&#xff0c;咱们今天就用C语言来实现通讯录。 一. 实验名称 通讯录 二. 实验目标 1.数据的储存 2.数据的增加 3.数据的删除 4.数据的修改 5.数据的展示 6.数据…...

Python-生成列表

1.生成列表使用列表前必须先生成列表。1.1使用运算符[ ]生成列表在运算符[ ]中以逗号隔开各个元素会生成包含这些元素的新列表。另外&#xff0c;如果[ ]中没有元素就会生成空列表示例>>> list01 [] >>> list01 [] >>> list02 [1, 2, 3] >>…...

如何写好controller层

前言本篇主要要介绍的就是controller层的处理&#xff0c;一个完整的后端请求由4部分组成&#xff1a;1. 接口地址(也就是URL地址)、2. 请求方式(一般就是get、set&#xff0c;当然还有put、delete)、3. 请求数据(request&#xff0c;有head跟body)、4. 响应数据(response)本篇…...

MySQL---视图的概念与操作

MySQL—视图的概念与操作 常见的数据库对象 对象描述表(TABLE)表是存储数据的逻辑单元&#xff0c;以行和列的形式存在&#xff0c;列就是字段&#xff0c;行就是记录数据字典就是系统表&#xff0c;存放数据库相关信息的表。系统表的数据通常由数据库系统维护&#xff0c; 程…...

ChatGPT,会是现实世界的MOSS吗?

最近&#xff0c;两个人工智能系统彻底火出了圈&#xff0c;成为许多网友热议的焦点。 一个是冷酷无情的“幕后主角”MOSS&#xff0c;一个是多才多艺的“全能网友”ChatGPT。 先来说说MOSS。今年春节档&#xff0c;科幻电影《流浪地球2》热映&#xff0c;电影中一个面试环节令…...

安卓大厂面试题_安卓开发面经_Android大厂面经(22/30)之JNI全解析

系列专栏: 《150道安卓常见面试题全解析》 安卓专栏目录见帖子 : 安卓面经_anroid面经_150道安卓常见基础面试题全解析 安卓系统Framework面经专栏:《Android系统Framework面试题解析大全》 安卓系统Framework面经目录详情:Android系统面经_Framework开发面经_150道面试题答…...

记一次docker虚拟机横向移动渗透测试

本次渗透在几个docker虚拟机间多次横向移动&#xff0c;最终找到了一个可以进行docker逃逸的出口&#xff0c;拿下服务器。渗透过程曲折但充满了乐趣&#xff0c;入口是172.17.0.6的docker虚拟机&#xff0c;然后一路横向移动&#xff0c;最终在172.17.0.2出实现了docker逃逸&a…...

计算机网络-物理层

计算机网络-物理层 计算机网络学习笔记 学习视频&#xff1a;https://www.bilibili.com/video/BV1c4411d7jb/?p14&spm_id_from333.1007.top_right_bar_window_history.content.click&vd_source75dce036dc8244310435eaf03de4e330 物理层的基本概念 物理层考虑的是怎样…...

Kubernetes Nginx 发布

kubernetes发布nginx 目录 Nginx Pod启动Service访问Nginx 2.1. NodePort访问Nginx 2.2. ClusterIP访问Nginx 2.3. LoadBalancer访问Nginx 2.4. ExternalName访问NginxDeployment方式部署Nginx 3.1 Nginx Replicas Nginx Pod 启动 nginx-v1.yaml apiVersion: v1 kind: Pod…...

华为OD机试真题Python实现【非严格递增连续数字序列】真题+解题思路+代码(20222023)

非严格递增连续数字序列 题目 输入一个字符串仅包含大小写字母和数字 求字符串中包含的最长的非严格递增连续数字序列长度 比如: 12234 属于非严格递增数字序列 🔥🔥🔥🔥🔥👉👉👉👉👉👉 华为OD机试(Python)真题目录汇总 ## 输入 输入一个字符串仅…...

12-render函数

render函数 一、render的作用 非单文件组件时&#xff0c;创建vm的写法&#xff1a; new Vue({el: #root,template: <App></App>,components: {App} })但是该写法在脚手架中会报错。因为脚手架默认引入的是个残缺版的vue。 如果要避免报错&#xff0c;有2种解决…...

磨金石教育摄影技能干货分享|杨元惺佳作欣赏——诗意人文

一般来说&#xff0c;人文摄影总会体现现实性多些。但杨老师是个摄影诗人&#xff0c;他的内心总能将刻板的现实融入美好的光芒。你在他的照片里&#xff0c;看着现实的摄影素材&#xff0c;所感受到的是诗意的绵绵未尽。春网&#xff08;中国&#xff09;正所谓春水碧于天&…...

在Pandas中通过时间频率来汇总数据的三种常用方法

当我们的数据涉及日期和时间时&#xff0c;分析随时间变化变得非常重要。Pandas提供了一种方便的方法&#xff0c;可以按不同的基于时间的间隔(如分钟、小时、天、周、月、季度或年)对时间序列数据进行分组。 在Pandas中&#xff0c;有几种基于日期对数据进行分组的方法。我们将…...

基于SPI的增强式插件框架设计

很久之前&#xff0c;为了诊断线上的问题&#xff0c;就想要是能有工具可以在线上出问题的时候&#xff0c;放个诊断包进去马上生效&#xff0c;就能看到线上问题的所在&#xff0c;那该是多么舒服的事情。后来慢慢的切换到 java 领域后&#xff0c;这种理想也变成了现实&#…...

176、【动态规划】leetcode ——1143. 最长公共子序列(C++版本)

题目描述 原题链接&#xff1a;1143. 最长公共子序列 题目描述 本题和 718. 最长重复子数组&#xff08;动态规划&#xff09; 的区别在于此时不要求令一个数组中元素连续。 动态规划五步曲&#xff1a; &#xff08;1&#xff09;dp[i][j]含义&#xff1a; 截止到text1[i …...

16行代码采集原神官网角色全图+全语音

嗨害大家好鸭&#xff01;我是小熊猫~ 本来是不玩原神的&#xff0c; 但是实在是经不住诱惑鸭~ 毕竟谁能拒绝可以爬树、炸鱼、壶里造房子、抓小动物、躲猫猫的对战游戏捏~ 准备工具 源码资料电子书:点击此处跳转文末名片获取 准备模块 import requests import re import ex…...

Unity(二)--通过简单例子了解UGUI几个常用对象操作(Text,Image,Button)

目录 文本框等UI对象的添加Canvas 画布给Canvas添加脚本,绑定要操作的对象文本框Text的使用图像Image的使用更换图片Type:显示图片相关按钮Button的使用过渡导航事件绑定文本框等UI对象的添加 Canvas 画布 所有的UI元素的父物体,。 当创建一个UI元素的时候,如果没有Canvas…...

手写一个文件上传demo

背景 最近闲来无事&#xff0c;同事闻了一下上传文件的基本操作&#xff0c;如何用文件流来实现一个文件的上传功能 基本概念 流&#xff08;Stream&#xff09;是指在计算机的输入输出操作中各部件之间的数据流动。可以按照数据传输的方向&#xff0c;将流可分为输入流和输出…...

通过 Apifox Echo 了解 Content-Length

Content-Length 用以指定 Body 的体积。响应头中的 Content-Length 指定 Response Body 的体积&#xff0c;请求头中的 Content-Length 指定 Request Body 的体积。 通过 Content-Length&#xff0c;HTTP 客户端/服务器端将会根据该头部计算出 Body 的大小。 请求头中的 Cont…...

ESP32设备驱动-CPU频率设置

CPU频率设置 文章目录 CPU频率设置1、ESP32的CPU频率介绍1.1 CPU时钟1.2 外设时钟2、CPU频率设置API3、软件准备4、硬件准备5、CPU频率设置实例5.1 CPU频率读取5.2 CPU频率设置在本文中,将介绍如何通过Arduino Core设置或更改ESP32 CPU时钟速度(频率)。 1、ESP32的CPU频率介…...

C++实现分布式网络通信框架RPC(3)--rpc调用端

目录 一、前言 二、UserServiceRpc_Stub 三、 CallMethod方法的重写 头文件 实现 四、rpc调用端的调用 实现 五、 google::protobuf::RpcController *controller 头文件 实现 六、总结 一、前言 在前边的文章中&#xff0c;我们已经大致实现了rpc服务端的各项功能代…...

【大模型RAG】Docker 一键部署 Milvus 完整攻略

本文概要 Milvus 2.5 Stand-alone 版可通过 Docker 在几分钟内完成安装&#xff1b;只需暴露 19530&#xff08;gRPC&#xff09;与 9091&#xff08;HTTP/WebUI&#xff09;两个端口&#xff0c;即可让本地电脑通过 PyMilvus 或浏览器访问远程 Linux 服务器上的 Milvus。下面…...

Java - Mysql数据类型对应

Mysql数据类型java数据类型备注整型INT/INTEGERint / java.lang.Integer–BIGINTlong/java.lang.Long–––浮点型FLOATfloat/java.lang.FloatDOUBLEdouble/java.lang.Double–DECIMAL/NUMERICjava.math.BigDecimal字符串型CHARjava.lang.String固定长度字符串VARCHARjava.lang…...

【ROS】Nav2源码之nav2_behavior_tree-行为树节点列表

1、行为树节点分类 在 Nav2(Navigation2)的行为树框架中,行为树节点插件按照功能分为 Action(动作节点)、Condition(条件节点)、Control(控制节点) 和 Decorator(装饰节点) 四类。 1.1 动作节点 Action 执行具体的机器人操作或任务,直接与硬件、传感器或外部系统…...

Android Bitmap治理全解析:从加载优化到泄漏防控的全生命周期管理

引言 Bitmap&#xff08;位图&#xff09;是Android应用内存占用的“头号杀手”。一张1080P&#xff08;1920x1080&#xff09;的图片以ARGB_8888格式加载时&#xff0c;内存占用高达8MB&#xff08;192010804字节&#xff09;。据统计&#xff0c;超过60%的应用OOM崩溃与Bitm…...

RNN避坑指南:从数学推导到LSTM/GRU工业级部署实战流程

本文较长&#xff0c;建议点赞收藏&#xff0c;以免遗失。更多AI大模型应用开发学习视频及资料&#xff0c;尽在聚客AI学院。 本文全面剖析RNN核心原理&#xff0c;深入讲解梯度消失/爆炸问题&#xff0c;并通过LSTM/GRU结构实现解决方案&#xff0c;提供时间序列预测和文本生成…...

大语言模型(LLM)中的KV缓存压缩与动态稀疏注意力机制设计

随着大语言模型&#xff08;LLM&#xff09;参数规模的增长&#xff0c;推理阶段的内存占用和计算复杂度成为核心挑战。传统注意力机制的计算复杂度随序列长度呈二次方增长&#xff0c;而KV缓存的内存消耗可能高达数十GB&#xff08;例如Llama2-7B处理100K token时需50GB内存&a…...

CVE-2020-17519源码分析与漏洞复现(Flink 任意文件读取)

漏洞概览 漏洞名称&#xff1a;Apache Flink REST API 任意文件读取漏洞CVE编号&#xff1a;CVE-2020-17519CVSS评分&#xff1a;7.5影响版本&#xff1a;Apache Flink 1.11.0、1.11.1、1.11.2修复版本&#xff1a;≥ 1.11.3 或 ≥ 1.12.0漏洞类型&#xff1a;路径遍历&#x…...

LINUX 69 FTP 客服管理系统 man 5 /etc/vsftpd/vsftpd.conf

FTP 客服管理系统 实现kefu123登录&#xff0c;不允许匿名访问&#xff0c;kefu只能访问/data/kefu目录&#xff0c;不能查看其他目录 创建账号密码 useradd kefu echo 123|passwd -stdin kefu [rootcode caozx26420]# echo 123|passwd --stdin kefu 更改用户 kefu 的密码…...

基于Springboot+Vue的办公管理系统

角色&#xff1a; 管理员、员工 技术&#xff1a; 后端: SpringBoot, Vue2, MySQL, Mybatis-Plus 前端: Vue2, Element-UI, Axios, Echarts, Vue-Router 核心功能&#xff1a; 该办公管理系统是一个综合性的企业内部管理平台&#xff0c;旨在提升企业运营效率和员工管理水…...