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

通讯录(C语言版)

用c语言实现一个通讯录

功能:.添加、删除、查找、更改、显示、排序联系人

内存存储方式:结构体数组

1.打印菜单,各个功能分别用函数实现,将函数声明放在头文件中。

2.定义联系人信息,将联系人信息与count(用来记录当前通讯录人数),capacity(记录最大容量)。一起定义到结构体中,完成联系人的创建:

//定义联系人
typedef struct PenInfo {char name[20];int age;char sex[10];char tele[12];char addr[30];
}PenInfo;typedef struct Contact {PenInfo* data;int count;int capacity;//设置我通讯录的最大容量
}Contact;

3.创建联系人,并用malloc或calloc函数向内存堆区开辟内存,在退出时记得释放内存,将count初始化为0

int Initcontact(Contact* pc) {assert(pc);pc->data =(PenInfo*) calloc(3,3 * sizeof(PenInfo));//为data开辟三个空间if (pc->data == NULL) {printf("%s", strerror(errno));return 1;}pc->count = 0;pc->capacity = 3;//设置最大容量为3//从文件中读取联系人内容LoadContact(pc);return 0;
}

4.添加联系人,当联系人容量达到最大容量时,增容,然后进行联系人的信息输入:

void AddCapacity(Contact *pc) {assert(pc);if (pc->count == pc->capacity) {pc->capacity +=2;//printf("增容成功\n");}return;
}void Addcontact(Contact* pc){assert(pc);//增容AddCapacity(pc);//添加联系人printf("请输入联系人名字:>");scanf("%s", pc->data[pc->count].name);printf("请输入联系人年龄:>");scanf("%d", &(pc->data[pc->count].age));printf("请输入联系人性别:>");scanf("%s", pc->data[pc->count].sex);printf("请输入联系人电话:>");scanf("%s", pc->data[pc->count].tele);printf("请输入联系人地址:>");scanf("%s", pc->data[pc->count].addr);pc->count++;printf("添加成功\n");
}

5. 显示联系人(记得格式化)通过for循环标准输出:

void Showcontact(const Contact* pc) {assert(pc);if (pc->count == 0) {printf("还没添加联系人\n");return;}printf("%-20s\t%-3s\t%-10s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");printf("-----------------------------------------------------------------------------\n");for (int i = 0; i < pc->count; i++) {printf("%-20s\t%-5d\t%-10s\t%-12s\t%-30s", pc->data[i].name,pc->data[i].age,pc->data[i].sex,pc->data[i].tele,pc->data[i].addr);printf("\n");printf("-----------------------------------------------------------------------------\n");}
}

6.当查找、删除、更改时都需要进行查找,所以直接封装一个查找函数(找到了返回下标)

//查找联系人
static int Findcontact(Contact* pc, char* name) {assert(pc);for (int i = 0; i < pc->count; i++) {if (strcmp(name, pc->data[i].name) == 0) {return i;}}return -1;
}

7.查找联系人

void Searchcontact(Contact* pc) {assert(pc);char name[20] = { 0 };printf("请输入要查找的联系人名字:>");scanf("%s", name);//查找int pos = Findcontact(pc, name);//删除if (pos == -1) {printf("没有该联系人\n");return;}else {printf("找到了\n");printf("%-20s\t%-3s\t%-10s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");printf("-----------------------------------------------------------------------------\n");printf("%-20s\t%-5d\t%-10s\t%-12s\t%-30s", pc->data[pos].name,pc->data[pos].age,pc->data[pos].sex,pc->data[pos].tele,pc->data[pos].addr);printf("\n");printf("-----------------------------------------------------------------------------\n");}}

 8.删除联系人,删除后,由于是顺序表存储,将后面的数据向前一个结点覆盖,且count--

void Delcontact(Contact* pc) {assert(pc);char name[20] = { 0 };printf("请输入要删除的联系人名字:>");scanf("%s", name);//查找int pos = Findcontact(pc, name);//删除if (pos == -1) {printf("没有该联系人\n");return;}for (int i = pos; i < pc->count + 1; i++) {//count+1防止溢出pc->data[i] = pc->data[i + 1];}pc->count--;printf("删除成功\n");}

9.修改联系人,将信息覆盖即可:

void Modifycontact(Contact* pc) {assert(pc);char name[20] = { 0 };printf("请输入要修改的联系人的名字:>");scanf("%s", name);//查找int pos = Findcontact(pc, name);//删除if (pos == -1) {printf("没有该联系人\n");return;}else {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].addr);printf("\n");printf("修改成功\n");}
}

10.排序,分为年龄和姓名,都用库函数qsort(默认升序)实现,年龄进行时稍微复杂,细节如下:

//按姓名排序函数
int Sort_by_name(const void* e1, const void* e2) {return strcmp(((PenInfo*)e1)->name, ((PenInfo*)e2)->name);
}
//按年龄排序
int Sort_by_age(const void* e1, const void* e2) {if (((PenInfo*)e1)->age < ((PenInfo*)e2)->age) {return -1;}if (((PenInfo*)e1)->age > ((PenInfo*)e2)->age) {return 1;}if (((PenInfo*)e1)->age == ((PenInfo*)e2)->age) {return 0;}
}void Sortcontact(Contact* pc) {assert(pc);int chiose;printf("(1.按姓名排序 2.按年龄排序):>");scanf("%d", &chiose);if (chiose == 1) {qsort(pc->data,pc->count,sizeof(PenInfo), Sort_by_name);printf("排序成功\n");}else if (chiose == 2) {qsort(pc->data, pc->count, sizeof(PenInfo), Sort_by_age);printf("排序成功\n");}else {printf("无效输入\n");return;}
}

11.保存联系人到相对路径“contact.txt”中,通过二进制进行输入输出,退出时保存,打开时读取。

void SaveContact(Contact* pc) {FILE* pfWrite = fopen("contact.txt", "wb");if (pfWrite == NULL) {perror("SaveContact");return;}//用二进制的形式写入文件for (int i = 0; i <= pc->count; i++) {fwrite(pc->data+i, sizeof(PenInfo), 1, pfWrite);}//关闭文件fclose(pfWrite);pfWrite = NULL;
}void LoadContact(Contact* pc) {assert(pc);FILE* pfRead = fopen("contact.txt", "rb");if (pfRead == NULL) {//perror("LoadContact");return;}while (fread(&(pc->data[pc->count]), sizeof(PenInfo), 1, pfRead) == 1) {//判断是否空间不足AddCapacity(pc);pc->count++;}fclose(pfRead);pfRead = NULL;
}

12.测试代码:

#define _CRT_SECURE_NO_WARNINGS#include "contact.h"//创建菜单
void menu() {printf("------------My ContactBook------------\n");printf("-----------1.Add linkman  ------------\n");printf("-----------2.Del linkman  ------------\n");printf("-----------3.Show linkman ------------\n");printf("-----------4.Search linkman ----------\n");printf("-----------5.Modify linkman ----------\n");printf("-----------6.Sort linkman -----------\n");printf("-----------0.Exit menu  -------------\n");}
int main() {    int input = 0;//创建联系人Contact con;//初始化通讯录Initcontact(&con);do {menu();printf("请选择对应功能的编号:>");scanf("%d", &input);switch (input) {case 1:         Addcontact(&con);break;case 2:Delcontact(&con);break;case 3:Showcontact(&con);break;case 4:Searchcontact(&con);break;case 5:Modifycontact(&con);break;case 6:Sortcontact(&con);break;case 0:SaveContact(&con);DestoryContact(&con);printf("退出通讯录\n");break;default :printf("选择错误\n");break;}} while (input);return 0;
}

测试结果:

相关文章:

通讯录(C语言版)

用c语言实现一个通讯录 功能&#xff1a;.添加、删除、查找、更改、显示、排序联系人 内存存储方式&#xff1a;结构体数组 1.打印菜单&#xff0c;各个功能分别用函数实现&#xff0c;将函数声明放在头文件中。 2.定义联系人信息&#xff0c;将联系人信息与count&#xff…...

natapp内网穿透-将本地运行的程序/服务器通过公网IP供其它人访问

文章目录 1.几个基本概念1.1 局域网1.2 内网1.3 内网穿透1.4 Natapp 2.搭建内网穿透环境3.本地服务测试 1.几个基本概念 1.1 局域网 LAN&#xff08;Local Area Network&#xff0c;局域网&#xff09;是一个可连接住宅&#xff0c;学校&#xff0c;实验室&#xff0c;大学校…...

数据结构八大排序Java源码

文章目录 [1]. 堆排序[2]. 冒泡排序[3]. 选择排序[4]. &#xff08;直接&#xff09;插入排序[5]. 希尔排序&#xff08;属于插入算法&#xff09;[6]. 快速排序[7]. 归并排序[8]. 基数排序 王道数据结构排序讲解 排序算法最佳时间复杂度最坏时间复杂度平均时间复杂度空间复杂度…...

区块链加密虚拟货币交易平台安全解决方案

区块链机密货币交易锁遭入侵&#xff0c;安全存在隐患。使用泰雷兹Protect server HSM加密机&#xff0c;多方位保护您的数据&#xff0c;并通过集中化管理&#xff0c;安全的存储密钥。 引文部分&#xff1a; 损失7000万美元!黑客入侵香港区块链加密货币交易所 2023年9月&…...

【SoC FPGA】HPS启动过程

SoC HPS启动流程 Boot ROMPreloaderBoot Loader HPS的启动是一个多阶段的过程&#xff0c;每一个阶段都会完成对应的工作并且将下一个阶段的执行代码引导起来。每个阶段均负责加载下一个阶段。第一个软件阶段是引导 ROM&#xff0c;引导 ROM 代码查找并且执行称为预加载器的第 …...

Wireshark CLI | Mergecap 篇

简介 Mergecap 是 Wireshark 程序安装时附带的可选工具之一&#xff0c;用于合并数据包文件的命令行工具。 mergecap [ -a ] [ -F <file format> ] [ -I <IDB merge mode> ] [ -s <snaplen> ] [ -V ] -w <outfile>|- <infile> [<infile>…...

10个打工人必备AI神器,升职加薪靠AI

HI&#xff0c;同学们&#xff0c;我是赤辰&#xff0c;本期是第18篇AI工具类教程&#xff0c;文章底部准备了粉丝福利&#xff0c;看完后可领取&#xff01;1. Runway&#xff08;文字转视频AI工具&#xff09; 只需要一句提示词就能精确生成你所想象的视频场景&#xff0c;还…...

Java架构师缓存架构设计

目录 1 导学2 高性能概述2.1 高性能的定义和衡量指标2.2 如何实现高性能的计算机系统或软件程序2.3 木桶理论2.4 如何实现计算机系统或软件程序的高性能3 多级缓存设计3.1 浏览器缓存3.2 CDN缓存3.3 负载均衡的缓存3.4 进程内缓存3.5 分布式缓存4 缓存技术方案5 如何进行缓存拆…...

Linux 安全 - DAC机制

文章目录 一、安全简介二、DAC2.1 UNIX 的自主访问控制2.2 Linux 的自主访问控制 三、进程凭证3.1 简介3.2 uid/gid3.3 系统调用 四、客体标记4.1 简介4.2 系统调用 五、UGO规则源码分析参考资料 一、安全简介 计算机系统应对安全挑战的办法大致有四种&#xff1a;隔离、控制、…...

解决Windows系统win+shift+s截图快捷键失效问题

文章目录 打开任务管理器找到Windows资源管理器&#xff0c;选择重新启动 打开任务管理器 按“Win R”打开&#xff1a; 输入taskmgr.exe&#xff0c;运行&#xff0c;即可打开任务管理器&#xff1a; 找到Windows资源管理器&#xff0c;选择重新启动 点击右下角的“重新启…...

Excel 快速填充

文章目录 利用快速填充进行提取数据利用快速填充进行拆分重组 2013 及以上版本才有的功能. 利用快速填充进行提取数据 有一列的数据已有, 需要提取部分数据到另一列, 只需要输入部分内容, 后面内容可以自动显示, 按下回车即可快速填充. 只要前面手动输入的内容没有错得太离谱…...

OPENCV图像和视频处理

图像基本操作&#xff1a; Opencv图像处理&#xff08;全&#xff09;_胖墩会武术的博客-CSDN博客 import cv2 import matplotlib.pyplot as plt import numpy as npimgcv2.imread(1.jpg) #图像的显示 cv2.imshow(image,img) cv2.waitKey(0) …...

QDir实践

现在有多个文件&#xff0c;路径为&#xff1a; a\xxx\kmd_config\c.json 其中xxx是变量 startcalc,,,,,, 目标&#xff1a; 访问每一个json文件 实例&#xff1a; QString app_path QApplication::applicationDirPath() "/app";QDir dir(app_path);QStringLi…...

网络通信三要素

三要素概述 IP地址&#xff1a;设备在网络中的地址&#xff0c;是唯一的标识。 端口&#xff1a;应用程序在设备中唯一的标识。 协议: 数据在网络中传输的规则&#xff0c;常见的协议有UDP协议和TCP协议。 网络通信过程 A程序通过IP和端口连接到到B程序&#xff0c;再互…...

2023年中国渔业研究报告

第一章 行业概况 1.1 定义 渔业&#xff0c;作为全球经济的重要支柱之一&#xff0c;其核心活动包括捕捞、水产养殖、产品加工与销售等。其不仅是食物安全的重要保障&#xff0c;还是许多沿海和内陆地区经济发展的重要动力。 首先&#xff0c;捕捞活动是渔业的基础。通过海洋…...

python字符串中的\“

data {"text": "\"abc\""} print(data) # {"text": ""abc""}从结果可以看到并没有出现反斜杠&#xff0c;反斜杠与双引号作为一个整体&#xff0c;转义为了一个双引号&#xff0c;如果要在字符串中出现反斜杠&am…...

Elasticsearch 分片内部原理—使文本可被搜索、动态更新索引

目录 一、使文本可被搜索 不变性 二、动态更新索引 删除和更新 一、使文本可被搜索 必须解决的第一个挑战是如何使文本可被搜索。 传统的数据库每个字段存储单个值&#xff0c;但这对全文检索并不够。文本字段中的每个单词需要被搜索&#xff0c;对数据库意味着需要单个字…...

lvgl 界面管理器

lv_scr_mgr lvgl 界面管理器 适配 lvgl 8.3 降低界面之间的耦合使用较小的内存&#xff0c;界面切换后会自动释放内存内存泄漏检测 使用方法 在lv_scr_mgr_port.h 中创建一个枚举&#xff0c;用于界面ID为每个界面创建一个页面管理器句柄将界面句柄添加到 lv_scr_mgr_por…...

一篇文章让你了解“JWT“

一.JWT简介 1.概念 JWT (JSON Web Token) 是一种用于在网络上安全传输信息的开放标准&#xff08;RFC 7519&#xff09;。它是一种紧凑且自包含的方式&#xff0c;用于在不同组件之间传递信息&#xff0c;通常用于身份验证和授权目的。JWT 是以 JSON 格式编码的令牌&#xff…...

HJ73 计算日期到天数转换

HJ73 计算日期到天数转换 int main() {int year, month, day;cin >> year >> month >> day;int monthDays[13] { 0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334, 365 };int nday monthDays[month - 1] day;if (month > 2 &&((year…...

OpenClaw × 88API:不用注册 Anthropic,5 分钟让 AI Agent 接入 Claude 4.6(2026 完整教程)

折腾了两天&#xff0c;最后 5 分钟搞定 上周我想用 OpenClaw 搭一个能自动重构代码的 Agent。选定 Claude 4.6 当大脑——毕竟它在 Tool Use 精准度和长上下文推理上确实是第一梯队。 结果卡在了第一步&#xff1a;Anthropic 官方账号注册要海外手机号&#xff0c;好不容易注…...

终极Windows XP错误对话框组件:怀旧系统提示的优雅实现指南

终极Windows XP错误对话框组件&#xff1a;怀旧系统提示的优雅实现指南 【免费下载链接】winXP &#x1f3c1; Web based Windows XP desktop recreation. 项目地址: https://gitcode.com/gh_mirrors/wi/winXP 你是否怀念Windows XP那个经典的错误提示对话框&#xff1…...

导师推荐 2026 最新!降AI率软件测评与好用工具推荐

2026年真正好用的AI论文降重与改写工具&#xff0c;核心看降重效果、去AI味、格式保留、学术适配四大指标。综合实测&#xff0c;千笔AI、ThouPen、豆包、DeepSeek、Grammarly 是当前最值得推荐的梯队&#xff0c;覆盖从免费到付费、从中文到英文、从文科到理工的全场景需求。 …...

从理论到实践:几何完备扩散模型GCDM在SBDD任务中的实战评测与性能剖析

1. 几何完备扩散模型GCDM的核心原理 GCDM&#xff08;Geometry-Complete Diffusion Model&#xff09;作为新一代3D分子生成模型&#xff0c;其核心创新在于解决了传统方法无法有效学习分子几何特性的痛点。想象一下搭积木的场景&#xff1a;普通模型只能看到积木的颜色&#x…...

Vue3 + Cornerstone3D:从零构建支持本地Nifti文件上传与四视图联动的医学影像查看器

1. 为什么选择Vue3Cornerstone3D开发医学影像查看器 医学影像处理一直是前端开发中颇具挑战性的领域&#xff0c;特别是当需要处理专业格式如Nifti时。我在实际项目中尝试过多种技术方案后&#xff0c;发现Vue3和Cornerstone3D的组合特别适合快速构建高性能的医学影像应用。 …...

[带AI]基于SpringBoot+Vue的青少年心理健康管理系统设计与实现+文档+指导搭建视频

&#xff5c;前后端分离&#xff5c;Java&#xff5c;SpringBoot&#xff5c;Vue3&#xff5c;Spring AI智能对话一、项目技术栈项目采用技术&#xff1a;① 架构模式&#xff1a;前后端分离开发② 系统环境&#xff1a;Windows、Mac③ 开发环境&#xff1a;IDEA、JDK21、MySQL…...

【2026年阿里巴巴春招- 3月28日-算法岗-第二题- 隐式素数计算】(题目+思路+JavaC++Python解析+在线测试)

题目内容 我们称一个正整数为隐式素数,如果它不同的正因子的个数是一个素数。给定一个闭区间$ [l,r]$,请计算该区间内隐式素数的个数 输入描述 每个测试文件均包含多组测试数据。第一行输入一个整数$ T (1 ≤ T ≤ 10^4)$,代表数据组数,每组测试数据描述如下: 在一行上…...

Windows下OpenClaw安装指南:一键连接GLM-4.7-Flash模型

Windows下OpenClaw安装指南&#xff1a;一键连接GLM-4.7-Flash模型 1. 为什么选择OpenClawGLM-4.7-Flash组合 去年我在处理日常办公自动化时&#xff0c;发现很多重复性工作既耗时又容易出错。尝试过多个自动化工具后&#xff0c;最终被OpenClaw的"本地化AI智能体"…...

程序员成长之路:从技术热爱到工程艺术

1. 程序人生&#xff1a;从技术热爱到工程艺术1.1 技术启蒙与早期实践1987年进入武汉大学计算机系标志着一段技术人生的开始。最初接触的是Motorola 68000处理器系统&#xff0c;配置540KB内存&#xff0c;运行UNIX操作系统。这种八人共享的计算环境成为编程技术的第一课堂。大…...

想实现SpringCloud的负载均衡,需要实现哪些接口和规范

前几天有个大兄弟问了我一个问题&#xff0c;注册中心要集成SpringCloud&#xff0c;想实现SpringCloud的负载均衡&#xff0c;需要实现哪些接口和规范。既然这个兄弟问到我了&#xff0c;而我又刚好知道&#xff0c;这不得好好写一篇文章来回答这个问题&#xff0c;虽然在后面…...