服务器之家

服务器之家 > 正文

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

时间:2021-09-14 14:00     来源/作者:believe_s

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

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

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

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

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

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

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
#ifndef HEAD_H_
#define HEAD_H_
 
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>       // sleep函数头文件
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
 
#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 关于结构体的部分要进行修改

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
#include "head.h"
 
//显示操作界面
int Interface_Display ()
{
  system ("clear");
  printf ("\t*****************************************\n");
  printf ("\t~     欢迎使用通讯录        ~\n");
  printf ("\t~                    ~\n");
  printf ("\t~   1 >>>>>>>> 添加好友信息     ~\n");
  printf ("\t~   2 >>>>>>>> 列表好友信息     ~\n");
  printf ("\t~   3 >>>>>>>> 搜索好友       ~\n");
  printf ("\t~   4 >>>>>>>> 删除好友       ~\n");
  printf ("\t~   5 >>>>>>>> 退出         ~\n");
  printf ("\t~                    ~\n");
  printf ("\t~                    ~\n");
  printf ("\t~            作者:believe ~\n");
  printf ("\t*****************************************\n");
  printf ("                      \n");
  printf ("                      \n");
  printf ("\t请输入对应数字选择相应功能:");
}
 
//添加好友信息 (尾插法)
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 ("\t*************添加好友***************\n");
 
  p->data.ID = num;
  printf ("\t好友的ID为:%d\n", p->data.ID);
  printf ("\n");
 
  printf ("\t请输入好友的名字:");
  scanf ("%s", p->data.Name);
  printf ("\n");
 
  printf ("\t请输入好友的手机号:");
  scanf ("%s", p->data.Mobile_Phone);
  printf ("\n");
 
  printf ("\t请输入好友的家庭住址:");
  scanf ("%s", p->data.Home_Address);
  printf ("\n");
 
  printf ("\t请输入好友的公司电话:");
  scanf ("%s", p->data.Company_Tell);
  printf ("\n");
 
  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 ("\tID\t姓名\t\t手机号\t\t住址\t\t\t公司电话\n");
 
  while (p)
  {
    printf ("\t%d\t%s\t\t%s\t\t%s\t\t\t%s\n", p->data.ID, p->data.Name,p->data.Mobile_Phone, p->data.Home_Address, p->data.Company_Tell);
    p = p->next;
  }
  putchar('\n');
 
  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 ("\t好友信息: \n\tID:%d\n\t姓名: %s\n\t手机号码: %s\n\t家庭地址:%s\n\t公司电话: %s\n", q->data.ID, q->data.Name, q->data.Mobile_Phone, q->data.Home_Address, q->data.Company_Tell);
      }
      else
      {
        printf ("\t对不起,您的通讯录没有该好友!\n");
      }
      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 ("\t好友信息:\n\tID: %d\n\t姓名: %s\n\t手机号码: %s\n\t家庭地址: %s\n\t公司电话: %s\n", p->data.ID, p->data.Name, p->data.Mobile_Phone, p->data.Home_Address, p->data.Company_Tell);
    }
  }
  fi (flag)
  {
    printf ("\t对不起,您的通讯录没有该好友!\n");
  }
 
  putchar('\n');
  */
 
  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 ("\t正在删除\n");
      printf ("\t请稍候");
      fflush (stdout);      //强制刷新缓存,输出显示
      for (j = 0; j < 2; j++)
      {
        sleep (1);       //linux使用sleep,参数为秒
        printf (".");
        fflush(stdout);     //强制刷新缓存,输出显示
      }
      printf ("\n");
      printf ("\t该好友已成功删除!\n");
    }
    else if (NULL == q->next && 0 != strcmp(q->data.Name, Name))
    {
      printf ("\t您的通讯录没有该好友!\n");
    }
    p = p->next;
  }
}

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

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
/*******************************************************************
需求:制作一个电子通讯录,通过该通讯录能录入好友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 < count; i++)
    {
      // 创建新结点用来保存读取的数据
      Node *node = (Node*)malloc(sizeof(Node)/sizeof(char));  
 
      int len;         
      fread (&len, sizeof(int), 1, fp1);   // 读取数据长度
      fread (&(node->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 ("\t正在添加\n");
        printf ("\t请稍候");
        fflush (stdout);    // 强制刷新缓存,输出显示
        for (j = 0; j < 2; j++)
        {
          sleep (1);     // Linux 使用sleep,参数为秒
          printf (".");
          fflush (stdout);  // 强制刷新缓存,输出显示
        }
        printf ("\n");
        printf ("\t添加成功!\n");
        printf ("\t返回主菜单请输入1:");
        scanf ("%d", &cho);
        if (1 == cho)
        {
          break;
        }
        else
        {
          printf ("\t对不起!您的输入有误!请重新输入:");
          scanf ("%d", &cho);
          break;
        }
        break;
      }     
      case 2:         // 显示好友信息
      {
        system ("clear");
        printf ("\t***********好友信息******************\n");
        printf ("\n");
 
        Friend_Information (head_node);
        Function = 0;
        printf ("\t返回主菜单请输入1:");
        scanf ("%d", &cho);
        if (1 == cho)
        {
          break;
        }
        else
        {
          printf ("\t对不起!您的输入有误!请重新输入:");
          scanf ("%d", &cho);
          break;
        }
        break;
      }
      case 3:         // 查找好友
      {
        system ("clear");
        printf ("\t*************查找好友*************\n");
        printf ("\t请输入您要查找的好友姓名:");
        scanf ("%s", Name);
        printf ("\n");
 
        int j;    
        printf ("\t正在查找\n");
        printf ("\t请稍候");
        fflush (stdout);    // 强制刷新缓存,输出显示
        for (j = 0; j < 2; j++)
        {
          sleep (1);     // Linux 使用sleep,参数为秒
          printf (".");
          fflush (stdout);  // 强制刷新缓存,输出显示
        }
        printf ("\n");
        Search_Friend (head_node, Name);
        printf ("\t返回主菜单请输入1:");
        scanf ("%d", &cho);
        if (1 == cho)
        {
          break;
        }
        else
        {
          printf ("\t对不起!您的输入有误!请重新输入:");
          scanf ("%d", &cho);
          break;
        }
        break;
      }
      case 4:           //删除好友
      {
        system ("clear");
        printf ("\t*************删除好友*************\n");
        printf ("\t请输入要删除好友的姓名:");
        scanf ("%s", Name);
        printf ("\n");
        Delete_Friend (head_node, Name);
        printf ("\t返回主菜单请输入1:");
        scanf ("%d", &cho);
        if (1 == cho)
        {
          break;
        }
        else
        {
          printf ("\t对不起!您的输入有误!请重新输入:");
          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 ("\t对不起!您的输入有误!请重新输入:");
        scanf ("%d", &Function);
        break;
      }     
    }   
  
 
  return 0;
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持服务器之家。

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

标签:

相关文章

热门资讯

yue是什么意思 网络流行语yue了是什么梗
yue是什么意思 网络流行语yue了是什么梗 2020-10-11
2020微信伤感网名听哭了 让对方看到心疼的伤感网名大全
2020微信伤感网名听哭了 让对方看到心疼的伤感网名大全 2019-12-26
背刺什么意思 网络词语背刺是什么梗
背刺什么意思 网络词语背刺是什么梗 2020-05-22
苹果12mini价格表官网报价 iPhone12mini全版本价格汇总
苹果12mini价格表官网报价 iPhone12mini全版本价格汇总 2020-11-13
2021年耽改剧名单 2021要播出的59部耽改剧列表
2021年耽改剧名单 2021要播出的59部耽改剧列表 2021-03-05
返回顶部