我们想实现一个通讯录; 通讯录可以用来存储1000个人的信息,每个人的信息包括:姓名、性别、年龄、电话、住址。 通讯录的功能有: 1.添加联系人信息 2.删除指定联系人信息 3.查找指定联系人信息 4.修改指定联系人信息 5.显示所有联系人信息 6.清空所有联系人 7.以名字排序所有联系人
1.通讯录菜单
void menu()
{
printf("*****************************************\n");
printf("******* 1.add 2.del *******\n");
printf("******* 3.search 4.modify *******\n");
printf("******* 5.show 6.empty *******\n");
printf("******* 7.sort 0.exit *******\n");
printf("*****************************************\n");
}
2.通讯录主函数
int main()
{
int input = 0;
do
{
menu();
printf("请输入:>");
scanf("%d", &input);
switch (input)
{
case 1:
break;
case 2:
break;
case 3:
break;
case 4:
break;
case 5:
break;
case 6:
break;
case 7:
break;
case 0:
printf("退出通讯录\n");
break;
default:
printf("选择错误");
break;
}
} while (input);
return 0;
}
3.类型的声明
3.1 联系人的信息
typedef struct PeoInfo
{
char name[MAX_NAME];
int age;
char sex[MAX_SEX];
char tele[MAX_TELE];
char addr[MAX_ADDR];
}PeoInfo;
3.2 通讯录的内容
typedef struct Contact
{
PeoInfo data[MAX];
int count;
}Contact;
4.初始化通讯录
void InitContact(Contact* pc)
{
assert(pc);
memset(pc->data, 0, sizeof(pc->data));
pc->count = 0;
}
5.实现通讯录的功能
5.1添加联系人信息
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");
}
5.2删除指定联系人信息
删除分为两步:①找到指定联系人,②删除
void DelContact(Contact* pc)
{
char name[MAX_NAME] = { 0 };
assert(pc);
if (pc->count == 0)
{
printf("通讯录为空,无法删除信息\n");
return;
}
printf("请输入你要删除的名字:>");
scanf("%s", name);
int pos = FindByName(pc, name);
if (pos == -1)
{
printf("要删除的人不存在\n");
return;
}
int i = 0;
for (i = pos; i < pc->count - 1; i++)
{
pc->data[i] = pc->data[i + 1];
}
pc->count--;
printf("删除成功\n");
}
5.3查找指定联系人信息
static int FindByName(Contact* pc, char name[])
{
assert(pc);
int i = 0;
for (i = 0; i < pc->count; i++)
{
if (strcmp(pc->data[i].name, name) == 0)
{
return i;
}
}
return -1;
}
5.4修改指定联系人信息
修改分为两步:①找到指定联系人,②修改
void ModifyContact(Contact* pc)
{
assert(pc);
printf("请输入你要修改的联系人的姓名:>");
char name[MAX_NAME] = { 0 };
scanf("%s", name);
int ret = FindByName(pc, name);
if (ret == -1)
{
printf("要修改的人不存在\n");
return;
}
printf("请输入修改后的名字:>");
scanf("%s", pc->data[ret].name);
printf("请输入修改后的年龄:>");
scanf("%d", &(pc->data[ret].age));
printf("请输入修改后的性别:>");
scanf("%s", pc->data[ret].sex);
printf("请输入修改后的电话:>");
scanf("%s", pc->data[ret].tele);
printf("请输入修改后的地址:>");
scanf("%s", pc->data[ret].addr);
printf("修改完成\n");
}
5.5显示所有联系人信息
void ShowContact(const Contact* pc)
{
assert(pc);
printf("%-20s\t%-5s\t%-5s\t%-12s\t%-30s\t\n", "姓名", "年龄", "性别", "电话", "地址");
int i = 0;
for (i = 0; i < pc->count; i++)
{
printf("%-20s\t%-5d\t%-5s\t%-12s\t%-30s\t\n", pc->data[i].name,
pc->data[i].age,
pc->data[i].sex,
pc->data[i].tele,
pc->data[i].addr);
}
}
5.6以名字排序所有联系人
方法1
int cmp (const void* e1, const void* e2)
{
return strcmp(((PeoInfo*)e1)->name,((PeoInfo*)e2)->name);
}
void SortContact(Contact* pc)
{
assert(pc);
qsort(pc->data, pc->count, sizeof(PeoInfo),
cmp);
printf("排序成功\n");
}
方法2
void SortContact(Contact* pc)
{
assert(pc);
int i = 0;
int j = 0;
for (i = 0; i < pc->count - 1; i++)
{
for (j = 0; j < pc->count - 1 - i; j++)
{
if (strcmp(pc->data[j].name, pc->data[j + 1].name) > 0)
{
PeoInfo tmp = pc->data[j];
pc->data[j] = pc->data[j + 1];
pc->data[j + 1] = tmp;
}
}
}
printf("排序成功\n");
}
5.7清空所有联系人
void EmptyContact(Contact* pc)
{
assert(pc);
memset(pc->data, 0, sizeof(pc->data));
pc->count = 0;
printf("已清空所有联系人\n");
}
6.枚举选项
enum Option
{
EXIT,
ADD,
DEL,
SEARCH,
MODIFY,
SHOW,
EMPTY,
SORT
};
7.动态增长空间
通讯录初始容量设置为3个,通讯录满了之后每次增加2个
typedef struct Contact
{
PeoInfo* data;
int count;
int capacity;
}Contact;
int InitContact(Contact* pc)
{
assert(pc);
pc->count = 0;
pc->data=(PeoInfo*)calloc(DEFAULT_SZ,sizeof(PeoInfo));
if (pc->data == NULL)
{
printf("InitContact::%s\n", strerror(errno));
return 1;
}
pc->capacity = 3;
return 0;
}
void Check_Capacity(Contact* pc)
{
if (pc->count == pc->capacity)
{
PeoInfo* ptr = (PeoInfo*)realloc(pc->data, sizeof(PeoInfo) * (pc->capacity + INC_SZ));
if (ptr == NULL)
{
printf("AddContact::%s\n", strerror(errno));
return;
}
else
{
pc->data = ptr;
pc->capacity += INC_SZ;
printf("增容成功\n");
}
}
}
8.通讯录信息保存到文件中
void SaveContact(const Contact* pc)
{
assert(pc);
FILE* pfWrite = fopen("contact.txt", "wb");
if (pfWrite == NULL)
{
perror("SaveContact");
return;
}
int i = 0;
for (i = 0; i < pc->count; i++)
{
fwrite(pc->data + i,sizeof(PeoInfo),1,pfWrite);
}
fclose(pfWrite);
pfWrite = NULL;
}
9.文件中信息加载到通讯录中
void LoadContact(Contact* pc)
{
assert(pc);
FILE* pfRead = fopen("contact.txt","rb");
if (pfRead == NULL)
{
perror("LoadContact");
return;
}
PeoInfo tmp = { 0 };
while (fread(&tmp, sizeof(PeoInfo), 1, pfRead) == 1)
{
Check_Capacity(pc);
pc->data[pc->count] = tmp;
pc->count++;
}
fclose(pfRead);
pfRead = NULL;
}
10.完整代码
10.1 contact.h
#pragma once
#include <stdio.h>
#include <assert.h>
#include <string.h>
#include <stdlib.h>
#include <errno.h>
#define MAX 1000
#define MAX_NAME 20
#define MAX_SEX 10
#define MAX_TELE 12
#define MAX_ADDR 30
#define DEFAULT_SZ 3
#define INC_SZ 2
typedef struct PeoInfo
{
char name[MAX_NAME];
int age;
char sex[MAX_SEX];
char tele[MAX_TELE];
char addr[MAX_ADDR];
}PeoInfo;
typedef struct Contact
{
PeoInfo* data;
int count;
int capacity;
}Contact;
int InitContact(Contact* pc);
void DestroyContact(Contact* pc);
void AddContact(Contact* pc);
void ShowContact(const Contact* pc);
void DelContact(Contact* pc);
void SearchContact(const Contact* pc);
void ModifyContact(Contact* pc);
void SortContact(Contact* pc);
void EmptyContact(Contact* pc);
void SaveContact(const Contact* pc);
void LoadContact(Contact* pc);
10.2 contact.c
#define _CRT_SECURE_NO_WARNINGS
#include "contact.h"
void Check_Capacity(Contact* pc)
{
if (pc->count == pc->capacity)
{
PeoInfo* ptr = (PeoInfo*)realloc(pc->data, sizeof(PeoInfo) * (pc->capacity + INC_SZ));
if (ptr == NULL)
{
printf("AddContact::%s\n", strerror(errno));
return;
}
else
{
pc->data = ptr;
pc->capacity += INC_SZ;
printf("增容成功\n");
}
}
}
void LoadContact(Contact* pc)
{
assert(pc);
FILE* pfRead = fopen("contact.txt","rb");
if (pfRead == NULL)
{
perror("LoadContact");
return;
}
PeoInfo tmp = { 0 };
while (fread(&tmp, sizeof(PeoInfo), 1, pfRead) == 1)
{
Check_Capacity(pc);
pc->data[pc->count] = tmp;
pc->count++;
}
fclose(pfRead);
pfRead = NULL;
}
int InitContact(Contact* pc)
{
assert(pc);
pc->count = 0;
pc->data=(PeoInfo*)calloc(DEFAULT_SZ,sizeof(PeoInfo));
if (pc->data == NULL)
{
printf("InitContact::%s\n", strerror(errno));
return 1;
}
pc->capacity = 3;
LoadContact(pc);
return 0;
}
void AddContact(Contact* pc)
{
assert(pc);
Check_Capacity(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");
}
void ShowContact(const Contact* pc)
{
assert(pc);
printf("%-20s\t%-5s\t%-5s\t%-12s\t%-30s\t\n", "姓名", "年龄", "性别", "电话", "地址");
int i = 0;
for (i = 0; i < pc->count; i++)
{
printf("%-20s\t%-5d\t%-5s\t%-12s\t%-30s\t\n", pc->data[i].name,
pc->data[i].age,
pc->data[i].sex,
pc->data[i].tele,
pc->data[i].addr);
}
}
static int FindByName(Contact* pc, char name[])
{
assert(pc);
int i = 0;
for (i = 0; i < pc->count; i++)
{
if (strcmp(pc->data[i].name, name) == 0)
{
return i;
}
}
return -1;
}
void DelContact(Contact* pc)
{
char name[MAX_NAME] = { 0 };
assert(pc);
if (pc->count == 0)
{
printf("通讯录为空,无法删除信息\n");
return;
}
printf("请输入你要删除的名字:>");
scanf("%s", name);
int pos = FindByName(pc, name);
if (pos == -1)
{
printf("要删除的人不存在\n");
return;
}
int i = 0;
for (i = pos; i < pc->count - 1; i++)
{
pc->data[i] = pc->data[i + 1];
}
pc->count--;
printf("删除成功\n");
}
void SearchContact(const Contact* pc)
{
assert(pc);
printf("请输入你要查找的姓名:>");
char name2[MAX_NAME] = { 0 };
scanf("%s", name2);
int ret = FindByName(pc, name2);
if (ret == -1)
{
printf("要查找的人不存在\n");
return;
}
printf("%-20s\t%-5s\t%-5s\t%-12s\t%-30s\t\n", "姓名", "年龄", "性别", "电话", "地址");
printf("%-20s\t%-5d\t%-5s\t%-12s\t%-30s\t\n", pc->data[ret].name,
pc->data[ret].age,
pc->data[ret].sex,
pc->data[ret].tele,
pc->data[ret].addr);
}
void ModifyContact(Contact* pc)
{
assert(pc);
printf("请输入你要修改的联系人的姓名:>");
char name[MAX_NAME] = { 0 };
scanf("%s", name);
int ret = FindByName(pc, name);
if (ret == -1)
{
printf("要修改的人不存在\n");
return;
}
printf("请输入修改后的名字:>");
scanf("%s", pc->data[ret].name);
printf("请输入修改后的年龄:>");
scanf("%d", &(pc->data[ret].age));
printf("请输入修改后的性别:>");
scanf("%s", pc->data[ret].sex);
printf("请输入修改后的电话:>");
scanf("%s", pc->data[ret].tele);
printf("请输入修改后的地址:>");
scanf("%s", pc->data[ret].addr);
printf("修改完成\n");
}
int cmp (const void* e1, const void* e2)
{
return strcmp(((PeoInfo*)e1)->name,((PeoInfo*)e2)->name);
}
void SortContact(Contact* pc)
{
assert(pc);
qsort(pc->data, pc->count, sizeof(PeoInfo),
cmp);
printf("排序成功\n");
}
void EmptyContact(Contact* pc)
{
assert(pc);
memset(pc->data, 0, sizeof(pc->data));
pc->count = 0;
printf("已清空所有联系人\n");
}
void DestroyContact(Contact* pc)
{
assert(pc);
free(pc->data);
pc->data = NULL;
}
void SaveContact(const Contact* pc)
{
assert(pc);
FILE* pfWrite = fopen("contact.txt", "wb");
if (pfWrite == NULL)
{
perror("SaveContact");
return;
}
int i = 0;
for (i = 0; i < pc->count; i++)
{
fwrite(pc->data + i,sizeof(PeoInfo),1,pfWrite);
}
fclose(pfWrite);
pfWrite = NULL;
}
10.3 test.c
#define _CRT_SECURE_NO_WARNINGS
#include "contact.h"
void menu()
{
printf("*****************************************\n");
printf("******* 1.add 2.del *******\n");
printf("******* 3.search 4.modify *******\n");
printf("******* 5.show 6.empty *******\n");
printf("******* 7.sort 0.exit *******\n");
printf("*****************************************\n");
}
enum Option
{
EXIT,
ADD,
DEL,
SEARCH,
MODIFY,
SHOW,
EMPTY,
SORT
};
int main()
{
int input = 0;
Contact 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 EMPTY:
EmptyContact(&con);
break;
case SORT:
SortContact(&con);
break;
case EXIT:
SaveContact(&con);
DestroyContact(&con);
printf("退出通讯录\n");
break;
default:
printf("选择错误");
break;
}
} while (input);
return 0;
}
|