热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

C语言实现简单电子通讯录(2)

这篇文章主要为大家详细介绍了C语言实现简单电子通讯录的第二部分,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的

C语言实现简单电子通讯录(2)

本文实例为大家分享了C语言实现简单电子通讯录的具体代码,供大家参考,具体内容如下

这两天学完系统调用和标准IO,之前的通讯录可以进行一些改进,将数据保存到文件中(图我这里就不发了)。

原理:每次启动程序时先从预设文件中以只读的形式读取保存的通讯录信息,然后将读取到的信息导入到结构体中;每次执行退出时,先将预设文件以更新方式打开,并将文件中之前保存的信息清空,然后将结构体的数据保存到文件中去,再退出程序。

还有在文件写入时和读取时,先写入(读取)保存数据的总个数,让程序知道有多少数据要写入(读取),然后保存每个数据的字节数(程序会根据每个数据的字节数来进行保存),最后再保存数据。

与之前版本相比,改动的只有主函数中加入了读取文件和写入数据这两个步骤,还有的变动就是结构体从一个变成了两个,将指针从原来一个结构体中分离出来,方便数据从文件中的导入导出。下面是代码(免得你们往前翻,我把之前的代码都修改了考过来):

头文件 head.h 唯一修改的就是结构体

#ifndef HEAD_H_
#define HEAD_H_

#include 
#include 
#include 
#include        // sleep函数头文件
#include 
#include 
#include 

#define uint unsigned int
#define OK      0
#define ERROR     -1
#define MALLOC_ERROR -2
#define N       20 


typedef int ElementType;
typedef struct data
{
  ElementType ID;       // ID号
  char Name [N];       // 姓名
  char Mobile_Phone [N];   // 手机号码
  char Home_Address [N];   // 家庭住址
  char Company_Tell [N];   // 公司电话

}Data;
typedef struct _Node
{
  Data data;
  struct _Node* next;     // 节点指针
}Node;

typedef Node* PNode;      // 重命名节点指针类型

//显示操作界面
int Interface_Display ();

//添加好友信息 (尾插法)
int Add_Friend (PNode head, ElementType num);

//显示所有好友信息
int Friend_Information (PNode head);

//查找好友
int Search_Friend (PNode head, char* Name);

//删除好友
void Delete_Friend (PNode head, char* Name);

#endif

源文件 head.c 关于结构体的部分要进行修改

#include "head.h"

//显示操作界面
int Interface_Display ()
{
  system ("clear");
  printf ("	*****************************************
");
  printf ("	~     欢迎使用通讯录        ~
");
  printf ("	~                    ~
");
  printf ("	~   1 >>>>>>>> 添加好友信息     ~
");
  printf ("	~   2 >>>>>>>> 列表好友信息     ~
");
  printf ("	~   3 >>>>>>>> 搜索好友       ~
");
  printf ("	~   4 >>>>>>>> 删除好友       ~
");
  printf ("	~   5 >>>>>>>> 退出         ~
");
  printf ("	~                    ~
");
  printf ("	~                    ~
");
  printf ("	~            作者:believe ~
");
  printf ("	*****************************************
");
  printf ("                      
");
  printf ("                      
");
  printf ("	请输入对应数字选择相应功能:");
}

//添加好友信息 (尾插法)
int Add_Friend (PNode head, ElementType num)
{
  if (NULL == head)
  {
    return ERROR;
  }

  //创建一个新的结点
  PNode p = (PNode) malloc(sizeof(Node)/sizeof(char));
  if (NULL == p)
  {
    return MALLOC_ERROR;
  }

  //将新数据赋给新结点
  system("clear");  
  printf ("	*************添加好友***************
");

  p->data.ID = num;
  printf ("	好友的ID为:%d
", p->data.ID);
  printf ("
");

  printf ("	请输入好友的名字:");
  scanf ("%s", p->data.Name);
  printf ("
");

  printf ("	请输入好友的手机号:");
  scanf ("%s", p->data.Mobile_Phone);
  printf ("
");

  printf ("	请输入好友的家庭住址:");
  scanf ("%s", p->data.Home_Address);
  printf ("
");

  printf ("	请输入好友的公司电话:");
  scanf ("%s", p->data.Company_Tell);
  printf ("
");

  p->next = NULL;

  //找到最后一个结点
  PNode Ptmp;         //将头结点地址给临时指针Ptmp
  Ptmp = head;
  while (Ptmp->next)
  {
    Ptmp = Ptmp->next;
  }
  Ptmp->next = p;

  return OK;

}

//显示所有好友信息
int Friend_Information (PNode head)
{
  if (NULL == head)
  {
    return ERROR;
  }

  PNode p = head->next;

  printf ("	ID	姓名		手机号		住址			公司电话
");

  while (p)
  {
    printf ("	%d	%s		%s		%s			%s
", p->data.ID, p->data.Name,p->data.Mobile_Phone, p->data.Home_Address, p->data.Company_Tell);
    p = p->next;
  }
  putchar("
");

  return OK;
}

//通过名字查找好友
int Search_Friend (PNode head, char* Name)     
{
  PNode p = head;
  PNode q = NULL;

  if ((NULL != p) && NULL != (p->next))
  {
    while (p->next) 
    {
      q = p->next;
      if ((NULL != q) && 0 == (strcmp(q->data.Name, Name)))
      {
        printf ("	好友信息: 
	ID:%d
	姓名: %s
	手机号码: %s
	家庭地址:%s
	公司电话: %s
", q->data.ID, q->data.Name, q->data.Mobile_Phone, q->data.Home_Address, q->data.Company_Tell);
      }
      else
      {
        printf ("	对不起,您的通讯录没有该好友!
");
      }
      p = p->next;
    }
  }

  /* 另一种做法
  if (NULL == head)
  {
    return ERROR;
  }

  PNode p;
  int flag = 1;
  for (p = head->next; p != NULL; p = p->next)
  {
    if (0 == strcmp(p->data.Name, Name))
    {
      flag = 0;
      printf ("	好友信息:
	ID: %d
	姓名: %s
	手机号码: %s
	家庭地址: %s
	公司电话: %s
", p->data.ID, p->data.Name, p->data.Mobile_Phone, p->data.Home_Address, p->data.Company_Tell);
    }
  }
  fi (flag)
  {
    printf ("	对不起,您的通讯录没有该好友!
");
  }

  putchar("
");
  */

  return OK;
}

//删除好友
void Delete_Friend (PNode head, char* Name)
{
  PNode p = head;
  PNode q = NULL;

  while (NULL != p && NULL != (p->next))
  {
    q = p->next;
    if (NULL != q && 0 == strcmp(q->data.Name, Name))
    {
      p->next = q->next;
      free(q);

      int j;

      printf ("	正在删除
");
      printf ("	请稍候");
      fflush (stdout);      //强制刷新缓存,输出显示
      for (j = 0; j <2; j++)
      {
        sleep (1);       //linux使用sleep,参数为秒
        printf (".");
        fflush(stdout);     //强制刷新缓存,输出显示
      }
      printf ("
");
      printf ("	该好友已成功删除!
");
    }
    else if (NULL == q->next && 0 != strcmp(q->data.Name, Name))
    {
      printf ("	您的通讯录没有该好友!
");
    }
    p = p->next;
  }
}

主函数 main.c 加入了数据的读取和写入,已标明

/*******************************************************************
需求:制作一个电子通讯录,通过该通讯录能录入好友ID号、姓名(英文)、手
机号码,家庭住址,公司电话。
模块:
  主界面:主要显示软件功能,A)添加好友信息 B)列表好友信息。(包含排序
  功能) C)搜索好友 D)删除好友
  A)用户输入INSERT命令后,让用户输入好友信息。添加成功或失败都需提示
  B)用户输入DISPLAY命令后,好友信息升序排列
  C)用户输入SEARCH命令后,让用户输入将要搜索好友姓名查询。如果未搜索
  到请友好提示。如果搜索到,显示处该好友信息
  D)用户输入DELETE命令后,让用户输入将要删除好友姓名删除,如果存在同
  名的多个好友,则列表出,所有同名的好友信息,让用户通过输入ID号删除
  提示用户删除成功。
**********************************************************************/
#include "head.h"

int main ()
{
  int Function;
  int i = 0;
  char Name[N];
  int cho;

  // 创建头结点并为其分配空间
  PNode head_node = (PNode) malloc(sizeof(Node)/sizeof(char));
  if (NULL == head_node)
  {
    return MALLOC_ERROR;
  }
  head_node->next = NULL;


/****************************************************************
  打开存放信息的文件并将里面的数据导入到链表中区
****************************************************************/
  // 以只读方式打开存放信息的文件,
  FILE *fp1 = fopen ("student.txt", "r+"); 
  if (NULL == fp1) 
  { 
    printf ("fopen"); 
    return -1; 
  } 

  PNode tmp = head_node; 
  int count; 
  int ret; 

  // 保存读记录的个数并判断是否读到文件结尾,如果读到文件结尾,
  // 它返回一个非0 的值
  ret = fread (&count, sizeof(int), 1, fp1); 
  if(ret != 0) 
  { 
    for (i = 0; i data), len, 1, fp1);   // 读取数据 

      node->next = NULL;   
      while (tmp->next)            // 向后遍历
      { 
        tmp = tmp->next; 
      } 
      tmp->next = node;            // 将结点导入链表中 
    } 
  } 

  if (ret == 0 && !feof(fp1))           // 读取失败
  { 
    perror ("fread"); 
    return -1; 
  } 
  fclose (fp1);                  // 关闭刚刚打开的文件
/****************************************************************
  文件导入链表结束,关闭文件
****************************************************************/  

  i = 1;               // i初始化(i既是ID编号)
  while (1)
  {
    Interface_Display ();      // 主界面
    scanf ("%d", &Function);

    switch (Function)        // 功能选择
    {
      case 1:           // 添加好友
      {
        Function = 0;
        Add_Friend (head_node, i++);
        int j;

        printf ("	正在添加
");
        printf ("	请稍候");
        fflush (stdout);    // 强制刷新缓存,输出显示
        for (j = 0; j <2; j++)
        {
          sleep (1);     // Linux 使用sleep,参数为秒
          printf (".");
          fflush (stdout);  // 强制刷新缓存,输出显示
        }
        printf ("
");
        printf ("	添加成功!
");
        printf ("	返回主菜单请输入1:");
        scanf ("%d", &cho);
        if (1 == cho)
        {
          break;
        }
        else
        {
          printf ("	对不起!您的输入有误!请重新输入:");
          scanf ("%d", &cho);
          break;
        }
        break;
      }      
      case 2:         // 显示好友信息
      {
        system ("clear");
        printf ("	***********好友信息******************
");
        printf ("
");

        Friend_Information (head_node);
        Function = 0;
        printf ("	返回主菜单请输入1:");
        scanf ("%d", &cho);
        if (1 == cho)
        {
          break;
        }
        else
        {
          printf ("	对不起!您的输入有误!请重新输入:");
          scanf ("%d", &cho);
          break;
        }
        break;
      }
      case 3:         // 查找好友
      {
        system ("clear");
        printf ("	*************查找好友*************
");
        printf ("	请输入您要查找的好友姓名:");
        scanf ("%s", Name);
        printf ("
");

        int j;     
        printf ("	正在查找
");
        printf ("	请稍候");
        fflush (stdout);    // 强制刷新缓存,输出显示
        for (j = 0; j <2; j++)
        {
          sleep (1);     // Linux 使用sleep,参数为秒
          printf (".");
          fflush (stdout);  // 强制刷新缓存,输出显示
        }
        printf ("
");
        Search_Friend (head_node, Name);
        printf ("	返回主菜单请输入1:");
        scanf ("%d", &cho);
        if (1 == cho)
        {
          break;
        }
        else
        {
          printf ("	对不起!您的输入有误!请重新输入:");
          scanf ("%d", &cho);
          break;
        }
        break;
      }
      case 4:           //删除好友
      {
        system ("clear");
        printf ("	*************删除好友*************
");
        printf ("	请输入要删除好友的姓名:");
        scanf ("%s", Name);
        printf ("
");
        Delete_Friend (head_node, Name);
        printf ("	返回主菜单请输入1:");
        scanf ("%d", &cho);
        if (1 == cho)
        {
          break;
        }
        else
        {
          printf ("	对不起!您的输入有误!请重新输入:");
          scanf ("%d", &cho);
          break;
        }
        break;
      }  
      case 5:           //退出通讯录
      {

/****************************************************************
        退出程序前,将链表中的数据导入到文件中去
****************************************************************/
        // 以更新方式打开存放信息的文件(打开时会清空文件)
        FILE *fp2 = fopen ("student.txt", "wb+"); 
        if(NULL == fp2) 
        { 
          printf ("fopen"); 
          return -1; 
        } 

        tmp = head_node->next;     // tmp为第一个结点
        count = 0;           // 用来存放链表长度
        while(tmp)           // 求链表长度
        { 
          count++; 
          tmp = tmp->next; 
        }

        // 要写入数据的个数
        fwrite(&count, sizeof(int), 1, fp2); 
        tmp = head_node;        // tmp初始化
        while (tmp->next) 
        { 
          Node* p = tmp->next; 
          tmp->next = p->next; 

          // 写入数据的长度         
          int len = sizeof(p->data); 
          fwrite (&len, sizeof(int), 1, fp2); 

          //写入数据 
          fwrite (&(p->data), sizeof(Data), 1, fp2); 

          free (p);          
        } 
        fclose (fp2); 
/****************************************************************
        数据保存结束,即将退出程序
****************************************************************/

        Function = 0;
        system ("clear");
        exit (0);
      }
      default:          //输入有误
      {
        Function = 0;
        printf ("	对不起!您的输入有误!请重新输入:");
        scanf ("%d", &Function);
        break;
      }      
    }    
  }  

  return 0;
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持编程笔记。

原文链接:https://blog.csdn.net/believe_s/article/details/77151762


推荐阅读
  • 深入解析Linux下的I/O多路转接epoll技术
    本文深入解析了Linux下的I/O多路转接epoll技术,介绍了select和poll函数的问题,以及epoll函数的设计和优点。同时讲解了epoll函数的使用方法,包括epoll_create和epoll_ctl两个系统调用。 ... [详细]
  • 本文讨论了clone的fork与pthread_create创建线程的不同之处。进程是一个指令执行流及其执行环境,其执行环境是一个系统资源的集合。在调用系统调用fork创建一个进程时,子进程只是完全复制父进程的资源,这样得到的子进程独立于父进程,具有良好的并发性。但是二者之间的通讯需要通过专门的通讯机制,另外通过fork创建子进程系统开销很大。因此,在某些情况下,使用clone或pthread_create创建线程可能更加高效。 ... [详细]
  • 本文详细介绍了Linux中进程控制块PCBtask_struct结构体的结构和作用,包括进程状态、进程号、待处理信号、进程地址空间、调度标志、锁深度、基本时间片、调度策略以及内存管理信息等方面的内容。阅读本文可以更加深入地了解Linux进程管理的原理和机制。 ... [详细]
  • Vagrant虚拟化工具的安装和使用教程
    本文介绍了Vagrant虚拟化工具的安装和使用教程。首先介绍了安装virtualBox和Vagrant的步骤。然后详细说明了Vagrant的安装和使用方法,包括如何检查安装是否成功。最后介绍了下载虚拟机镜像的步骤,以及Vagrant镜像网站的相关信息。 ... [详细]
  • 全面介绍Windows内存管理机制及C++内存分配实例(四):内存映射文件
    本文旨在全面介绍Windows内存管理机制及C++内存分配实例中的内存映射文件。通过对内存映射文件的使用场合和与虚拟内存的区别进行解析,帮助读者更好地理解操作系统的内存管理机制。同时,本文还提供了相关章节的链接,方便读者深入学习Windows内存管理及C++内存分配实例的其他内容。 ... [详细]
  • 本文介绍了数据库的存储结构及其重要性,强调了关系数据库范例中将逻辑存储与物理存储分开的必要性。通过逻辑结构和物理结构的分离,可以实现对物理存储的重新组织和数据库的迁移,而应用程序不会察觉到任何更改。文章还展示了Oracle数据库的逻辑结构和物理结构,并介绍了表空间的概念和作用。 ... [详细]
  • Linux重启网络命令实例及关机和重启示例教程
    本文介绍了Linux系统中重启网络命令的实例,以及使用不同方式关机和重启系统的示例教程。包括使用图形界面和控制台访问系统的方法,以及使用shutdown命令进行系统关机和重启的句法和用法。 ... [详细]
  • Python语法上的区别及注意事项
    本文介绍了Python2x和Python3x在语法上的区别,包括print语句的变化、除法运算结果的不同、raw_input函数的替代、class写法的变化等。同时还介绍了Python脚本的解释程序的指定方法,以及在不同版本的Python中如何执行脚本。对于想要学习Python的人来说,本文提供了一些注意事项和技巧。 ... [详细]
  • eclipse学习(第三章:ssh中的Hibernate)——11.Hibernate的缓存(2级缓存,get和load)
    本文介绍了eclipse学习中的第三章内容,主要讲解了ssh中的Hibernate的缓存,包括2级缓存和get方法、load方法的区别。文章还涉及了项目实践和相关知识点的讲解。 ... [详细]
  • 本文介绍了在Linux下安装Perl的步骤,并提供了一个简单的Perl程序示例。同时,还展示了运行该程序的结果。 ... [详细]
  • Linux磁盘的分区、格式化的观察和操作步骤
    本文介绍了如何观察Linux磁盘的分区状态,使用lsblk命令列出系统上的所有磁盘列表,并解释了列表中各个字段的含义。同时,还介绍了使用parted命令列出磁盘的分区表类型和分区信息的方法。在进行磁盘分区操作时,根据分区表类型选择使用fdisk或gdisk命令,并提供了具体的分区步骤。通过本文,读者可以了解到Linux磁盘分区和格式化的基本知识和操作步骤。 ... [详细]
  • Go GUIlxn/walk 学习3.菜单栏和工具栏的具体实现
    本文介绍了使用Go语言的GUI库lxn/walk实现菜单栏和工具栏的具体方法,包括消息窗口的产生、文件放置动作响应和提示框的应用。部分代码来自上一篇博客和lxn/walk官方示例。文章提供了学习GUI开发的实际案例和代码示例。 ... [详细]
  • 先看官方文档TheJavaTutorialshavebeenwrittenforJDK8.Examplesandpracticesdescribedinthispagedontta ... [详细]
  • IOS开发之短信发送与拨打电话的方法详解
    本文详细介绍了在IOS开发中实现短信发送和拨打电话的两种方式,一种是使用系统底层发送,虽然无法自定义短信内容和返回原应用,但是简单方便;另一种是使用第三方框架发送,需要导入MessageUI头文件,并遵守MFMessageComposeViewControllerDelegate协议,可以实现自定义短信内容和返回原应用的功能。 ... [详细]
  • 上图是InnoDB存储引擎的结构。1、缓冲池InnoDB存储引擎是基于磁盘存储的,并将其中的记录按照页的方式进行管理。因此可以看作是基于磁盘的数据库系统。在数据库系统中,由于CPU速度 ... [详细]
author-avatar
月色清朗
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有