目录
前言
1、通讯录框架,基本介绍
2、程序实现,步骤+解释
2.1. 建立菜单
2.2. 主函数框架
2.3 前期工作
2.4 初始化通讯录函数:void InitContact(Contact* pc);
2.5 添加信息操作:void AddContact(Contact* pc);
2.6 显示通讯录信息:void ShowContact(Contact* pc);
2.7 删除通讯录信息:void DelContact(Contact* pc);
2.8 查找通讯录信息:void SearchContact(Contact* pc);
2.9 修改通讯录信息:void ModifyContact(Contact* pc);
2.10 排序通讯录信息:void SortContact(Contact* pc);
3 、效果展示
总结
前言
此篇介绍了一个C语言的项目:通讯录。该项目是C语言进阶的实战项目,希望对你有帮助。
1、通讯录框架,基本介绍
一个通讯录的人包含的信息:名字;年龄;性别;电话;地址。
通讯录基本要求:
1、能存放100人的信息
2、能增加联系人
3、能删除指定联系人
4、查找联系人
5、修改联系人
6、排序
7、显示联系人
2、程序实现,步骤+解释
2.1. 建立菜单
首先我们运行程序时必须有一个菜单,指引我们如何操作:
如上图。这样的菜单建立很简单,只需要用到printf函数打印即可:
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");
我们将其封装成一个函数,以便我们多次调用。
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");
}
2.2. 主函数框架
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: //查找
SearchContact(&con);
break;
case 4: //修改
ModifyContact(&con);
break;
case 5: //显示
ShowContact(&con);
break;
case 6: //排序
SortContact(&con);
break;
case 0:
printf("退出通讯录\n");
break;
default:
printf("选择错误,请重新输入\n");
break;
}
} while(input);
return 0;
}
主函数的框架其实很简单,一开始时初始化一个通讯录,用到我们自己定义的函数:InitContact()。然后读取一个输入值,这个值是用户输入的,用户根据菜单打印他想要进行的操作,包括增删查改、排序和显示。此内容于菜单对应。输入1:增加信息;输入2:删除信息;输入3:查找信息;输入4:修改信息;输入5:显示信息;输入6:对信息排序;输入0:退出;其他输入:输入错误,重新输入。这样的逻辑我们用一个switch语句恰好实现,如上图。接下来我们要做的就是实现我们主函数中出现的函数的操作了。
2.3 前期工作
定义通讯录结构体:
typedef struct Contact
{
PeoInfo data[100];//存放人的信息
int count; //记录当前通讯录中实际存储人的信息个数。
}Contact;
此结构体包括,一个存放人的信息的结构体,和一个记录通讯录存储信息个数的变量。
PeoInfo结构体定义:
typedef struct PeoInfo
{
char name[20];
int age;
char sex[10];
char tele[12];
char addr[30];
}PeoInfo;
存放人信息的结构体PeoInfo内容包括:姓名;年龄;性别;电话;地址。
2.4 初始化通讯录函数:void InitContact(Contact* pc);
void InitContact(Contact* pc)
{
assert(pc);
pc->count = 0;
memset(pc->data, 0, sizeof(pc->data));
}
利用memset函数,将传输过来的通讯录结构体初始化,前提包含头文件:string.h。
2.5 添加信息操作:void AddContact(Contact* pc);
void AddContact(Contact* pc)
{
assert(pc);
if (pc->count == MAX)
{
printf("通讯录已满,无法添加\n");
return;
}
//
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");
}
其中MAX表示通讯录最大容量,目前设置为100。当通讯录已满时,无法加入新信息。
通讯录未满,则要求用户对每一个量进行输入,程序将其放在对应位置,最后count++,记录通讯录信息个数的变量加一,表示通讯录又多加了一条信息。
看效果:
可以看到,成功添加了该信息,只是我们测试时还用到了“5”的功能,“5”的功能是显示通讯录现有信息,后面会讲解如何实现。
2.6 显示通讯录信息:void ShowContact(Contact* pc);
我们不按照数字顺序来,我们按照自己的逻辑顺序进行讲解,我先讲这个“5”的功能是怎么实现的,也就是如何显示通讯录:
void ShowContact(Contact* pc)
{
assert(pc);
printf("%-20s\t%-5s\t%-5s\t%-12s\t%-30s\n", "姓名", "年龄", "性别", "电话", "地址");
int i = 0;
for (i = 0;i < pc->count;i++)
{
printf("%-20s\t", pc->data[i].name);
printf("%-5d\t", pc->data[i].age);
printf("%-5s\t", pc->data[i].sex);
printf("%-12s\t", pc->data[i].tele);
printf("%-30s\n", pc->data[i].addr);
}
}
其实逻辑很简单,就是从0开始,一直到count,也就是通讯录有的信息个数,每一个信息都全部打印一遍,用一个for循环就好了。效果就不展示了,上面已经展示了。
2.7 删除通讯录信息:void DelContact(Contact* pc);
void DelContact(Contact* pc)
{
assert(pc);
printf("请输入要删除的人名:>");
char name[20] = { 0 };
scanf("%s", name);
int flag = FindName(pc, name);
if (flag == -1)
{
printf("要删除的信息不存在\n");
return;
}
else
{
int i = 0;
for (i = flag;i < pc->count - 1;i++)
{
pc->data[i] = pc->data[i + 1];
}
pc->count--;
printf("删除成功。\n");
}
}
删除信息的逻辑是:先通过用户输入的名字找到该信息所在的位置,然后用后面的信息覆盖掉这个信息:后面所有信息都向前挪1位。count自减1,表示通讯录存储信息少一个。
这中间需要用到查找信息的函数:FindName。我们来看看它如何实现:
FindName输入参数为:通讯录结构体指针;要查找的名字字符串
返回值为:如果找到返回该数据在data中的位置,找不到返回-1。
static int FindName(Contact* pc,char name[])
{
assert(pc);
int i = 0;
for (i = 0;i < pc->count;i++)
{
if (0 == strcmp(pc->data[i].name, name))
{
return i;
}
}
return -1;
}
static是为了保密,这个作用就相当于给函数加锁,防止给别人看到。可以不加,函数照样实现。
此函数逻辑为,遍历整个通讯录,找到名字和输入的一致时,返回该下表,遍历到通讯录所记录的信息个数时还未找到,返回-1。
删除通讯录信息,效果:
2.8 查找通讯录信息:void SearchContact(Contact* pc);
void SearchContact(Contact* pc)
{
assert(pc);
printf("请输入要查找人的名字:");
char name[20] = { 0 };
scanf("%s", name);
//查找
int flag = FindName(pc, name);
if (flag == -1)
{
printf("找不到该信息。\n");
return;
}
else
{
printf("%-20s\t%-5d\t%-5s\t%-12s\t%-30s\n",pc->data[flag].name,pc->data[flag].age, pc->data[flag].sex, pc->data[flag].tele, pc->data[flag].addr);
}
}
根据输入名字查找,再次运用到函数:FindName。将其返回下表对应的数据输出,如果返回-1,则打印:找不到该信息。
2.9 修改通讯录信息:void ModifyContact(Contact* pc);
void ModifyContact(Contact* pc)
{
assert(pc);
printf("请输入要修改的人名:");
char name[20] = { 0 };
scanf("%s", name);
//查找
int flag = FindName(pc, name);
if (flag == -1)
{
printf("找不到该信息。\n");
return;
}
else
{
printf("请输入修改后的名字:>");
scanf("%s", pc->data[flag].name);
printf("请输入修改后的年龄:>");
scanf("%d", &pc->data[flag].age);
printf("请输入修改后的性别:>");
scanf("%s", pc->data[flag].sex);
printf("请输入修改后的电话:>");
scanf("%s", pc->data[flag].tele);
printf("请输入修改后的地址:>");
scanf("%s", pc->data[flag].addr);
printf("修改成功\n");
}
}
仍旧是先通过输入的名字找到信息对应下标,要用到函数:FindName。然后再用新输入的数据覆盖老的数据,达到修改的效果。
2.10 排序通讯录信息:void SortContact(Contact* pc);
通过冒泡排序,根据信息中的人名进行排序:
void SortContact(Contact* pc)
{
assert(pc);
//冒泡排序,按名字升序
int i = 0;
int j = 0;
PeoInfo C = { 0 };
for (i = 0;i < pc->count;i++)
{
for (j = 1;j < pc->count - i;j++)
{
if (strcmp(pc->data[j - 1].name, pc->data[j].name) > 0)
{
C = pc->data[j - 1];
pc->data[j - 1] = pc->data[j];
pc->data[j] = C;
}
}
}
printf("排序成功。");
}
排序完成,输出:排序成功。
3 、效果展示
总结
通过这样一个项目,可以很好地加强我们对结构体的运用,以及对C语言的掌握。希望这样的讲解对你有所帮助。