C语言-文件操作-文件改造通讯录(13.2)

简介: C语言-文件操作-文件改造通讯录(13.2)

文件改造通讯录需要修改的地方:

1.在通讯录退出前写入文件

在contact.c文件中实现:


//写入文件(保存通讯录)
void SaveContact(const Contact* pc)
{
  //写数据
  FILE* pf = fopen("contact.txt", "wb");
  //判断写入是否成功
  if (pf == NULL)
  {
  perror("SaveContact::fopen");
  }
  else
  {
  int i = 0;
  for (i = 0; i < pc->sz; i++)
  {
    //遍历数组,将数组每个元素写入文件
    fwrite(pc->data + i, sizeof(PeoInfo), 1, pf);
  }
  //关闭文件
  fclose(pf);
  pf = NULL;
  printf("保存成功\n");
  }
}


在contact.h文件中引用:


//保存通讯录中的信息到文件中
void SaveContact(const Contact* pc);

2.改造初始化通讯录

在contact.c文件中实现:


//初始化通讯录//动态版//文件版
void InitContact(Contact* pc)
{
  assert(pc);
  pc->sz = 0;//通讯录中存放0个人的信息
  PeoInfo* ptr = (PeoInfo*)calloc(DEFAULT_SZ, sizeof(PeoInfo));//通过calloc开辟空间
  //判断
  if (ptr == NULL)
  {
  perror("InitContact::calloc");
  return;
  }
  pc->data = ptr;//data指针得到开辟的空间的地址
  pc->capacity = DEFAULT_SZ;//初始容量赋值
  //添加:
  //加载文件信息到通讯录
  LoadContact(pc);
}


在通讯录初始化的时候加载文件的信息。


分装一个加载函数实现:


//加载文件
void LoadContact(Contact* pc)
{
  //打开文件
  FILE* pf = fopen("contact.txt", "rb");
  //判断打开是否成功
  if (pf == NULL)
  {
  perror("LoadContact::fopen");
  }
  else
  {
  //创建一个空的指针变量
  PeoInfo tmp = { 0 };
  int i = 0;
  //循环读取文件信息
  while (fread(&tmp, sizeof(PeoInfo), 1, pf))//当文件返回0时跳出循环
  {
    //每次读取前调用增容函数,判断是否需要增容
    CheckCapacity(pc);
    //通讯录数组接收文件信息
    pc->data[i] = tmp;
    //通讯录存放人数++
    pc->sz++;
    i++;
  }
  //关闭文件
  fclose(pf);
  pf = NULL;
  }
}


这样,文件改造的通讯录就完成了。


以下是通讯录终极版本的源码。


3.通讯录源码:

test.c文件:


#define _CRT_SECURE_NO_WARNINGS 1
#include "contact.h"
void menu()
{
  printf("\n");
  printf("—————————— 通讯录 ———————————-\n");
  printf("—————————————————————————-\n");
  printf("————————   1.添加联系人    ————————\n");
  printf("—————————————————————————-\n");
  printf("————————   2.删除联系人    ————————\n");
  printf("—————————————————————————-\n");
  printf("————————   3.查找联系人    ————————\n");
  printf("—————————————————————————-\n");
  printf("————————   4.修改联系人信息  ———————\n");
  printf("—————————————————————————-\n");
  printf("————————   5.整理通讯录    ————————\n");
  printf("—————————————————————————-\n");
  printf("————————   6.查看整个通讯录  ———————\n");
  printf("—————————————————————————-\n");
  printf("————————   0.保存并退出    ————————\n");
  printf("—————————————————————————-\n");
  printf("\n");
}
void test()
{
  int input = 0;
  //创建通讯录con
  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 SORT:
    SortContact(&con);
    break;
  case SHOW:
    ShowContact(&con);
    break;
  case EXIT:
    //DestroyContact(&con);
    //保存文件信息
    SaveContact(&con);
    printf("通讯录已退出\n");
    break;
  default:
    printf("选择错误\n");
    break;
  }
  } while (input);
}
int main()//主函数里不要放太多东西
{
  test();
  return 0;
}


contact.h文件:


#pragma once//防止头文件重复引用
//提前将需要使用的头文件引用,
//具体实现通讯录是,需要头文件可以直接添加
#include 
#include 
#include 
#include 
//通过#define定义的常量,方便管理和使用
#define MAX 100
#define NAME_MAX 20
#define SEX_MAX 5
#define ADDR_MAX 30
#define TELE_MAX 12
//通讯录初始的大小和每次增容的大小
#define DEFAULT_SZ 3
#define INC_SZ 2
//通讯录中存放一个人的信息
typedef struct PeoInfo//typedef简化结构体名称
{
  char name[NAME_MAX];
  int age;
  char sex[SEX_MAX];
  char addr[ADDR_MAX];
  char tele[TELE_MAX];
}PeoInfo;
//动态增长版本的通讯录
typedef struct Contact
{
  PeoInfo* data;//指向存放人信息的空间
  int sz;//用来存放数组元素个数
  int capacity;//当前通讯录的最大容量
}Contact;
创建一个结构体将数组和数组中存放的元素数封装//静态版本
//typedef struct Contact
//{
//  PeoInfo data[MAX];//data这个数组元素的类型的是结构体PeoInfo
//                   //用来存放联系人信息
//  int sz;//用来存放数组元素个数
//}Contact;
enum Option
{
  EXIT,//0
  ADD,
  DEL,
  SEARCH,
  MODIFY,
  SORT,
  SHOW
};
//初始化通讯录
void InitContact(Contact* pc);
销毁通讯录
//void DestroyContact(Contact* pc);
//增加联系人
void AddContact(Contact* pc);
//删除联系人
void DelContact(Contact* pc);
//查找联系人
void SearchContact(const Contact* pc);
//修改指定联系人
void ModifyContact(Contact* pc);
//整理通讯录
void SortContact(Contact* pc);
//显示通讯录的信息
void ShowContact(const Contact* pc);
//保存通讯录中的信息到文件中
void SaveContact(const Contact* pc);
//加载文件信息到通讯录
void LoadContact(Contact* pc);


contact.c文件:

#define _CRT_SECURE_NO_WARNINGS 1
#include "contact.h"
初始化通讯录//静态版
//void InitContact(Contact* pc)
//{
//  assert(pc);
//  pc->sz = 0;//代表数组中有0个元素
//  memset(pc->data, 0, sizeof(pc->data));//data是整个数组的大小,初始化成0
//}
//初始化通讯录//动态版//文件版
void InitContact(Contact* pc)
{
  assert(pc);
  pc->sz = 0;//通讯录中存放0个人的信息
  PeoInfo* ptr = (PeoInfo*)calloc(DEFAULT_SZ, sizeof(PeoInfo));//通过calloc开辟空间
  //判断
  if (ptr == NULL)
  {
  perror("InitContact::calloc");
  return;
  }
  pc->data = ptr;//data指针得到开辟的空间的地址
  pc->capacity = DEFAULT_SZ;//初始容量赋值
  //添加:
  //加载文件信息到通讯录
  LoadContact(pc);
}
销毁创建的内存
//void DestroyContact(Contact* pc)
//{
//  free(pc->data);
//  pc->data = NULL;
//}
//增容函数
void CheckCapacity(Contact* pc)
{
  if (pc->sz == pc->capacity)//如果容量满了,就进来
  {
  //通过realloc函数进行增容,原容量+INC_SZ(可以根据自己喜好设置)
  PeoInfo* ptr = (PeoInfo*)realloc(pc->data, (pc->capacity + INC_SZ) * sizeof(PeoInfo));
  //判断
  if (ptr == NULL)
  {
    perror("CheckCapacity::realloc");
    return;
  }
  pc->data = ptr;//data指针接收增容后的内存的地址
  pc->capacity += INC_SZ;//容量也按设定增加
  printf("增容成功\n");//提示增容成效
  }
}
增加联系人//静态版
//void AddContact(Contact* pc)
//{
//  assert(pc);
//  if (pc->sz == MAX)//如果通讯录满了
//  {
//  printf("通讯录已满,无法添加\n");
//  return;//就会直接返回
//  }
//
//  //增加一个人的信息
//  printf("请输入名字:>");
//  scanf("%s", pc->data[pc->sz].name);
//  //通过pc指针访问data数组的结构体类型的元素,进而访问结构体成员
//
//  printf("请输入年龄:>");
//  scanf("%d", &(pc->data[pc->sz].age));//age不是数组,需要取地址
//
//  printf("请输入性别:>");
//  scanf("%s", pc->data[pc->sz].sex);
//
//  printf("请输入地址:>");
//  scanf("%s", pc->data[pc->sz].addr);
//
//  printf("请输入电话:>");
//  scanf("%s", pc->data[pc->sz].tele);
//
//  pc->sz++;//代表数组中的元素个数+1
//}
//增加联系人//动态版
void AddContact(Contact* pc)
{
  assert(pc);
  //增容
  CheckCapacity(pc);
  //增加一个人的信息
  printf("请输入名字:>");
  scanf("%s", pc->data[pc->sz].name);
  //通过pc指针访问data数组的结构体类型的元素,进而访问结构体成员
  printf("请输入年龄:>");
  scanf("%d", &(pc->data[pc->sz].age));//age不是数组,需要取地址
  printf("请输入性别:>");
  scanf("%s", pc->data[pc->sz].sex);
  printf("请输入地址:>");
  scanf("%s", pc->data[pc->sz].addr);
  printf("请输入电话:>");
  scanf("%s", pc->data[pc->sz].tele);
  pc->sz++;//代表数组中的元素个数+1
}
//显示通讯录的信息
void ShowContact(const Contact* pc)
{
  assert(pc);
  printf("%-10s\t%-4s\t%-5s\t%-20s\t%-12s\n", "名字", "年龄", "性别", "地址", "电话");
  int i = 0;
  for (i = 0; i < pc->sz; i++)//遍历通讯录并打印
  {
  printf("%-10s\t%-4d\t%-5s\t%-20s\t%-12s\n",
    pc->data[i].name,
      pc->data[i].age,
    pc->data[i].sex,
    pc->data[i].addr,
    pc->data[i].tele);
  }
}
//查找函数
int FindByName(const Contact* pc, char name[])
{
  int i = 0;
  int del = 0;
  for (i = 0; i < pc->sz; i++)//遍历通讯录
  {
  if (strcmp(pc->data[i].name, name) == 0)
  {   //通过strcmp函数判断要查找的联系人是否存在
    del = i;
    return del;//返回数组下标(要查找的元素的位置)
  }
  }
  return -1;//找不到
}
//删除联系人
void DelContact(Contact* pc)
{
  assert(pc);
  char name[NAME_MAX] = { 0 };//初始化name数组(字符串)
  if (pc->sz == 0)//判断通讯录中是否存在联系人
  {
  printf("通讯录为空,无法删除\n");
  return;
  }
  //找到要删除的人
  printf("请输入要删除的人的名字:>");
  scanf("%s", name);//输入字符串
  int ret = FindByName(pc, name);//分装字符串查找函数
  if (-1 == ret)
  {
  printf("要删除的人不存在\n");
  return;
  }
  //删除
  int i = 0;
  for (i = ret; i < pc->sz - 1; i++)
  {
  pc->data[i] = pc->data[i + 1];//将存放在被删除的联系人后面的联系人信息,
    }                                 //通过循环一个个往前覆盖
  pc->sz--;//数组元素-1
  printf("删除成功\n");
}
//查找联系人
void SearchContact(const Contact* pc)
{
  assert(pc);
  char name[NAME_MAX] = { 0 };//初始化name数组(字符串)
  printf("请输入要查找人的名字:>");
  scanf("%s", name);
  int pos = FindByName(pc, name);//函数复用
  if (-1 == pos)
  {
  printf("要查找的人不存在\n");
  return;
  }
  //打印信息//我实现的是左对齐,并用水平制表符使打印出来的观感更好
  printf("%-10s\t%-4s\t%-5s\t%-20s\t%-12s\n", "名字", "年龄", "性别", "地址", "电话");
  printf("%-10s\t%-4d\t%-5s\t%-20s\t%-12s\n",
    pc->data[pos].name,
    pc->data[pos].age,
    pc->data[pos].sex,
    pc->data[pos].addr,
    pc->data[pos].tele);
}
//修改指定联系人
void ModifyContact(Contact* pc)
{
  assert(pc);
  char name[NAME_MAX] = { 0 };//初始化name数组(字符串)
  printf("请输入要修改人的名字:>");
  scanf("%s", name);
  int pos = FindByName(pc, name);//函数复用
  if (-1 == pos)
  {
  printf("要修改的人不存在\n");
  return;
  }
  //重新录入
  printf("请输入修改后的名字:>");
  scanf("%s", pc->data[pos].name);
  printf("请输入修改后的年龄:>");
  scanf("%d", &(pc->data[pos].age));//age不是数组,需要取地址
  printf("请输入修改后的性别:>");
  scanf("%s", pc->data[pos].sex);
  printf("请输入修改后的地址:>");
  scanf("%s", pc->data[pos].addr);
  printf("请输入修改后的电话:>");
  scanf("%s", pc->data[pos].tele);
  printf("修改完成\n");
}
//整形排序的实现
int CmpContactByAge(const void* e1, const void* e2)
{
  //这个排出来的是升序,如果想排降序,只需将e1和e2的位置调换即可
  return ((Contact*)e1)->data->age - ((Contact*)e2)->data->age;
}
//整理通讯录
void SortContact(Contact* pc)
{
  assert(pc);
  int sz = pc->sz;
  //通过qsort函数辅助排序
  qsort(pc->data, sz, sizeof(pc->data[0]), CmpContactByAge);
  printf("排序成功\n");
}
//写入文件(保存通讯录)
void SaveContact(const Contact* pc)
{
  //写数据
  FILE* pf = fopen("contact.txt", "wb");
  //判断写入是否成功
  if (pf == NULL)
  {
  perror("SaveContact::fopen");
  }
  else
  {
  int i = 0;
  for (i = 0; i < pc->sz; i++)
  {
    //遍历数组,将数组每个元素写入文件
    fwrite(pc->data + i, sizeof(PeoInfo), 1, pf);
  }
  //关闭文件
  fclose(pf);
  pf = NULL;
  printf("保存成功\n");
  }
}
//加载文件
void LoadContact(Contact* pc)
{
  //打开文件
  FILE* pf = fopen("contact.txt", "rb");
  //判断打开是否成功
  if (pf == NULL)
  {
  perror("LoadContact::fopen");
  }
  else
  {
  //创建一个空的指针变量
  PeoInfo tmp = { 0 };
  int i = 0;
  //循环读取文件信息
  while (fread(&tmp, sizeof(PeoInfo), 1, pf))//当文件返回0时跳出循环
  {
    //每次读取前调用增容函数,判断是否需要增容
    CheckCapacity(pc);
    //通讯录数组接收文件信息
    pc->data[i] = tmp;
    //通讯录存放人数++
    pc->sz++;
    i++;
  }
  //关闭文件
  fclose(pf);
  pf = NULL;
  }
}


写在最后:

以上就是本篇文章的内容了,感谢你的阅读。


如果喜欢本文的话,欢迎点赞和评论,写下你的见解。


如果想和我一起学习编程,不妨点个关注,我们一起学习,一同成长。


相关文章
|
1月前
|
存储 程序员 C语言
【C语言】文件操作函数详解
C语言提供了一组标准库函数来处理文件操作,这些函数定义在 `<stdio.h>` 头文件中。文件操作包括文件的打开、读写、关闭以及文件属性的查询等。以下是常用文件操作函数的详细讲解,包括函数原型、参数说明、返回值说明、示例代码和表格汇总。
51 9
|
1月前
|
存储 数据管理 C语言
C 语言中的文件操作:数据持久化的关键桥梁
C语言中的文件操作是实现数据持久化的重要手段,通过 fopen、fclose、fread、fwrite 等函数,可以实现对文件的创建、读写和关闭,构建程序与外部数据存储之间的桥梁。
|
2月前
|
算法 C语言
C语言中的文件操作技巧,涵盖文件的打开与关闭、读取与写入、文件指针移动及注意事项
本文深入讲解了C语言中的文件操作技巧,涵盖文件的打开与关闭、读取与写入、文件指针移动及注意事项,通过实例演示了文件操作的基本流程,帮助读者掌握这一重要技能,提升程序开发能力。
129 3
|
2月前
|
存储 C语言
【c语言】玩转文件操作
本文介绍了C语言中文件操作的基础知识,包括文件的打开和关闭、文件的顺序读写、文件的随机读写以及文件读取结束的判定。详细讲解了`fopen`、`fclose`、`fseek`、`ftell`、`rewind`等函数的使用方法,并通过示例代码展示了如何进行文件的读写操作。最后,还介绍了如何判断文件读取结束的原因,帮助读者更好地理解和应用文件操作技术。
52 2
|
3月前
|
存储 编译器 C语言
如何在 C 语言中判断文件缓冲区是否需要刷新?
在C语言中,可以通过检查文件流的内部状态或使用`fflush`函数尝试刷新缓冲区来判断文件缓冲区是否需要刷新。通常,当缓冲区满、遇到换行符或显式调用`fflush`时,缓冲区会自动刷新。
|
3月前
|
存储 编译器 C语言
C语言:文件缓冲区刷新方式有几种
C语言中文件缓冲区的刷新方式主要包括三种:自动刷新(如遇到换行符或缓冲区满)、显式调用 fflush() 函数强制刷新、以及关闭文件时自动刷新。这些方法确保数据及时写入文件。
|
3月前
|
存储 C语言
简述C语言文件操作
简述C语言文件操作
13 0
|
存储 C语言
C语言 文件操作 深度解析 #重点知识:文件操作函数的使用#(下)
C语言 文件操作 深度解析 #重点知识:文件操作函数的使用#(下)
|
Linux 编译器 C语言
C语言 文件操作 深度解析 #重点知识:文件操作函数的使用#(上)
C语言 文件操作 深度解析 #重点知识:文件操作函数的使用#(上)