之前写过静态开辟内存版的,也写过动态开辟内存版的,但是都有很大的问题。
每次再重新运行代码时,先前输入的信息都会刷新掉。
那么我们该如何保存这些信息呢?
刚刚学完了文件处理,我们可以将这些信息保存在硬盘中,下次运行代码时再将信息输入即可。
我们在动态版本的基础上再添加几个函数。
1. 保存联系人信息
该函数用于每次程序结束时,将联系人信息保存只硬盘中。
void saveContact(struct Contact* pc) { //打开文件 FILE* pfw = fopen("date.txt", "wb"); if (pfw == NULL) { perror("saveContact::fopen"); return; } //操作文件 int i = 0; for (i = 0; i < pc->data; i++) { fwrite(pc->data + i, sizeof(struct PeoInfo), 1, pfw); } //关闭文件 fclose(pfw); pfw = NULL; }
2. 加载联系人信息
该函数用于初始化时,将硬盘中的联系人信息加载至“ 容器 ” 中。
LoadContact(struct Contact* pc) { //打开文件 FILE* pfr = fopen("date.txt", "rb"); if (pfr == NULL) { perror("saveContact::fopen"); return; } //操作文件 struct PeoInfo tmp = { 0 }; while (fread(&tmp, sizeof(struct PeoInfo), 1, pfr)) { //检查容量是否足够 check_capacity(&pc); pc->data[pc->sz] = tmp; pc->sz++; } //关闭文件 fclose(pfr); pfr = NULL; }
在该函数的结尾调用LoadContact。
下面是源代码:
contact.c #define _CRT_SECURE_NO_WARNINGS #include "contact.h" //声明一下 static int check_capacity(struct Contact* pc); LoadContact(struct Contact* pc) { //打开文件 FILE* pfr = fopen("date.txt", "rb"); if (pfr == NULL) { perror("saveContact::fopen"); return; } //操作文件 struct PeoInfo tmp = { 0 }; while (fread(&tmp, sizeof(struct PeoInfo), 1, pfr)) { //检查容量是否足够 check_capacity(&pc); pc->data[pc->sz] = tmp; pc->sz++; } //关闭文件 fclose(pfr); pfr = NULL; } void InitContact(struct Contact* pc) { assert(pc); pc->data = (struct PeoInfo*)malloc(DEFAULT_SZ * sizeof(struct PeoInfo)); if (pc->data == NULL) { perror("InitContact"); return; } pc->sz = 0; pc->capacity = DEFAULT_SZ; //加载文件中的数据到通讯录; LoadContact(pc); } void DestoryContact(struct Contact* pc) { free(pc->data); pc->data = NULL; pc->sz = 0; pc->capacity = 0; } static int check_capacity(struct Contact* pc) { //增容 if (pc->sz == pc->capacity) { struct PeoInfo* str = realloc(pc->data, (pc->capacity + 2) * sizeof(struct PeoInfo)); if (str != NULL) { pc->data = str; pc->capacity += INT_SZ; return 1; } else { perror("AddContact"); return 0; } } else return 1; } void AddContact(struct Contact* pc) { assert(pc); /*if (pc->sz == MAX) { printf("通讯录已满"); return 0; }*/ if (0 == check_capacity(pc)) { return; } //增加人的信息 printf("请输入名字:>"); scanf("%s", pc->data[pc->sz].name); printf("请输入性别:>"); scanf("%s", pc->data[pc->sz].sex); printf("请输入年龄:>"); scanf("%d", &(pc->data[pc->sz].age)); printf("请输入电话:>"); scanf("%s", pc->data[pc->sz].tele); printf("请输入地址:>"); scanf("%s", pc->data[pc->sz].addr); pc->sz++; printf("成功增加联系人\n"); } void ShowContact(struct Contact* pc) { int i = 0; printf("%-20s\t%-5s\t%-5s\t%-12s\t%-30s\n", "姓名", "性别", "年龄", "电话", "地址"); for (i = 0; i < pc->sz; i++) { printf("%-20s\t%-5s\t%-5d\t%-12s\t%-30s\n", pc->data[i].name, pc->data[i].sex, pc->data[i].age, pc->data[i].tele, pc->data[i].addr); } } static int FindByName(const struct Contact* pc, char name[]) { int i = 0; for (i = 0; i < pc->sz; i++) { if (0 == strcmp(pc->data[i].name, name)) { return i; } } return -1; } void DelContact(struct Contact* pc) { char name[MAX_NAME]; printf("请输入要删除人的名字:>"); scanf("%s", name); //查找一下指定的人是否存在 int ret = FindByName(pc, name); if (ret == -1) printf("要删除的人不存在\n"); else { //删除 int j = 0; for (j = ret; j < pc->sz - 1; j++) { pc->data[j] = pc->data[j + 1]; } pc->sz--; printf("成功删除指定联系人\n"); } } void SerachContact(struct Contact* pc) { char name[MAX_NAME]; printf("输入你要查找的联系人姓名"); scanf("%s", name); int ret = FindByName(pc, name); if (ret == -1) printf("要查找的人不存在\n"); else { printf("%-20s\t%-5s\t%-5s\t%-12s\t%-30s\n", "姓名", "性别", "年龄", "电话", "地址"); printf("%-20s\t%-5s\t%-5d\t%-12s\t%-30s\n", pc->data[ret].name, pc->data[ret].sex, pc->data[ret].age, pc->data[ret].tele, pc->data[ret].addr); } } void ModifyContact(struct Contact* pc) { printf("请输入要修改人的名字:>"); char name[MAX_NAME]; scanf("%s", name); int ret = FindByName(pc, name); if (ret == -1) printf("要修改的人不存在\n"); else { printf("请输入名字:>"); scanf("%s", pc->data[ret].name); printf("请输入性别:>"); scanf("%s", pc->data[ret].sex); printf("请输入年龄:>"); scanf("%d", &(pc->data[ret].age)); printf("请输入电话:>"); scanf("%s", pc->data[ret].tele); printf("请输入地址:>"); scanf("%s", pc->data[ret].addr); printf("修改成功\n"); } } //int CmpByAge(const void* e1, const void* e2) //{ // return ((struct PeoInfo*)e1)->age - ((struct PeoInfo*)e2)->age; //} // 按照年龄来排序 //void SortContact(struct Contact* pc) //{ // qsort(pc->data, pc->sz, sizeof(struct PeoInfo), CmpByAge); //} int CmpByName(const void* e1, const void* e2) { return strcmp(((struct PeoInfo*)e1)->name, ((struct PeoInfo*)e2)->name); } //按照年龄来排序 void SortContact(struct Contact* pc) { qsort(pc->data, pc->sz, sizeof(struct PeoInfo), CmpByName); } void saveContact(struct Contact* pc) { //打开文件 FILE* pfw = fopen("date.txt", "wb"); if (pfw == NULL) { perror("saveContact::fopen"); return; } //操作文件 int i = 0; for (i = 0; i < pc->data; i++) { fwrite(pc->data + i, sizeof(struct PeoInfo), 1, pfw); } //关闭文件 fclose(pfw); pfw = NULL; }
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; struct Contact con;//通讯录 InitContact(&con);//初始化通讯录 do { menu(); printf("请输入你的选择: "); scanf("%d", &input); switch (input) { case 1: AddContact(&con); break; case 2: DelContact(&con); break; case 3: SerachContact(&con); break; case 4: ModifyContact(&con); break; case 5: ShowContact(&con); break; case 6: break; case 0: saveContact(&con); DestoryContact(&con); printf("退出成功"); break; default: printf("输入错误,请重新输入\n"); break; } } while (input); return 0; }
contact.h #pragma once #pragma once #include <stdio.h> #include <assert.h> #include <string.h> #include <stdlib.h> #define MAX 100 #define MAX_NAME 20 #define MAX_SEX 20 #define MAX_TELE 20 #define MAX_ADDR 20 #define DEFAULT_SZ 3 #define INT_SZ 2 //个人信息 struct PeoInfo { char name[MAX_NAME]; char sex[MAX_SEX]; char tele[MAX_TELE]; int age; char addr[MAX_ADDR]; }; //通讯录信息 struct Contact { struct PeoInfo* data;//指向存放数据的空间 int sz;//目前所存的大小 int capacity;//容量 }; //增加联系人 void AddContact(struct Contact* pc); //显示联系人 void ShowContact(struct Contact* pc); //删除联系人 void DelContact(struct Contact* pc); //查找联系人 void SerachContact(struct Contact* pc); //查找指定联系人 void SearchContact(const struct Contact* pc); //修改指定联系人的信息 void ModifyContact(struct Contact* pc); //排序通讯录中的信息 - 年龄 void SortContact(struct Contact* pc); //销毁通讯录 void DestoryContact(struct Contact* pc); //保存联系人信息 void saveContact(struct Contact* pc); //加载文件中的数据到通讯录; LoadContact(struct Contact* pc);
3. 确认是否在硬盘中保存信息
这是我们的date.txt文件。
这是我们文件中的内容,因为写时是用 rw 的形式写的也就是二进制,读取也是 rb 的形式读的,所以这个只有计算机能读懂。