1 静态通讯录
1.1 头文件Contact.h
头文件中分别定义联系人的信息:姓名,年龄,性别,电话,地址;以及通讯录的增删改查,qsort排序,文件保存联系人信息,文件加载联系人信息。
#pragma once
#include<stdio.h>
#include<string.h>
#include<assert.h>
#include<stdlib.h>
typedef struct PeoInfo
{
char name[50];
int age;
char sex[10];
char tele[12];
char addr[30];
}PeoInfo;
typedef struct Contact
{
PeoInfo data[100];
int count;
}Contact;
//初始化通讯录
void InitContact(Contact* pc);
//添加联系人
void AddContact(Contact* pc);
//删除联系人
void DelContact(Contact* pc);
//查找联系人
void SearchContact(Contact* pc);
//修改联系人
void ModifyContact(Contact* pc);
//排序联系人
void SortContact(Contact* pc);
1.2 通讯录定义Contact.c
静态通讯录的缺点是只能用数组存放联系人的信息,而数组是提前定义好的仅能存放100人的信息。当容量不足,访问越界的数组但是不提醒越界访问,产生段错误,所以接下来引入动态通讯录,可以通过calloc和realloc随时扩容,更加方便存取。
#define _CRT_SECURE_NO_WARNINGS
#include"Contact.h"
//初始化通讯录
void InitContact(Contact* pc)
{
assert(pc);
pc->count = 0;
memset(pc->data, 0,sizeof(pc->data));
}
//添加联系人
void AddContact(Contact* pc)
{
assert(pc);
if (pc->count == 100)
{
printf("通讯录已满,无法添加\n");
}
printf("请输入联系人姓名:> ");
scanf("%s", pc->data[pc->count].name);
printf("请输入联系人年龄:> ");
scanf("%d", &(pc->data[pc->count].age));
printf("请输入联系人电话:> ");
scanf("%s", pc->data[pc->count].tele);
printf("请输入联系人性别:> ");
scanf("%s", pc->data[pc->count].sex);
printf("请输入联系人地址:> ");
scanf("%s", pc->data[pc->count].addr);
pc->count++;
printf("增容成功!\n");
}
//显示通讯录
void ShowContact(Contact* pc)
{
assert(pc);
if (pc->count == 0)
{
printf("通讯录为空,无法显示");
}
printf("%-20s\t%-3s\t%-12s\t%-10s\t%-30s\n", "姓名", "年龄", "性别", "电话", "地址");
for (int i = 0; i < pc->count; i++)
{
printf("%-20s\t%-3d\t%-12s\t%-10s\t%-30s\n", pc->data[i].name,
pc->data[i].age,
pc->data[i].sex,
pc->data[i].tele,
pc->data[i].addr);
}
}
int FineByName(Contact* pc, char name[])
{
assert(pc);
int pos = 0;
for (int i = 0; i < pc->count; i++)
{
if (0 == strcmp(pc->data[i].name, name))
{
pos = i;
return pos;
}
}
return -1;
}
//删除联系人
void DelContact(Contact* pc)
{
assert(pc);
char name[30] = { 0 };
if (pc->count == 0)
{
printf("通讯录为空\n");
}
printf("请输入要删除的姓名:> ");
scanf("%s", name);
int pos=FineByName(pc,name);
if (pos == -1)
{
printf("未找到此人\n");
}
for (int i = pos; i < pc->count-1; i++)
{
pc->data[i] = pc->data[i + 1];
}
printf("删除成功!\n");
pc->count--;
}
void SearchContact(Contact* pc)
{
assert(pc);
char name[30] = { 0 };
if (pc->count == 0)
{
printf("通讯录为空\n");
}
printf("请输入要查找的姓名:> ");
scanf("%s", name);
int pos = FineByName(pc, name);
if (pos == -1)
{
printf("未找到此人\n");
}
else
{
printf("%-20s\t%-3d\t%-12s\t%-10s\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(Contact* pc)
{
assert(pc);
char name[30] = { 0 };
if (pc->count == 0)
{
printf("通讯录为空\n");
}
printf("请输入要修改的姓名:> ");
scanf("%s", name);
int pos = FineByName(pc, name);
if (pos == -1)
{
printf("未找到此人\n");
}
else
{
printf("请输入修改的姓名:> ");
scanf("%s", pc->data[pos].name);
printf("请输入修改的年龄:> ");
scanf("%d", &(pc->data[pos].age));
printf("请输入修改的电话:> ");
scanf("%s", pc->data[pos].tele);
printf("请输入修改的性别:> ");
scanf("%s", pc->data[pos].sex);
printf("请输入修改的地址:> ");
scanf("%s", pc->data[pos].addr);
}
printf("修改成功!\n");
}
int cmp_peo_by_name(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_peo_by_name);
printf("姓名排序成功!\n");
}
1.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.sort **\n");
printf("********* 0.exit **\n");
printf("******************************\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:
SearchContact(&con);
break;
case 4:
ModifyContact(&con);
break;
case 5:
ShowContact(&con);
break;
case 6:
SortContact(&con);
break;
case 0:
exit(-1);
break;
default:
printf("输入错误,请重新输入!\n");
break;
}
} while (input);
return 0;
}
2 动态通讯录+文件存取联系人信息
2.1 头文件Contact.c
动态通讯录中改善了静态通讯录的缺点,开发了动态增容函数以及文件保存加载联系人信息函数。
需要注意的是,增加数据后,退出程序时不可以使用ctrl+c,或者回车键,必须通过0.exit退出函数,只有这样才可以将数据保存至文件内,下次打开文件会自动写入内存,通过pc->data指针打印在终端。
其中用到的函数介绍:
1) perror("字符串")等价于printf("AddContact::%s\n",strerror(errno));它们的头文件是#include<errno.h>,但是perror可以打印的信息更加全面,errno是一些错误代码返回信息,在errno.h中做了宏定义;
2) #pragma once 防止头文件包含;
#pragma pack 4 结构体4字节对齐;
3) memset(数组,设置值,数组大小); 可以快捷设置数组,初始化为0;
4) 在malloc,calloc,realloc开辟空间后,要用指针接收并用free释放,指针必须检测是否为NULL;
5) fwrite 和 fread 函数分别写读文件内容。
#pragma once
#include<stdio.h>
#include<assert.h>
#include<string.h>
#include<stdlib.h>
#define DEFAULT_SZ 3
#define INC_SZ 2
#define MAX 100
#define MAX_NAME 20
#define MAX_SEX 10
#define MAX_TELE 12
#define MAX_ADDR 30
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(Contact* pc);
//删除联系人
void DelContact(Contact* pc);
//查找联系人
void SeaContact(Contact* pc);
//修改联系人
void ModifyContact(Contact* pc);
//按名字排序
void SortContact(Contact* pc);
//保存通讯录v
void SaveContact(const Contact* pc);
//加载文件信息到通讯录
void LoadContact(Contact* pc);
2.2 实现函数Contact.c
具体实现如下:
#define _CRT_SECURE_NO_WARNINGS
#include"Contact.h"
//增容
void CheckCapacity(Contact* pc)
{
if (pc->count == pc->capacity)
{
PeoInfo* ptr = (PeoInfo*)realloc(pc->data, (pc->capacity + INC_SZ) * sizeof(PeoInfo));
if (ptr == NULL)
{
printf("AddContact::%s\n", strerror(errno));
return;
}
else
{
pc->data = ptr;
pc->capacity += INC_SZ;
printf("增容成功!\n");
}
}
}
//加载文件信息到通讯录
void LoadContact(Contact* pc)
{
FILE* pfRead = fopen("contact.txt", "rb");
if (pfRead == NULL)
{
perror("LoadContact");
return;
}
PeoInfo tmp = {0};
while(fread(&tmp,sizeof(PeoInfo),1,pfRead)==1)
{
CheckCapacity(pc);
pc->data[pc->count] = tmp;
pc->count++;
}
fclose(pfRead);
pfRead = NULL;
}
//销毁通信录
void DestroyContact(Contact* pc)
{
assert(pc);
free(pc->data);
pc->data = 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 = DEFAULT_SZ;
//加载文件的信息到通讯录中
LoadContact(pc);
return 0;
}
//动态版本增加联系人
void AddContact(Contact* pc)
{
assert(pc);
CheckCapacity(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);
printf("增加成功!\n");
pc->count++;
}
//显示联系人
void ShowContact(Contact* pc)
{
assert(pc);
if (pc->count == 0)
{
printf("通讯录为空\n");
}
printf("%-20s\t%-10s\t%-10s\t%-12s\t%-30s\n", "姓名", "年龄", "性别", "电话", "地址");
for (int i = 0; i < pc->count; i++)
{
printf("%-20s\t%-10d\t%-10s\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);
}
}
int FindByName(Contact* pc,char name[])
{
int pos = 0;
for (int i = 0; i < pc->count; i++)
{
if(0==strcmp(pc->data[i].name,name))
return i;
}
return -1;
}
//删除联系人
void DelContact(Contact* pc)
{
char name[20] = { 0 };
assert(pc);
if (pc->count == 0)
{
printf("通讯录为空\n");
return;
}
printf("请输入要删除的名字:> ");
scanf("%s", name);
int pos=FindByName(pc,name);
if (pos < 0)
{
printf("查无此人,无法删除\n");
}
else
{
for (int i = pos; i < pc->count - 1; i++)
{
pc->data[i] = pc->data[i + 1];
}
printf("删除成功!\n");
pc->count--;
}
}
//查找联系人
void SeaContact(Contact* pc)
{
char name[20] = { 0 };
assert(pc);
if (pc->count == 0)
{
printf("通讯录为空\n");
}
printf("请输入要查找的名字:> ");
scanf("%s", name);
int pos = FindByName(pc, name);
printf("%-20s\t%-10d\t%-10s\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(Contact* pc)
{
char name[20] = { 0 };
assert(pc);
if (pc->count == 0)
{
printf("通讯录为空\n");
}
printf("请输入要修改的名字:> ");
scanf("%s", name);
int pos = FindByName(pc, name);
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");
}
int cmp_peo_by_name(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_peo_by_name);
printf("排序成功!\n");
}
//保存通讯录
void SaveContact(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;
}
2.3 主函数test.c
#define _CRT_SECURE_NO_WARNINGS
#include"Contact.h"
void menu()
{
printf("**************************\n");
printf("****** 1.Add 2.Del *****\n");
printf("****** 3.Sea 4.Mod *****\n");
printf("****** 5.Show 6.Sort *****\n");
printf("****** 0.Exit *****\n");
printf("**************************\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:
SeaContact(&con);
break;
case 4:
ModifyContact(&con);
break;
case 5:
ShowContact(&con);
break;
case 6:
SortContact(&con);
break;
case 0:
SaveContact(&con);
DestroyContact(&con);
printf("退出通讯录\n");
break;
default:
printf("选择错误\n");
break;
}
} while (input);
return 0;
}