通讯录管理文件升级版

前言:

没错它它它又来了,通讯录管理文件升级版,这已经是我们的第三个版本了,大家想一想,前两个版本虽然也没啥问题,但是不能保存我们录入的信息,导致每次录入都要重新输入,今天的文件版本则可以完美实现这个功能。我们这次升级的部分不多,只有几个部分来让我们看一看吧

注:

本次升级涉及文件操作的相关知识,如果又不懂文件操作的朋友,可以看我下一篇博客,详解文件操作

1.fwrite写文件函数,fread读文件函数

在讲解通讯录之前,我们需要了解两个函数fwrite写文件函数,fread读文件函数,大家先参考一下

1.1举例说明fwrite函数

首先的我们创建一个结构体s,然后给结构体赋值,利用fopen打开文件test.txt,这里说明一下,由于我们采取的是“wb”写操作,所以test.txt文件不需要我们在文件夹中新建,只要这里给出名字,系统会自动创建,紧接着判断文件是否能打开成功,避免打开失败,产生错误,然后就用到了我们的fwrite函数了,我们一次写两个信息将结构体中s中的信息写入pf所指向的文件test.txt,最后关闭文件即可
 

//以二进制的形式写文件
struct S
{
  char name[20];
  int age;
  double score;
};
int main()
{
  struct S s[2] = { {"张三",20,55.6},{"李四",18,99.9}};
  FILE* pf = fopen("test.txt","wb");
  if (pf == NULL)
  {
    return 0;
  }
  //二进制的形式写文件
  fwrite(&s,sizeof(struct S),2,pf);
  fclose(pf);
  pf = NULL;
  return 0;
}

1.2举例说明fread函数

读文件操作,创建一个与写文件结构体一致的结构体tmp,用来存放读出来的信息,“rb”不同于“wb”注意区别,“rb”是读出test.txt中的数据,fread函数将pf所指向的test.txt中的数据读到tmp中,然后打印出来

struct S
{
  char name[20];
  int age;
  double score;
};
int main()
{
  //struct S s = { "张三",20,55.6 };
  struct S tmp[2] = { 0 };
  FILE* pf = fopen("test.txt","rb");
  if (pf == NULL)
  {
    return 0;
  }
  //二进制的形式读文件
  fread(&tmp,sizeof(struct S),2,pf);
  printf("%s %d %lf\n",tmp[0].name,tmp[0].age,tmp[0].score);
  printf("%s %d %lf\n", tmp[1].name, tmp[1].age, tmp[1].score);
  fclose(pf);
  pf = NULL;
  return 0;
}

2.将信息写进文件,增加保存函数 save

首先使用fopen将pfwrite指向contact.txt文件并打开,(contact.txt这个文件名随便写),判断是否打开成功,若打开失败printf("%s\n",strerror(errno));提示打开失败信息,注strerror打印错误码函数,(errno)是错误码变量这里不是重点(strerror需要引用头文件#include<errno.h>),我会放到文件操作博客中细讲,接着用fwrite函数将通讯录中的数据写到pfwrite所指向的contact.txt文件中,一次写一个
 

void savecontact(struct contact* ps)//保存到文件函数
{
  FILE* pfwrite = fopen("contact.txt", "wb");
  if (pfwrite == NULL)
  {
    printf("%s\n",strerror(errno));
    return;
  }
  //写通讯录中的数据到文件中
  int i = 0;
  for (i=0;i<ps->size;i++)
  {
    fwrite(&(ps->data[i]), sizeof(struct peoinfo), 1, pfwrite);
  }
  printf("保存成功\n");
  fclose(pfwrite);
  pfwrite = NULL;
}

3.从文件中读到通讯录中

我们现在只是将数据写进了文件,但是还没将数据展现出来,这就要用到了fread函数了,我们来分析一下代码,因为每次打开通讯录都会初始化,所以我们就将通讯录初始化时将数据读到通讯录,写一个loadcontact函数(作用:把文件中已经存放的文件信息加载到通讯录中),同样将pfread指向的文件contact.txt打开,注意时“rb”哦,我们新建一个struct peoinfo结构体tmp,用于存放读出的数据,然后判断这个loadcontact::%s\n,没有啥意义,只是做一个错误位置的具体提醒,根据习惯可加可不加,然后开始我们真正的读文件操作,将pfread指向的contact.txt文件中的数据读到tmp中,一次读一个,fread函数是有返回值的,具体看图

本题中我们一次读一个,最终将数据读完时就返回0,我们使用while循环读数据,数据是读到了,但是在tmp中并没有在通讯录中,还有一个问题,就是通讯录中是否有容量放下这些数据,我们就要再次检查容量,不够则增容,只需将增容函数cheakcontact放在加载信息loadcontact前函数声明一下就行了,随后把信息放入下标为size的data元素上,注意size++哦

void initcontact(struct contact* ps)//通讯录初始化函数
{
  ps->data = (struct peoinfo*)malloc(default_sz *sizeof(struct peoinfo));
  if (ps->data == NULL)
  {
    return;
  }
  ps->size = 0;
  ps->capacity = default_sz;
  //把文件中已经存放的文件信息加载到通讯录中
  loadcontact(ps);
}

void cheakcapacity(struct contact* ps);//声明增容函数

void loadcontact(struct contact* ps)//将文件信息加载到通讯录中函数
{
  struct peoinfo tmp = { 0 };
  FILE* pfread = fopen("contact.txt", "rb");
  if (pfread == NULL)
  {
    printf("loadcontact::%s\n",strerror(errno));
    return;
  }
  //读文件存放到通讯录中
  while (fread(&(tmp), sizeof(struct peoinfo), 1, pfread))//fread的返回值是其真实读到的数据的个数
  {
    cheakcapacity(ps);//检查容量够不够,不够增容
    ps->data[ps->size] = tmp;//把信息放入下标为size的data元素上
    ps->size++;
  }

  fclose(pfread);
  pfread = NULL;
}

效果展示

4.提示

其实我们也可将退出函数out中添加savecontact(&con)保存函数,这样更为方便

case out:
      //销毁通讯录:释放动态开辟的空间free
      savecontact(&con);
      destroycontact(&con);
      printf("退出通讯录\n");
      break;

5.完整代码

course-22.c

#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<string.h>
#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("*****    7.save     0.out    *****\n");
  printf("**********************************\n");
}
int main()
{
  int input = 0;
  //创建通讯录
  struct contact con;//通讯录,包含:data指针size,capacity
  //初始化通讯录
  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:
      sortcontact(&con);//排序通讯录
      break;
    case out:
      //销毁通讯录:释放动态开辟的空间free
      savecontact(&con);
      destroycontact(&con);
      printf("退出通讯录\n");
      break;
    case save:
      savecontact(&con);
      break;
    default:
      printf("选择错误,请重新选择\n");
      break;
    }
  } while (input);
  return 0;
}

contact.h

#define _CRT_SECURE_NO_WARNINGS 1

//#define MAX 1000
#define default_sz 3
#define MAX_NAME 20
#define MAX_SEX 5
#define MAX_TELE 12
#define MAX_ADDR 30
#include<stdlib.h>
#include<errno.h>
enum option//枚举,用于替换swith case语句中的0 1 2 3···,增强代码的可读性
{
  out,
  add,
  del,
  search,
  modify,
  show,
  sort,
  save
};
struct peoinfo
{
  char name[MAX_NAME];
  int age;
  char sex[MAX_SEX];
  char tele[MAX_TELE];
  char addr[MAX_ADDR];
};

//通讯录类型
struct contact
{
  struct peoinfo *data;//存放信息
  int size;//记录当前已经有的元素个数
  int capacity;//当前通讯录的最大容量
};

//函数声明
void initcontact(struct contact* ps);//初始化
void addcontact(struct contact* ps);//增加一个信息
void showcontact(const struct contact* ps);//打印通讯录中的信息
void delcontact(struct contact* ps);//删除信息
void searchcontact(const struct contact* ps);//查找信息
void modifycontact(struct contact* ps);//修改信息
void sortcontact(struct contact* ps);//排序信息
void destroycontact(struct contact*ps);//销毁通讯录
void savecontact(struct contact* ps);//保存到文件
void loadcontact(struct contact* ps);//加载文件中的信息到通讯录

contact.c

#define _CRT_SECURE_NO_WARNINGS 1
#include "contact.h"
#include<stdio.h>
#include<string.h>

void initcontact(struct contact* ps)//通讯录初始化函数
{
    ps->data = (struct peoinfo*)malloc(default_sz *sizeof(struct peoinfo));
    if (ps->data == NULL)
    {
        return;
    }
    ps->size = 0;
    ps->capacity = default_sz;
    //把文件中已经存放的文件信息加载到通讯录中
    loadcontact(ps);
}

void cheakcapacity(struct contact* ps);//声明增容函数

void loadcontact(struct contact* ps)//将文件信息加载到通讯录中函数
{
    struct peoinfo tmp = { 0 };
    FILE* pfread = fopen("contact.txt", "rb");
    if (pfread == NULL)
    {
        printf("loadcontact::%s\n",strerror(errno));
        return;
    }
    //读文件存放到通讯录中
    while (fread(&(tmp), sizeof(struct peoinfo), 1, pfread))//fread的返回值是其真实读到的数据的个数
    {
        cheakcapacity(ps);//检查容量够不够,不够增容
        ps->data[ps->size] = tmp;//把信息放入下标为size的data元素上
        ps->size++;
    }

    fclose(pfread);
    pfread = NULL;
}

void cheakcapacity(struct contact* ps)//增容函数
{
    if (ps->size == ps->capacity)
    {
        //给ps->data增容,每次增2
        struct peoinfo* ptr = realloc(ps->data,(ps->capacity + 2) * sizeof(struct peoinfo));
        if (ptr != NULL)//增容成功
        {
            ps->data = ptr;
            ps->capacity += 2;//容量+2
            printf("增容成功\n");
        }
        else
        {
            printf("增容失败\n");
        }
    }
}
void addcontact(struct contact* ps)//增加函数
{
    //检测通讯录容量
    // 1.如果满了增容
    // 2.没满啥也不干
    cheakcapacity(ps);
    //添加数据
    printf("请输入名字:");
    scanf("%s",ps->data[ps->size].name);//将信息放入下标为size的空间
    printf("请输入年龄:");
    scanf("%d", &(ps->data[ps->size].age));
    printf("请输入性别:");
    scanf("%s", ps->data[ps->size].sex);
    printf("请输入电话:");
    scanf("%s", ps->data[ps->size].tele);
    printf("请输入地址:");
    scanf("%s", ps->data[ps->size].addr);
    ps->size++;
    printf("添加成功\n");
    //if (ps->size==MAX)
    //{
    //  printf("通讯录以满,无法增加\n");
    //}
    //else
    //{
    //  printf("请输入名字:");
    //  scanf("%s",ps->data[ps->size].name);//将信息放入下标为size的空间
    //  printf("请输入年龄:");
    //  scanf("%d", &(ps->data[ps->size].age));
    //  printf("请输入性别:");
    //  scanf("%s", ps->data[ps->size].sex);
    //  printf("请输入电话:");
    //  scanf("%s", ps->data[ps->size].tele);
    //  printf("请输入地址:");
    //  scanf("%s", ps->data[ps->size].addr);

    //  ps->size++;
    //  printf("添加成功\n");
    //}
}

void showcontact(const struct contact* ps)//打印函数
{
    if (ps->size==0)
    {
        printf("通讯录为空\n");
    }
    else
    {
        int i = 0;
        //打印标题
        printf("%-20s\t%-4s\t%-5s\t%-12s\t%-20s\n","名字","年龄","性别","电话","地址");
        //打印数据
        for (i=0;i<ps->size;i++)
        {
            printf("%-20s\t%-4d\t%-5s\t%-12s\t%-20s\n", 
                ps->data[i].name,
                ps->data[i].age,
                ps->data[i].sex,
                ps->data[i].tele,
                ps->data[i].addr);
        }
    }
}

static int findbyname(const struct contact*ps,char name[MAX_NAME])//static修饰函数,使此函数今在本文件(contact.c)中起作用
{
    int i = 0;
    for (i = 0; i < ps->size; i++)
    {
        if (strcmp(ps->data[i].name, name) == 0)
        {
            return i;//找到返回下标
        }
    }
    return -1;//没找到
}
void delcontact(struct contact* ps)//删除函数
{
    char name[MAX_NAME];
    printf("请输入要删除的人的姓名:");
    scanf("%s",name);

    //查找要删除的人在什么位置
    int pos=findbyname(ps,name);//这里把查找名字写成一个函数,找到返回1,找不到返回-1

    //删除
    if (pos==-1)//没找到的情况
    {
        printf("要删除的人不存在\n");
    }
    else
    {
        int j = 0;
        for (j=pos;j<ps->size-1;j++)
        {
            ps->data[j] = ps->data[j+1];
        }
        ps->size--;
        printf("删除成功\n");
    }
}

void searchcontact(const struct contact* ps)//查找函数
{
    char name[MAX_NAME];
    printf("请输入要查找人的姓名:");
    scanf("%s",name);
    int pos = findbyname(ps,name);
    if (pos == -1)     //没找到的情况
    {
        printf("要查找的人不存在\n");
    }
    else//找到了
    {
        
        printf("%-20s\t%-4s\t%-5s\t%-12s\t%-20s\n", "名字", "年龄", "性别", "电话", "地址");//打印标题
        printf("%-20s\t%-4d\t%-5s\t%-12s\t%-20s\n",
                ps->data[pos].name,
                ps->data[pos].age,
                ps->data[pos].sex,
                ps->data[pos].tele,
                ps->data[pos].addr);
    }
}

void modifycontact(struct contact* ps)//修改函数
{
    char name[MAX_NAME];
    printf("请输入要修改人的名字:");
    scanf("%s",name);
    int pos = findbyname(ps,name);
    if (pos == -1)
    {
        printf("要修改人的信息不存在\n");
    }
    else
    {
        printf("请输入名字:");
        scanf("%s", ps->data[pos].name);//将信息放入下标为pos的空间
        printf("请输入年龄:");
        scanf("%d", &(ps->data[pos].age));
        printf("请输入性别:");
        scanf("%s", ps->data[pos].sex);
        printf("请输入电话:");
        scanf("%s", ps->data[pos].tele);
        printf("请输入地址:");
        scanf("%s", ps->data[pos].addr);

        printf("修改成功\n");
    }
}


int int_name(const void*e1,const void*e2)//比较函数
{
    return strcmp(((struct peoinfo*)e1)->name, ((struct peoinfo*)e2)->name);
}

void sortcontact(struct contact* ps)//排序函数,用库函数qsort实现
{
    if (ps->size == 0)
    {
        printf("通讯录中无联系人,不可排序\n");
    }
    else
    {
        qsort(ps->data, ps->size, sizeof(ps->data[0]), int_name);
        printf("排序成功\n");
    }
}



//void sortcontact(struct contact* ps)//排序函数,冒泡排序
//{
//  if (ps->size == 0)
//  {
//      printf("通讯录中无联系人,不可排序\n");
//  }
//  else
//  {
//      int i = 0;
//      for (i = 0; i < ps->size - 1; i++)//一趟
//      {
//          int j = 0;
//          for (j = 0; j < ps->size - 1 - j; j++)//一趟交换次数
//          {
//              if (strcmp(ps->data[j].name, ps->data[j + 1].name) > 0)
//              {
//                  struct peoinfo tmp = ps->data[j];
//                  ps->data[j] = ps->data[j + 1];
//                  ps->data[j + 1] = tmp;
//              }
//          }
//      }
//      printf("排序成功\n");
//  }
//}

void destroycontact(struct contact* ps)//销毁通讯录函数
{
    free(ps->data);
    ps->data = NULL;
}

void savecontact(struct contact* ps)//保存到文件函数
{
    FILE* pfwrite = fopen("contact.txt", "wb");
    if (pfwrite == NULL)
    {
        printf("%s\n",strerror(errno));
        return;
    }
    //写通讯录中的数据到文件中
    int i = 0;
    for (i=0;i<ps->size;i++)
    {
        fwrite(&(ps->data[i]), sizeof(struct peoinfo), 1, pfwrite);
    }
    printf("保存成功\n");
    fclose(pfwrite);
    pfwrite = NULL;
}

留言:

至此我们的通讯录管理已经完成,欢迎留言交流

  • 7
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值