(C语言)静态通讯录(正式版)(C语言小项目)

news2025/3/29 13:41:26
 1.首先是头文件:
//头文件
//contact.h

//防止头文件被重复包含
#pragma once
//定义符号常亮,方便维护和修改
//联系人基本信息容量
#define NAME_MAX 20
#define AGE_MAX 5
#define SEX_MAX 5
#define TELE_MAX 15
#define ADDR_MAX 30
//联系人最大容量100
#define MAX 100

//定义联系人结构体
struct PeopleInfo
{
	char name[NAME_MAX];
	char age[AGE_MAX];
	char sex[SEX_MAX];
	char tele[TELE_MAX];
	char addr[ADDR_MAX];
};
//定义通讯录结构体
struct Contact
{
	struct PeopleInfo data[MAX];
	int sz;
};

//声明函数
void clear_screen();
void AddContact(struct Contact* con);
void DelContact(struct Contact* con);
void ShowContact(struct Contact* con);
int FindContact(const struct Contact* con,char name[]);
void InitContact(struct Contact* con);
void menu();
void ModifyContact(struct Contact* con);
void SearchContact(struct Contact* con);
void SortContact(struct Contact* con);
void ClearContact(struct Contact* con);
2. 然后是功能函数contact.c文件
//功能函数文件
//contact.c	

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include "contact.h"

//定义菜单函数
void menu()
{
	printf("*********************************************\n");
	printf("******** 1.添加        2.删除  **************\n");
	printf("******** 3.查询        4.修改  **************\n");
	printf("******** 5.查看        6.排序  **************\n");
	printf("******** 7.清空        0.退出  **************\n");
	printf("*********************************************\n");
}

//定义清屏函数
//清屏操作
void clear_screen() {
	//判断是否为Windows系统
	#ifdef _WIN32
		system("cls");
	//其他系统
	#else
		system("clear");
	#endif
}

//初始化通讯录函数
void InitContact(struct Contact* con)
{
	con->sz=0;//当前联系人数量:0
	memset(con->data, 0, MAX * sizeof(struct PeopleInfo));//清空数组
}

//添加联系人函数
void AddContact(struct Contact* con) {
	//检查通讯录是否溢出
	if (con->sz == MAX) {
		printf("通讯录已满!\n");
	}
	else
	{
		printf("请输入姓名:");
		scanf("%s", con->data[con->sz].name);
		printf("请输入年龄:");
		scanf("%s", con->data[con->sz].age);
		printf("请输入性别:");
		scanf("%s", con->data[con->sz].sex);
		printf("请输入电话:");
		scanf("%s", con->data[con->sz].tele);
		printf("请输入地址:");
		scanf("%s", con->data[con->sz].addr);
		printf("添加成功!\n");
		(con->sz)++;
	}
}

//查找当前联系人函数
int  FindContact(const struct Contact* con,char name[]) {
	for (int i = 0; i < con->sz; i++) {
		//利用比较函数strcmp判断姓名是否相等
		if (strcmp(con->data[i].name, name) == 0) {
			return i;
		}
	}
	printf("用户不存在!\n");
	return -1;
}

//删除联系人函数
void DelContact(struct Contact* con) {
	if (con->sz == 0) {
		printf("通讯录为空!\n");
	}
	else
	{
		printf("请输入要删除的联系人姓名:\n");
		char name[NAME_MAX];
		scanf("%s", name);
		int ret = FindContact(con, name);
		if (ret != -1) {
			printf("删除成功!\n");
			for (int i = ret; i < con->sz-1; i++) {
				con->data[i] = con->data[i + 1];
			}
			(con->sz)--;
		}
	}
}

//查询当前通讯录
void ShowContact(struct Contact* con) {
	if (con->sz == 0) {
		printf("该通讯录为空\n");
	}
	else {
		printf("%-15s\t%-5s\t%-8s\t%-15s\t%-30s\n", "姓名", "年龄", "性别", "电话", "地址");

		for (int i = 0; i < con->sz; i++) {
			printf("%-15s\t%-5s\t%-8s\t%-15s\t%-30s\n", con->data[i].name,
				con->data[i].age, con->data[i].sex, con->data[i].tele, con->data[i].addr);
		}
	}
}

//查询联系人函数
void SearchContact(struct Contact* con) {
	if (con->sz == 0) {
		printf("当前通讯录为空\n");
	}
	else
	{
		printf("请输入要查询的联系人姓名:\n");
		char name[NAME_MAX];
		scanf("%s", name);
		int ret = FindContact(con, name);
		if (ret != -1) {
			printf("查询成功!\n");
			printf("%-15s\t%-5s\t%-8s\t%-15s\t%-30s\n", "姓名", "年龄", "性别", "电话", "地址");
			printf("%-15s\t%-5s\t%-8s\t%-15s\t%-30s\n",con->data[ret].name,
				con->data[ret].age, con->data[ret].sex, con->data[ret].tele, con->data[ret].addr);
		}
	}
}

//修改联系人函数
void ModifyContact(struct Contact* con) {
	if (con->sz == 0) {
		printf("通讯录为空\n");
	}
	else
	{
		printf("请输入要修改的联系人姓名:\n");
		char name[NAME_MAX];
		scanf("%s", name);
		int Mod_0 = 2;
		int ret = FindContact(con, name);
		if (ret != -1) {
			printf("当前联系人数据!\n");
			printf("%-15s\t%-5s\t%-8s\t%-15s\t%-30s\n", "姓名", "年龄", "性别", "电话", "地址");
			printf("%-15s\t%-5s\t%-8s\t%-15s\t%-30s\n", con->data[ret].name,
				con->data[ret].age, con->data[ret].sex, con->data[ret].tele, con->data[ret].addr);
			printf("是否确定修改??(按1继续,按0退出)\n");
			while (1) {
				if (scanf("%d", &Mod_0) != 1) {
					printf("输入不合法,请重新输入");
				}
				if (Mod_0 == 1) {
					printf("请输入姓名:");
					scanf("%s", con->data[ret].name);
					printf("请输入年龄:");
					scanf("%s", con->data[ret].age);
					printf("请输入性别:");
					scanf("%s", con->data[ret].sex);
					printf("请输入电话:");
					scanf("%s", con->data[ret].tele);
					printf("请输入地址:");
					scanf("%s", con->data[ret].addr);
					printf("修改成功!\n");
					break;
				}
				else if (Mod_0==0) {
					break;
				}
			}
		}
	}
}


//比较函数
int by_name_cmp1(const void* x1, const void* x2) {
	return strcmp(((struct PeopleInfo*)x1)->name, ((struct PeopleInfo*)x2)->name);
}
int by_name_cmp2(const void* x1, const void* x2) {
	return strcmp(((struct PeopleInfo*)x2)->name, ((struct PeopleInfo*)x1)->name);
}
//排序联系人函数
void SortContact(struct Contact* con) {
	if (con->sz == 0) {
		printf("通讯录为空\n");
	}
	else {
		int Sort_0 = 2;
		printf("请选择排序方式(按1升序,按0降序):\n");
		while (1) {
			if (scanf("%d", &Sort_0) != 1) {
				printf("输入不合法,请重新输入");
			}
			if (Sort_0 == 1) {
				qsort(con->data, con->sz, sizeof(struct PeopleInfo), by_name_cmp1);
				printf("升序排序已完成\n");
				break;
			}
			else if(Sort_0 == 0)
			{
				qsort(con->data, con->sz, sizeof(struct PeopleInfo), by_name_cmp2);
				printf("降序排序已完成\n");
				break;
			}
		}
	}
}
//清空通讯录函数
void ClearContact(struct Contact* con) {
	if (con->sz == 0) {
		printf("通讯录为空\n");
	}
	else
	{
		con->sz = 0;
		memset(con->data, 0, MAX * sizeof(struct PeopleInfo));
		printf("当前通讯录已经清空\n");
	}
}
3.最后是主程序test.c文件:
//主程序文件
//test.c

#include <stdio.h>
#include <stdlib.h>
#include "contact.h"

//枚举条件选择定义(搭配switch使用)
enum Option
{
	EXIT,//0,对应退出通讯录
	ADD,//1,对应添加联系人
	DEL,//2,对应删除联系人
	SEARCH,//3,对应查询联系人
	MODIFY,//4,对应修改联系人
	SHOW,//5,对应查看通讯录
	SORT,//6,对应排序通讯录
	CLEAR,//7,对应清空通讯录
};

//主函数
int main()
{
	int input = 0;
	int menu_0 = 0;
	//创建通讯录
	struct Contact con;
	//调用函数初始化通讯录
	InitContact(&con);//传递参数地址

	do
	{
		//打印菜单
		while (1) {
			printf("************按1继续************\n");
			if (scanf("%d", &menu_0) != 1 && menu_0!=1) {
				printf("输入不合法,请按1继续\n");
				return 1;
			}
			clear_screen();
			if (menu_0==1)
			{
				menu();
				break;
			}
		}
		printf("请选择对应模式(0-7):\n");
		if (scanf("%d", &input) != 1 || input < 0 || input > 7){
			printf("输入不合法,请输入整数0-7\n");
			return 1;
		}
		switch (input)
		{
		case EXIT: {
			clear_screen();
			printf("退出通讯录!\n");
			break;
		}
		case ADD: {
			clear_screen();
			AddContact(&con);
			break;
		}
		case DEL: {
			clear_screen();
			DelContact(&con);
			break;
		}
		case SEARCH: {
			clear_screen();
			SearchContact(&con);
			break;
		}
		case SHOW: {
			clear_screen();
			ShowContact(&con);
			break;
		}
		case MODIFY: {
			clear_screen();
			ModifyContact(&con);
			break;
		}
		case SORT: {
			clear_screen();
			SortContact(&con);
			break;
		}
		case CLEAR: {
			clear_screen();
			ClearContact(&con);
			break;
		}
		default:
			break;
		}

	} while (input);
    return 0;
}

整个项目只有三个文件,头文件和两个源代码

相对于测试版,添加并优化了很多代码,下面我来介绍更新内容:

添加清屏操作,是终端显示更加美观

  • void clear_screen() {
    	//判断是否为Windows系统
    	#ifdef _WIN32
    		system("cls");
    	//其他系统
    	#else
    		system("clear");
    	#endif
    }

    先判断系统,是Windows系统就执行第一个,其他系统就执行第二个

 运用排序函数qsort,添加头文件,#include <stdlib.h>

  • void qsort(
        void* base,         // 数组起始地址
        size_t num,         // 元素个数
        size_t size,        // 单个元素大小(字节)
        int (*compar)(const void*, const void*) // 比较函数
    );

    这是他的原型,在本次项目中,是:

  • void SortContact(struct Contact* con) {
        qsort(
            con->data,          // 排序的数组
            con->sz,            // 元素个数
            sizeof(struct PeopleInfo), // 每个元素的大小
            cmp_by_name         // 比较函数
        );
    }

补齐其他功能块,运用一些小的库函数,例如比较函数strcmp

  • int strcmp(const char* str1, const char* str2);
     这是他的原型,在本项目中:
  • int cmp_by_name(const void* x1, const void* x2) {
        return strcmp(
            ((struct PeopleInfo*)x1)->name, 
            ((struct PeopleInfo*)x2)->name
        );
    }

重点更新内容介绍完了,还有很多细节优化,希望大家可以自己探索

 (C语言)静态通讯录(测试版)(C语言小项目)-CSDN博客

测试版在这,里面有关键代码解释,大家可以看看

运行结果就不展示了

源代码:

(C语言)静态通讯录(C语言小项目)

注:该代码是本人自己所写,可能不够好,不够简便,欢迎大家指出我的不足之处。如果遇见看不懂的地方,可以在评论区打出来,进行讨论,或者联系我。上述内容全是我自己理解的,如果你有别的想法,或者认为我的理解不对,欢迎指出!!!如果可以,可以点一个免费的赞支持一下吗?谢谢各位彦祖亦菲!!!!!

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

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

相关文章

[特殊字符] 2025蓝桥杯备赛Day13——P10984 [蓝桥杯 2023 国 Python A] 残缺的数字

&#x1f50d; 2025蓝桥杯备赛Day13——P10984 [蓝桥杯 2023 国 Python A] 残缺的数字 &#x1f680; 题目速览 题目难度&#xff1a;⭐⭐⭐&#xff08;需掌握位运算与组合数学&#xff09; 考察重点&#xff1a;二进制状态处理、位运算、乘法原理、枚举 P10984 [蓝桥杯 2…

线程控制与线程库

目录 解析tid 线程的地址空间布局 线程栈 我们来学习线程控制与线程库 解析tid #include<iostream> #include<string> #include<cstdio> #include<cstring> #include<unistd.h> #include<thread> using namespace std;int shared_val…

P1182 数列分段 Section II

P1182 数列分段 Section II - 洛谷 题目描述 对于给定的一个长度为 N 的正整数数列 A1​∼AN​&#xff0c;现要将其分成 M&#xff08;M≤N&#xff09;段&#xff0c;并要求每段连续&#xff0c;且每段和的最大值最小。 关于最大值最小&#xff1a; 例如一数列 4 2 4 5 1…

比手动备份快 Iperius全自动加密备份,NAS/云盘/磁带机全兼容

IperiusBackupFull是一款专为服务器和工作站设计的备份解决方案&#xff0c;它同时也是一款针对Windows 7/8/10/11/Server系统的简洁且可靠的备份软件。该软件支持增量备份、数据同步以及驱动器镜像&#xff0c;确保能够实现完全的系统恢复。在备份存储方面&#xff0c;Iperius…

2025最新版Ubuntu Server版本Ubuntu 24.04.2 LTS下载与安装-详细教程,细致到每一步都有说明

官网 https://ubuntu.com/ 下载 点击菜单 Prodercts> Ubuntu OS>Ubuntu Server 点击下载 下载后会有个弹窗 安装 选择第一个 install Ubuntu Server 直接默认&#xff0c;选择English 【默认】 选择键盘布局【默认】 选择安装配置【默认】 配置网络 我这里选择…

更新测试环境构建命令以解决构建失败问题

本段代码解决 更新测试环境构建命令以解决构建失败问题 //本项目是reactumi3antdesign 搭建的后台管理系统 "build:test": "cross-env UMI_ENVtest NODE_OPTIONS--openssl-legacy-provider umi build"**原因&#xff1a;**Node.js v17 的 OpenSSL 3.0 与旧…

树莓派5-GPIO和40针引脚

1.树莓派5引脚图 2.GPIO 引脚作用 (1) 电压 板上有两个 5V 引脚和两个 3.3V 引脚&#xff0c;以及一些不可配置的接地引脚 (0V)。其余引脚均为通用 3.3V 引 脚&#xff0c;这意味着输出设置为 3.3V&#xff0c;输入可接 3.3V。 (2) 输出 指定为输出引脚的 GPIO 引脚可设置为…

【数据库】sql错题详解

1. 执行子查询 SELECT 供应商号 FROM 订购单 WHERE 职工号 IN (E1, E3) GROUP BY 供应商号 HAVING COUNT(DISTINCT 职工号) 2筛选职工号为 E1 或 E3 的记录&#xff1a; 依据 WHERE 职工号 IN (E1, E3) 这个条件&#xff0c;从 订购单 表中把职工号为 E1 或者 E3 的记录筛选出…

C#重写treeView控件

1.先准备两张图片downdrop.png、downdrop_open.png放在项目Resources里 2.新建用户控件BaseTreeView控件 3.重写控件继承TreeView&#xff0c;记得删除AutoScaleMode这一行&#xff0c;否则会报错 public partial class BaseTreeView : TreeView {//这个属性貌似不起作用&…

ArcGIS 10.8.1之后发布栅格数据的MapServer 动态工作空间 替换数据源渲染问题

背景 经过测试&#xff0c;Server 10.8.1、11.0、11.1发布相关服务设置动态空间之后&#xff0c;前端都无法自动读取同名的clr色彩映射表文件进行渲染&#xff0c;服务都是由ArcGIS Pro进行发布。 原因 基于ArcMap发布的服务才支持&#xff0c;但是10.8.1之后不支持ArcMap发…

Java集合框架深度剖析:从数据结构到实战应用

引言 Java集合框架是Java开发中的核心组件之一&#xff0c;其设计目标是提供高性能、高复用性的数据容器。无论是数据处理、缓存设计还是高并发场景&#xff0c;集合框架都扮演着关键角色。本文将从List、Map、Set三大核心接口出发&#xff0c;深入剖析其主流实现类&#xff0…

涅槃上岸,入陕进军,复试全程流程开启!

复试决胜局&#xff0c;整装待发&#xff0c;上岸西电&#xff01; 线下复试注意事项、全流程、录取后西安旅游提前告知&#xff01; 过两天考研复试笔试、机试&#xff08;如果有&#xff09;、面试就要开始了&#xff0c;我们需要准备很多东西&#xff0c;学长从以下几个方面…

msyql--基本操作之运维篇

检查 root 用户的权限 查看该用户针对这个数据库的权限 -- 如果在终端连接mysql时需要 mysql -u root -p -- 查看用户权限 SELECT user, host FROM mysql.user WHERE user root;可以看的出来root有他的访问权限&#xff0c;如过没有localhost或者% 说明没有访问权限 添加…

es6 fetch

对比XHR &#x1f6e0;️ fetch 所有配置项 fetch(url, {// 核心配置 method: GET, // HTTP 方法: GET, POST, PUT, DELETE, PATCH, HEAD, OPTIONSheaders: { // 请求头&#xff08;支持 Headers 对象或普通对象&#xff09;Content-Type: applicati…

智能汽车图像及视频处理方案,支持视频星轨拍摄能力

美摄科技作为智能汽车图像及视频处理领域的先行者&#xff0c;正以革新性的技术引领着行业的未来发展。美摄科技智能汽车图像及视频处理方案&#xff0c;一个集高效性、智能化、画质增强于一体的创新解决方案&#xff0c;旨在重塑智能汽车图像画质的新标准&#xff0c;并支持前…

C盘急救实录:从爆红到畅快

极速救援通道&#xff08;懒人专享&#xff09; 老规矩&#xff0c;先上王炸方案&#xff01;”小番茄C盘清理器”直达链接&#xff1a;https://cclean-cdn.xkbrowser.com/cleanmaster/FanQieClean_13046_st.exe 这个神器有三绝&#xff1a; 智能扫描引擎&#xff1a;能识别23…

UART转APB模块ModelSim仿真

一、简介 之前介绍过一个UART转AHB模块&#xff0c;这个代码的框架有个好处&#xff0c;就是FPGA内总线接口比较容易修改成其他总线接口。下图是UART转AHB模块中子模块uart_ahb_mst的框图&#xff0c;主要有三个状态机&#xff1a; &#xff08;1&#xff09; UART_RX_FSM将接收…

Cesium 自定义路径导航材质

cesium 自定义路径导航纹理图片随便更换&#xff0c;UI 提供设计图片即可达到效果&#xff1b; 打开小马的weix 关注下 搜索“技术链” 回复关键词《《路径》》获取原始代码&#xff1b; 拿到就能用轻松解决&#xff01;帮忙点个关注吧&#xff01;

【江协科技STM32】BKP备寄存器RTC实时时钟(学习笔记)

BKP备寄存器 BKP简介 BKP&#xff08;Backup Registers&#xff09;备份寄存器BKP可用于存储用户应用程序数据。当VDD&#xff08;2.0~3.6V&#xff09;电源被切断&#xff0c;他们仍然由VBAT&#xff08;1.8~3.6V&#xff09;维持供电。当系统在待机模式下被唤醒&#xff0…

卷积神经网络 - 参数学习

本文我们通过两个简化的例子&#xff0c;展示如何从前向传播、损失计算&#xff0c;到反向传播推导梯度&#xff0c;再到参数更新&#xff0c;完整地描述卷积层的参数学习过程。 一、例子一 我们构造一个非常简单的卷积神经网络&#xff0c;其结构仅包含一个卷积层和一个输出…