数据结构

设计并编程实现一个应用单链表存储结构的通信录管理系统。定义适当数据类型,设计并编写完成下列8项基本功能的C语言程序:

[root@huhy ~]# cat test.c
/*一、功能要求:
 1.添加 2.查找 3.删除 4.修改 */
/*二、包含信息
 每一个联系人包含编号、姓名、手机、qq等信息*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

typedef struct TongXun{
        char ID[15];
        char name[12];
        char tel[15];
        char QQ[15];
}datatype;

typedef struct node{
        datatype data;
        struct node *next;
}Node, *LinkNode;

LinkNode head =NULL; //先确定头结点

/*功能函数*/
void Menu();                            //0.显示界面
void InitList();                        //1.通讯录的初始化***

                                                        //2.建立通讯录(单链表的建立)*******
void CreateList_Head();         //2.1头插法创建
int JudgeID(LinkNode ptr1,LinkNode ptr2);//2.11判断ID是否重复
void CreateList_Tail();         //2.2尾插法创建

void DelList();                 //3.删除数据(根据姓名删除) ********

void UpdateList();      //4.修改信息******

void SearchList();              //5.查找信息

                                                        //6.插入信息 ******
void InsertList_Pre();          //6.1前插法
void InputList(LinkNode ptr);                   //6.11输入指定结点指针相关信息
void InsertList_Back();         //6.2后插法


void TraverseList(); //7.遍历通讯录
void PrintList(LinkNode ptr); //7.1输出指定信息

void DestroyList();  //8.释放通讯录***

void save(LinkNode stu);        //9.用于将数组中的信息保存到文件中 (n个同学)
int NumList(LinkNode head);     //9.1 通过头指针,遍历单链表进行人数统计并返回

/*主控函数*/
int  main(void)
{
        int choice;

        while (1)
        {
                Menu();
                printf("请选择菜单:");
                scanf("%d", &choice);
                switch (choice)
                {
                    case 1:             //初始化通讯录
                                        InitList(); break;
                        case 2:         //建立通讯录
                                {
                                                int num;
                                                printf("请选择建立方式(1.头插法 2.尾插法):");
                                                scanf("%d", &num);   /****下边的大括号不可以省略 ****/
                                                if (num == 1)   //头插法
                                                {
                                                   CreateList_Head();  break;
                                                }

                                                if (num == 2)   //尾插法
                                                {
                                                        CreateList_Tail();  break;
                                                }


                                }

                        case 3:         //删除信息
                                        DelList(); break;

                        case 4:         //修改信息
                                        UpdateList(); break;

                        case 5:         //查找信息
                                        SearchList(); break;

                        case 6:         //添加信息
                       {

                                        int num;
                                        printf("请选择插入方式(1.前插法 2.尾插法):");
                                        scanf("%d", &num);
                                        if (num == 1) //头插法
                                        {
                                                InsertList_Pre(); break;
                                        }
                                        if (num == 2) //尾插法
                                        {
                                            InsertList_Back(); break;
                                        }


                      }


                        case 7:         //遍历通讯录
                                        TraverseList(); break;

                        case 8: //释放通讯录
                                        DestroyList(); break;


                    default: return 0;
                }
        }
        return 0;
}

/*0.显示界面*/
void Menu()
{
        printf("\t\t\t****************************************************\n");
        printf("\t\t\t1.初始化通讯录                          2.建立通讯录\n");
        printf("\t\t\t3.删除联系人                            4.修改联系人\n");
        printf("\t\t\t5.查找联系人                            6.添加联系人\n");
        printf("\t\t\t7.输出通讯录                            8.释放通讯录\n");
        printf("\t\t\t***************其他按键视为退出**********************\n");
}
/*1.通讯录的初始化*/
//(建立头结点)
void InitList()
{
     head= (LinkNode)malloc(sizeof(Node));
     head->next = NULL;
     printf("初始化成功!\n");
     save(head);
}



/*2.建立通讯录(单链表的建立)*/
//2.1插入数据,建立通讯录(头插法)
//首先建立新结点,为其动态开辟内存空间
//接着为该结点的data域存入数据
//然后经典头插代码(为让新结点插入头结点和首元结点之间)
//先让新结点的指针域指向原首元结点;然后让头结点的指针域指向该新结点
//就是通过结点间的指针进行连接/相关联
void CreateList_Head()
{
    LinkNode ptr;
    LinkNode ptr2;
    int item=1;         //用来控制循环(决定是否继续录入信息)
    int n;

    //需要实现:循环直至自行决定,不继续建立新的联系人
    //另写函数2.11,实现排除输入ID重复
    while(item!=0)
    {
        ptr = (LinkNode)malloc(sizeof(Node));//动态分配内存(新结点)


        printf("请输入通讯信息:\n");
        printf("请输入姓名:");
        scanf("%s",&(ptr->data.ID));
                ptr2 = head->next;
        n = JudgeID(ptr,ptr2); //ID有重复n=1;
        //调用函数2.11进行判断ID是否重复

        while(n==1)
        {
                        printf("姓名已存在请重新输入:\n");
                        scanf("%s",&(ptr->data.ID));
                        n = JudgeID(ptr,ptr2); //ID有重复n=1;
                }

                    printf("请输入电话:\n");
                    scanf("%s",&(ptr->data.name));
                    printf("请输入QQ:\n");
                    scanf("%s",&(ptr->data.tel));
                    printf("请输入住址:\n");
                    scanf("%s",&(ptr->data.QQ));

                    //头插法的经典代码
                    ptr->next = head->next;
                    head->next = ptr;
                    printf("是否继续录入,继续:1,退出:0\n");
                    scanf("%d",&item);

        }
        save(head);
   return ;
}
//2.11实现排除输入ID重复
//传入需要比对的结点,遍历单链表逐个比对ID
//要和首元结点的指针域指向的结点比较(次首元结点),就是新结点的指针域所指结点
//如果有重复返回 1,否则返回0;
int JudgeID(LinkNode ptr1,LinkNode ptr2)
{
        while(ptr2!=NULL)
        {
          if(strcmp(ptr1->data.ID,ptr2->data.ID)==0)
             return 1;
          else
            ptr2 = ptr2->next;
        }
        return 0;
}

/*
   2.建立通讯录(单链表的建立)
//2.1插入数据,建立通讯录(头插法)
void CreateList_Head()
{
        List *node; //插入节点
        int count = 0;
        int flag = 1;//用于判断是否继续输入下一条记录
        while (flag)
        {
                count++;
                node= (List *)malloc(sizeof(List));  //为新建的结点开辟内存
                printf("插入第%d条记录:\n", count);
                printf("ID:");
                scanf("%s", &(node->data.ID));

                //判断ID是否重复
                List *p = L->next;
                while (p)
                {
                        if (strcmp(p->data.ID, node->data.ID) == 0)
                        {
                                printf("ID重复,请重新输入!\n");
                                printf("ID:");
                                scanf("%s", &(node->data.ID));
                        }
                        else
                                p = p->next;
                }
                printf("姓名:");
                scanf("%s", &(node->data.name));
                printf("电话:");
                scanf("%s", &(node->data.tel));
                printf("QQ:");
                scanf("%s", &(node->data.QQ));

                //头插法的核心 代码
                node->next = L->next; //对于新建立的结点,先确定其指针域,再确定头指针的指向
                L->next = node;
                printf("是否继续录入(1.继续 0.完成录入):");
                scanf("%d", &flag);
                if (flag == 0)
                        break;
        }
}
*/
/*2.2尾插法创建 */
//首先先建立一个新结点和一个尾结点rear(一直指向尾结点)
//依旧需要为新结点的data域赋值
//依旧让rear指向头结点
//但是尾插法核心仍然是:
//先让新结点作为rear的后继结点,然后让rear还指向尾结点;
void CreateList_Tail()
{
        LinkNode  ptr, rear=head;       //定义新结点和尾节点
        LinkNode  ptr2;
    int item = 1; //用来判断循环是否继续
    int n;
    while(item ==1 )
        {
                ptr = (LinkNode)malloc(sizeof(Node));//动态分配内存(新结点)
                printf("请输入通讯信息:\n");
        printf("请输入姓名:");
        scanf("%s",&(ptr->data.ID));
                  ptr2 = head->next;
            n = JudgeID(ptr,ptr2); //ID有重复n=1;
        //调用函数2.11进行判断ID是否重复

        while(n==1)             //如果加入的ID与之前重复,便一直循环重新输入
        {
                        printf("姓名已存在请重新输入:\n");
                        scanf("%s",&(ptr->data.ID));
                        n = JudgeID(ptr,ptr2); //ID有重复n=1;
                }
        printf("请输入电话:\n");
                scanf("%s",&(ptr->data.name));
                printf("请输入QQ:\n");
                scanf("%s",&(ptr->data.tel));
                printf("请输入住址:\n");
                scanf("%s",&(ptr->data.QQ));
        //尾插法的核心
                  rear->next =ptr;
                  ptr->next = NULL;
                  rear = ptr;

                printf("是否继续录入,继续:1,退出:0\n");
                scanf("%d",&item);
        }
        return ;
}

/*3.删除数据(根据姓名删除)
void DelList()
{
        LinkNode p = head->next;
        LinkNode q = head;
        char ID[10];
        printf("请输入要删除的ID:");
        scanf("%s", &ID);
        while (p && strcmp(p->data.ID, ID) != 0)//找到ID一样的,让P指向该结点
        {
                p = p->next;
                q = q->next;
        }
        if (p!= NULL)
        {
                q->next = p->next;
                printf("删除成功!\n");
                printf("被删除数据的信息为:\n");
                printf("ID:%s\n", p->data.ID);
                printf("姓名:%s\n", p->data.name);
                printf("电话:%s\n", p->data.tel);
                printf("QQ:%s\n", p->data.QQ);
                free(p);
        }
        else
        {
                printf("通讯录中不存在此人信息!\n");
        }
}
*/
/*3.删除数据(根据姓名删除)*/
//关键点:单链表所有结点间通过结点指针的指向来联系
//首先明确两个结点指针:p,q
//分别用来 使整个单链表联系;找到需要删除信息的结点
//然后需要保证-->单链表的完整连接性,
//最后释放-->需要删除的结点信息
void DelList()
{
   LinkNode p, q;
   p = head;
   q = head->next;
   char str[20];
   printf("请输入需要删除姓名:\n");
   scanf("%s",&str);
   //可能一开始就是空链表,或者遍历到最后(通过指针q来判断)
   while(q!=NULL && strcmp(q->data.ID,str)!=0) //遍历单链表,让结点指针q指向需要删除的结点
   {
          q = q->next;
          p = p->next;
   }

   if(q==NULL)//遍历单链表找不到该信息
   {
                printf("该信息不存在\n");
   }
   else
   {
           p->next = q->next;   //让p在单链表中保证其完整性
           printf("该信息如下:\n");
           printf("姓名为:%s\n",q->data.ID);
           printf("电话为:%s\n",q->data.name);
           printf("QQ号为:%s\n",q->data.tel);
           printf("住址为:%s\n",q->data.QQ);

           printf("通讯信息删除成功\n");
             free(q);
   }
   save(head);

}

/*4.修改信息*/
//先定义一个结点指针,让它指向首元结点
//进行遍历,与输入的ID进行比较,找到该信息所在结点
//然后进行(循环)修改信息:1.姓名 2.手机号 3.QQ号
void UpdateList()
{
        LinkNode ptr = head->next;
        int item=1,n=0;  //item控制是否继续修改信息
                                        //n用来判断是否修改该信息
        char str[20];
        printf("请输入需要修改的姓名:\n");
        scanf("%s",&str);
        while(ptr!=NULL&&strcmp(ptr->data.ID,str)!=0)//遍历单链表,找和所输入ID一致的结点
        {
                ptr = ptr->next;
        }
        if(ptr==NULL)
        {
                 printf("该信息不存在\n");
                 return ;
        }
          //能够找到该信息,就会进行修改
        while(item==1) //item为1表示继续修改信息
        {
                printf("是否要修改电话:(是:1 否:0)\n");
                   if(scanf("%d",&n),n==1)
                     scanf("%s",&ptr->data.name);
                printf("是否要修改QQ号:(是:1 否:0)\n");
                        if(scanf("%d",&n),n==1)
                      scanf("%s",&ptr->data.tel);
                printf("是否要修改住址:(是:1 否:0)\n");
                        if(scanf("%d",&n),n==1)
                      scanf("%s",&ptr->data.QQ);
                printf("是否需要重新修改信息:(是:1 否:0)\n");
                scanf("%d",&item);
        }
        save(head);
}

/*5.查找指定ID信息*/
//先定义一个结点指针,指向首元结点
//接着进行遍历单链表,找该信息所在结点
//然后输出该信息
void SearchList()
{
        LinkNode ptr = head->next;
        char str[20];
        printf("请输入需要查找的姓名:\n");
        scanf("%s",&str);
        while(ptr!=NULL&&strcmp(ptr->data.ID,str)!=0)//遍历单链表,找和所输入ID一致的结点
        {
                ptr = ptr->next;
        }
        if(ptr==NULL)
        {
                 printf("该信息不存在\n");
                 return ;
        }

        //如果找到该信息,就会输出

        printf("该通讯信息如下:\n");
        PrintList(ptr);

}


/*5.插入数据(前插法)
void InsertList_Pre()
{
   LinkNode p, q, r, node;
        p = head->next;
        q = head;
        char ID[10];
        printf("请输入要插入的位置(输入该位置的ID):");
        scanf("%s", &ID);//结点指针p用来遍历指向 所要添加的位置
        while (p && strcmp(p->data.ID, ID) != 0)
        {
                p = p->next;
                q = q->next;
        }

        if (p != NULL)
        {
                //给新节点分配空间
                node = (LinkNode)malloc(sizeof(Node));
                //录入新节点数据
                printf("请输入新节点信息:\n");
           judge:
                        printf("ID:");
                        scanf("%s", &(node->data.ID));
                        //判断ID是否重复
                    r = head->next;
                    while (r && strcmp(r->data.ID, node->data.ID) != 0)
                        {
                                r = r->next;
                    }
          if (r != NULL)
          {
                  printf("ID重复,请重新输入!\n");
                  //free(r); 这里不能free(r),如果释放r,会失去r的下一个节点的信息,链表会被破坏
                  goto judge;
          }
          printf("姓名:");
          scanf("%s", &(node->data.name));
          printf("电话:");
          scanf("%s", &(node->data.tel));
          printf("QQ:");
          scanf("%s", &(node->data.QQ));
        }



        else
        {
                //如果没有该ID
                int choice1;
                printf("通讯录中未找到该ID,是否使用默认方式将新节点插入到最后(1.是 0.否):");
                scanf("%d", &choice1);
                if (choice1 == 1)
                {
                        //给新节点分配空间
                        node = (LinkNode)malloc(sizeof(Node));
                        //给新节点输入数据
                        printf("请输入新节点信息:\n");
                judge1:
                        printf("ID:");
                        scanf("%s", &(node->data.ID));
                        //判断ID是否重复
                        r = head->next;
                        while (r && strcmp(r->data.ID, node->data.ID) != 0)
                        {
                                r = r->next;
                        }
                        if (r != NULL) //ID重复
                        {
                                printf("ID重复,请重新输入!\n");
                                goto judge1;
                        }
                        printf("姓名:");
                        scanf("%s", &(node->data.name));
                        printf("电话:");
                        scanf("%s", &(node->data.tel));
                        printf("QQ:");
                        scanf("%s", &(node->data.QQ));
                }
                else
                        return;
        }
        node->next = q->next;
        q->next = node;
        printf("插入成功!\n");
}
*/
/*6.1仿头插法-->前插法(添加信息到指定位置)*/
//首先输入ID,确定新结点插入位置 (让结点指针p指向该位置)
//接着为新结点 ptr 开辟内存并赋值
//然后就是模仿-->经典的头插法
//建立链表时,(头插法建立)已经写好的函数进行封装,接着在该函数进行调用;
void InsertList_Pre()
{
         LinkNode p,L,ptr;        //结点指针L伴随着p一起移动
         char str[20];
         L = head;
         p = head->next;

         //需要输入ID,遍历单链表,将p指向该结点

         //情况1:如果p不是空指针,最后进行仿头指针的经典代码(将指针L假想成头指针)
         //情况2:如果p是空指针,此时便是头结点
         //可以为新结点 ptr 依次输入信息(记得查重复ID)
         //因此两种情可以用一个经典代码(将结点指针L,假想为头结点)进行插入添加

        printf("请输入姓名(新信息会插入到该信息前):\n");
         scanf("%s",&str);
         while(p!=NULL && strcmp(p->data.ID,str)!=0)    //逻辑符号的使用
         {
                p = p->next;
                L = L->next;
         }

         if(p==NULL)
         {
                printf("所输入的姓名不存在\n");
                return;
         }


        //为新结点开辟内存并赋值
          ptr =(LinkNode)malloc(sizeof(Node));
          //函数InputList()   为指定结点指针-->输入相关信息
           InputList(ptr);

        //经典头插法核心代码
           //仿头插法(将 L的指向看作头结点,p的指向看作首元结点)
           ptr->next = L->next;
           L->next =ptr;
           printf("信息插入成功\n");
           save(head);
}
/*6.11输入指定结点指针相关信息*/
//功能中实现防输入重复ID
void InputList(LinkNode ptr)
{
        int n=0;
        printf("请输入通讯信息:\n");
    printf("请输入姓名:");
    scanf("%s",&(ptr->data.ID));
    LinkNode ptr2 = head->next;
      n = JudgeID(ptr,ptr2); //ID有重复n=1;
        //调用函数2.11进行判断ID是否重复

    while(n==1)         //如果加入的ID与之前重复,便一直循环重新输入
    {
                printf("姓名已存在请重新输入:\n");
                scanf("%s",&(ptr->data.ID));
                n = JudgeID(ptr,ptr2); //ID有重复n=1;
        }

                    printf("请输入电话:\n");
                    scanf("%s",&(ptr->data.name));
                    printf("请输入QQ号:\n");
                    scanf("%s",&(ptr->data.tel));
                    printf("请输入住址:\n");
                    scanf("%s",&(ptr->data.QQ));

}
/*6.2仿尾插法-->后插法(添加信息到指定位置)*/
//首先让指针结点遍历单链表,来找指定ID的位置
//然后创建新结点,为其开辟动态内存并赋值
//最后仿尾插法运用经典代码

//结点指针 p,L;分别用来作为当前遍历的首元结点,总是指向当前的尾结点
//就是用指针p来遍历单链表
//当p为NULL时,即没有找到指定ID,将此时L作为尾结点进行处理
//当p不是NULL,即找到指定ID,将此时p作为尾结点进行处理
void InsertList_Back()
{
        LinkNode p,L,ptr;         //结点指针L伴随着p一起移动
        LinkNode node;  //暂存所指结点后的结点(以防出现单链表中途插入新信息)
        char str[20];
        L = head;
        p = head->next;

        printf("请输入姓名(新信息会插入到该信息前):\n");
        scanf("%s",&str);
        while(p!=NULL && strcmp(p->data.ID,str)!=0)     //逻辑符号的使用
        {
                p = p->next;
                L = L->next;
        }


        if(p == NULL)
        {
          printf("所输入的姓名不存在\n");
           return;
         /*
          L->next = ptr;
          L = ptr;
          ptr = NULL;
        */
        }

                //为新结点开辟内存并赋值
          ptr =(LinkNode)malloc(sizeof(Node));
        //函数InputList()   为指定结点指针-->输入相关信息
           InputList(ptr);


           //暂存 所指结点后的结点(以防出现单链表中途插入新信息)
            node = p->next;

           //经典尾插法的核心代码
           p->next = ptr;
           p = ptr;

           //保证插入后单链表的完整性
           ptr->next = node;
          // ptr = NULL;   //仿尾插法,可能所要插入的ID不在单链表的最后

        save(head);
}


/*7.遍历通讯录*/
void TraverseList()
{
        LinkNode p;
        if (head == NULL)
        {
                printf("通讯录为空!\n");
                return;
        }
        else
        {
                p = head->next;
                printf("通讯录中全部信息如下:\n");
                while (p)
                {
                        PrintList(p);
                        p = p->next;
                }
        }
}

//7.1输出指定信息
void PrintList(LinkNode ptr)
{
    printf("姓名:%-15s\t", ptr->data.ID);
        printf("电话:%-10s\t", ptr->data.name);
        printf("QQ:%-15s\t", ptr->data.tel);
        printf("住址:%-15s\n", ptr->data.QQ);

}
/*8.释放通讯录*/
//需要释放内存后,接着把指针指向NULL
//释放头结点,首元结点和遍历单链表所指向的结点
//先定义两个结点指针 p,t;
//分别用来遍历单链表,暂时指向所遍历之处的结点
void DestroyList()
{
        LinkNode p, t;
        p = head->next;
        while(p!= NULL) //遍历单链表,进行内存释放,数据置空
        {
                t = p;
                free(t);
                t = NULL;
                p = p->next ;
        }
        //遍历完单链表,此时p为空指针
        free(p);
        free(head);
        head = NULL;

        printf("单链表清空成功\n");
         save(head);
}

/* 9.用于将数组中的信息保存到文件中 (n个同学)*/
//调用fwrite()函数 ,将数组中的信息存入文件中
//了解exit()函数 ,用来终止正在执行的程序
void save(LinkNode stu)
{
        FILE *fp;
        int n;
        n = NumList(stu);
        if((fp = fopen("D:\\student1.txt","w")) == NULL)
        {
                printf("打开文件失败\n");
                exit(0);
        }

        fwrite(stu , n*sizeof(Node) , 1,fp); //一个学生信息,用一个结构体数组存储
        fclose(fp);
}
//9.1 通过头指针,遍历单链表进行人数统计并返回
int NumList(LinkNode head)
{
        int num=0;
        LinkNode L;
        L = head->next;
        while(L!=NULL)
        {
           num++;
           L =L->next;
        }

        return num;
}

代码运行截图

在这里插入图片描述

在这里插入图片描述

posted @ 2022-09-21 11:05  hoyeong  阅读(72)  评论(0编辑  收藏  举报