通讯录中阶:点这里
实现一个通讯录,每个联系人包含姓名,年龄,性别,电话,地址五种信息,整个通讯录暂定包含100个联系人(后续可以通过宏定义直接修改人数)
这个通讯录包含的功能有增加联系人、删除联系人、查找联系人、修改联系人信息、展示通讯录所有信息、按照指定方式给通讯录排序六种功能
思路大体和扫雷以及n子棋相同,但是初阶是整合了目前所学的知识,以及相关知识的一些很妙的融合
扫雷
n子棋
qsort的使用和实现(实现通讯录sort排序功能的主力函数)
这个问题出现在InitContact这个函数中,当时不理解,尝试很多种方式初始化,但是都是错误的,下面展示一下我多种奇葩的错法,大家引以为戒
在函数中初始化结构体最简单的办法就memset,最好在创建时就直接赋值{0}
上面很多种错误写法就是发生在我想循环输入的这里,一定记住strcpy针对的是字符串,用于给字符串初始化时的0一定也要加双引号表示是字符串,不然类型不同会报语法错误
大家之前在学习枚举常量的时候,是不是一直觉得枚举很鸡肋,视乎排不上用场,今天我就在通讯录项目中使用了两次,目的是可以让代码更加简单易读
这里正是利用了枚举常量的性质,每个值都代表一个数字,可以放在case中充当该case的解释,很生动形象,同时也比用define一个个定义方便很多
这里是我个人平常使用printf时感悟不深所造成的,做个标记
在contact.c文件中,FindByName函数、CmpCharArr函数、CmpInt函数都是专供实现通讯录功能的函数使用的,只会在contact.c文件中被调用,故加上static防止其他文件的调用,这样保证的项目的安全性
qsort的使用和实现(实现通讯录sort排序功能的主力函数)
在sort函数里,为了实现用户指定的排序功能,,引入switch实现用户需要指定的信息,引入flag实现升序降序
#pragma once
#include
#include
#include
#include
#define MAX_CONTACT 100
#define MAX_NAME 20
#define MAX_SEX 5
#define MAX_TELE 12
#define MAX_ADDR 30
enum OPTION
{
EXIT0,
ADD,
DEL,
SEARCH,
MODIFY,
SHOW,
SORT
};
enum Peo
{
EXIT1,
NAME,
AGE,
SEX,
TELE,
ADDR
};
//一个人信息的结构体
typedef struct People
{
char name[MAX_NAME];
int age;
char sex[MAX_SEX];
char tele[MAX_TELE];
char addr[MAX_ADDR];
}Peo;
//存放大量人信息的通讯录
typedef struct Contact
{
Peo data[MAX_CONTACT];
int sz;
}Con;
//初始化通讯录
void InitContact(Con* pc);
//给通讯录添加联系人
void AddContact(Con* pc);
//显示通讯录信息
void ShowContact(const Con* pc);
//删除指定联系人
void DelContact(Con* pc);
//查找指定联系人
void SearchContact(const Con* pc);
//修改指定联系人的信息
void ModifyContact(Con* pc);
//按照指定方式排序
void SortContact(Con* pc);
#define _CRT_SECURE_NO_WARNINGS 1
#include"contact.h"
//初始化通讯录——————————————————————————————————————————————————————————
void InitContact(Con* pc)
{
assert(pc);
//循环初始化也可
//int i = 0;
//for (i = 0; i < MAX_CONTACT; i++)
//{
// strcpy(pc->data[i].name, "0");
// pc->data[i].age = 0;
// strcpy(pc->data[i].sex, "0");
// strcpy(pc->data[i].tele, "0");
// strcpy(pc->data[i].addr, "0");
//}
//pc->sz = 0;
//初始化结构体数组最简单的方法
memset(pc->data, 0, sizeof(pc->data));
pc->sz = 0;
}
//给通讯录添加联系人——————————————————————————————————————————————————————
void AddContact(Con* pc)
{
assert(pc);
if (pc->sz == MAX_CONTACT)
{
printf("通讯录已满,无法添加\n");
return;
}
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].addr);
pc->sz++;
printf("增加联系人成功\n");
}
//显示通讯录信息——————————————————————————————————————————————————————————
void ShowContact(const Con* pc)
{
assert(pc);
printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");
int i = 0;//如果后面需要i的值,就不能定义在for循环的初始化部分,因为出了循环就被销毁
for (i = 0; i < pc->sz; i++)
{
printf("%-20s\t%-4d\t%-5s\t%-12s\t%-30s\n",
pc->data[i].name,
pc->data[i].age,
pc->data[i].sex,
pc->data[i].tele,
pc->data[i].addr);
}
}
//删除指定联系人——————————————————————————————————————————————————————————
static int FindByName(const Con* pc,char* name)
{
int i = 0;
for (i = 0; i < pc->sz; i++)
{
if (strcmp(pc->data[i].name, name) == 0)
{
return i;
}
}
return -1;
}
void DelContact(Con* pc)
{
assert(pc);
if (pc->sz == 0)
{
printf("通讯录为空\n");
return;
}
char name[MAX_NAME] = { 0 };
printf("请输入要删除的人的名字:>");
scanf("%s", name);
int pos = FindByName(pc, name);
if (pos == -1)
{
printf("要删除的人不存在\n");
return;
}
//删除方法一:从后向前一个个覆盖
//for (i = pos; i < pc->sz - 1; i++)
//{
// pc->data[i] = pc->data[i + 1];
//}
//pc->sz--;
//删除方法二:memmove,和方法一相同
//memmove(&(pc->data[pos]), &(pc->data[pos + 1]), ((pc->sz) - pos - 1) * (sizeof(pc->data[0])));
//pc->sz--;
//删除方法三:将要删除的和最后一个交换,然后sz--
Peo tmp = pc->data[pos];
pc->data[pos] = pc->data[pc->sz - 1];
pc->data[pc->sz - 1] = tmp;
pc->sz--;
printf("删除联系人成功\n");
}
//查找指定联系人————————————————————————————————————————————————————————————
void SearchContact(const Con* pc)
{
assert(pc);
//学习c++之后可以用函数重载实现用任何信息都能进行检索
char name[MAX_NAME] = { 0 };
printf("请输入要查找人的名字:>");
scanf("%s", name);
int pos = FindByName(pc, name);
if (pos == -1)
{
printf("要查找的人不存在\n");
}
else
{
printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");
printf("%-20s\t%-4d\t%-5s\t%-12s\t%-30s\n",
pc->data[pos].name,
pc->data[pos].age,
pc->data[pos].sex,
pc->data[pos].tele,
pc->data[pos].addr);
}
}
//修改指定联系人的信息——————————————————————————————————————————————————————
void ModifyContact(Con* pc)
{
assert(pc);
char name[MAX_NAME] = { 0 };
printf("请输入要修改人的名字:>");
scanf("%s", name);
int pos = FindByName(pc, name);
if (pos == -1)
{
printf("要修改的人不存在");
}
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");
//结合switch指定修改某个信息
int input = 0;
do
{
printf("0.EXIT1 1.NAME 2.AGE 3.SEX 4.TELE 5.ADDR\n");
printf("请选择要修改的信息or选择0退出修改:>");
scanf("%d", &input);
switch (input)
{
case NAME:
printf("请输入修改后的名字:>");
scanf("%s", pc->data[pos].name);
break;
case AGE:
printf("请输入修改后的年龄:>");
scanf("%d", &(pc->data[pos].age));
break;
case SEX:
printf("请输入修改后的性别:>");
scanf("%s", pc->data[pos].sex);
break;
case TELE:
printf("请输入修改后的电话:>");
scanf("%s", pc->data[pos].tele);
break;
case ADDR:
printf("请输入修改后的地址:>");
scanf("%s", pc->data[pos].addr);
break;
case EXIT1:
break;
default:
printf("选择错误,重新选择\n");
break;
}
} while (input);
printf("修改成功\n");
}
}
//按照指定方式排序——————————————————————————————————————————————————————————————
int flag = 0;//qsort和Cmp函数已经固定好参数,要想体现升序和降序,只能定义全局变量,然后在函数里调用
static int CmpCharArr(const void* p1, const void* p2)
{
return flag * (strcmp((*(Peo*)p1).name, (*(Peo*)p2).name));
}
static int CmpInt(const void* p1, const void* p2)
{
return flag * (((*(Peo*)p1).age) - ((*(Peo*)p2).age));
}
void SortContact(Con* pc)
{
assert(pc);
int input = 0;
do
{
printf("0.EXIT1 1.NAME 2.AGE 3.SEX 4.TELE 5.ADDR\n");
printf("请选择要按照哪种方式排序or选择0退出排序:>");
scanf("%d", &input);
if (input != 0)
{
printf("升序选择1,降序选择-1:>");
while (flag != 1 && flag != -1)
{
scanf("%d", &flag);
if (flag != 1 && flag != -1)
{
printf("输入错误,请重新输入\n");
}
}
}
switch (input)
{
case NAME:
qsort(pc, pc->sz, sizeof(pc->data[0]),CmpCharArr);
printf("排序成功\n");
break;
case AGE:
qsort(pc, pc->sz, sizeof(pc->data[0]), CmpInt);
printf("排序成功\n");
break;
case SEX:
qsort(pc, pc->sz, sizeof(pc->data[0]), CmpCharArr);
printf("排序成功\n");
break;
case TELE:
qsort(pc, pc->sz, sizeof(pc->data[0]), CmpCharArr);
printf("排序成功\n");
break;
case ADDR:
qsort(pc, pc->sz, sizeof(pc->data[0]), CmpCharArr);
printf("排序成功\n");
break;
case EXIT1:
break;
default:
printf("选择错误,重新选择\n");
break;
}
} while (input);
}
#define _CRT_SECURE_NO_WARNINGS 1
#include"contact.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");
}
void test()
{
int input = 0;
Con con;
InitContact(&con);
do
{
menu();
printf("请选择:>");
scanf("%d", &input);
switch (input)
{
case ADD:
AddContact(&con);
break;
case DEL:
DelContact(&con);
break;
case SEARCH:
SearchContact(&con);
break;
case MODIFY:
ModifyContact(&con);
break;
case SHOW:
ShowContact(&con);
break;
case SORT:
//c++要是有函数重载会好写很多
SortContact(&con);
break;
case EXIT0:
printf("退出通讯录\n");
break;
default:
printf("选择错误,重新选择\n");
break;
}
} while (input);
}
int main()
{
test();
return 0;
}