利用二叉树设计同学录管理系统

    采用二叉树存储结构,利用预置数组建立二叉树;实现对通讯录的查找,基于查找实现对同学录的修改和新增成员;求所要操作节点的父节点,从而顺利地编写对同学录的删除操作:

问题的分析:

采用二叉树存储结构,利用预置数组建立二叉树;实现对通讯录的查找,基于查找实现对同学录的修改和新增成员;求所要操作节点的父节点,从而顺利地编写对同学录的删除操作

    
利用二叉树设计同学录管理系统
/*采用二叉树存储结构,
利用预置数组建立二叉树;
实现对通讯录的查找,
基于查找实现对同学录的修改
和新增成员;
求所要操作节点的父节点,
从而顺利地编写对同学录的删除操作。
*/
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#define M 100
#define len sizeof(Student)
/*定义一个学生类型的结构体 sizeof(Student)
所用的空间的大小赋值给变量len*/


typedef struct stu{
    int number;          //号码
    char name[M];        //姓名
    char sex[M];         //性别
    char phone[M];       //电话
    struct stu*left,*right;  //左子树 右子树
}Student;


Student *Stu;


//初始化
void init(){
    Stu==NULL;
}
//添加元素
Student *Add(int num,char n[],char s[],char p[],Student*root){
    if(root==NULL){           // 判断根节点是否有人,没有人则执行插入操作
        root=(Student*)malloc(len);
        root->number=num;
        strcpy(root->name,n);
        strcpy(root->sex,s);
        strcpy(root->phone,p);
        root->left=NULL;
        root->right=NULL;
    }
    else{                                      //如果根结点有人
        if(num<root->number){                  //如果插入的学号比根结点的学号小,则放在左孩子结点
            root->left=Add(num,n,s,p,root->left);
        }                                      //插入的学号比根节点的学号要大,则放在右孩子结点
        else if(num>root->number){
            root->right=Add(num,n,s,p,root->right);
        }
        else if(num=root->number){
            printf("插入结点失败,插入失败的结点是%s\n:",num,root->name);
        }
    }
    return root;              //插入失败时候,返回根结点
}
/*--------------------------------------1:显示原来的预置数组里同学的信息-------*/
//显示前序遍历的结果
void PerOrderTraverse(Student *Ptrl){
    if(Ptrl!=NULL){
        printf("\t%3d\t",Ptrl->number);
        printf("\t%3s\t",Ptrl->name);
        printf("\t%3s\t",Ptrl->sex);
        printf("\t%3s\t",Ptrl->phone);
        printf("\n");
        PerOrderTraverse(Ptrl->left);
        PerOrderTraverse(Ptrl->right);


    }
}
//中序遍历
void InOrderTraverse(Student *Ptrl){
    if(Ptrl!=NULL){
        PerOrderTraverse(Ptrl->left);
        printf("\t%3d\t",Ptrl->number);
        printf("\t%3s\t",Ptrl->name);
        printf("\t%3s\t",Ptrl->sex);
        printf("\t%3s\t",Ptrl->phone);
        printf("\n");
        PerOrderTraverse(Ptrl->right);
    }
}
//后序遍历
void PostOrderTraverse(Student*Ptrl){
    if(Ptrl!=NULL){
        PerOrderTraverse(Ptrl->left);
        PerOrderTraverse(Ptrl->right);
        printf("\t%3d\t",Ptrl->number);
        printf("\t%3s\t",Ptrl->name);
        printf("\t%3s\t",Ptrl->sex);
        printf("\t%3s\t",Ptrl->phone);
    }
}
/*--------------------------------------2:查找同学的信息--------------------*/
void SearchInformationByNum(Student *T,int num){        //按照学号进行查找
    if (T != NULL)
    {
        SearchInformationByNum(T->left,num);
        if(T->number==num)
        {
            printf("\t%d\t", T->number);
            printf("%s\t", T->name);
            printf("%s\t", T->sex);
            printf("%s\n", T->phone);
        }
        SearchInformationByNum(T->right,num);
    }
}


void SearchInformationByName(Student* T,char n[])    //按照名字查找 遍历查找
{
    if (T != NULL)
    {
        SearchInformationByName(T->left,n);
        /*strcmp函数比较两个字符串,
        str1<str2 返回的值是0
        str=str2  返回的值是负数
        str1>str2 返回的值是正数
        */
        if(strcmp(T->name,n) == 0)
        {
            printf("\t%d\t", T->number);
            printf("%s\t", T->name);
            printf("%s\t", T->sex);
            printf("%s\n", T->phone);
        }
        SearchInformationByName(T->right,n);
    }
}




void SearchInformationBySex(Student*T,char s[]){     //按照性别进行查找
 if (T != NULL)
    {
        SearchInformationBySex(T->left,s);
    if(strcmp(T->sex,s) == 0){
        printf("\t%d\t", T->number);
        printf("%s\t", T->name);
        printf("%s\t", T->sex);
        printf("%s\n", T->phone);
    }
        SearchInformationBySex(T->right,s);
    }
}


void SearchInformationByPhone(Student*T,char p[]){    //按照电话号码进行查找
    if(T!=NULL){
        SearchInformationByPhone(T->left,p);
         if(strcmp(T->phone,p)==0){
          printf("\t%d\t",T->number);
          printf("\t%s\t",T->name);
          printf("\t%s\t",T->sex);
          printf("\t%s\n",T->phone);
         }
         SearchInformationByPhone(T->right,p);
}
}


Student *FindMin(Student *s){
    if(!s){                        //空的二叉树,返回NULL
        return NULL;
    }
    else{
        if(!s->left)
         return s;                  //找到最左叶的结点并返回
        else{
         return FindMin(s->left);     //沿着左分支继续查找
        }
}
}


/*---------------------------3:修改同学的信息------------------------------------------*/
void ModifyInformationByNum(Student *T,int num){
 int  k;
    if (T != NULL){
    ModifyInformationByNum(T->left,num);
    if(T->number==num){
    printf("\t%d\t", T->number);
    printf("%s\t", T->name);
    printf("%s\t", T->sex);
    printf("%s\n", T->phone);
   while(1){
   	printf("\n\n");
    printf(" 1:    姓名\n");
    printf(" 2:    性别\n");
    printf(" 3:    电话\n");
    printf("显示修改后的信息并且退出本次的循环\n");
	scanf("%d",&k);
    switch(k){
      case 1:{
      char name[M];
          printf("请输入名字:");
          scanf("%s",&name);
          strcpy(T->name,name);
      }break;
      case 2:{
      char sex[M];
         printf("请输入性别:");
         scanf("%s",&sex);
         strcpy(T->sex,sex);
        }break;
      case 3:{
        char phone[M];
        printf("请输入新号码:");
        scanf("%s",phone);
        strcpy(T->phone,phone);
        }break;
      case 0:{
        printf("\t学号\t姓名\t性别\t电话\n");
        printf("\t%d\t",T->number);
        printf("%s\t",T->name);
        printf("%s\t",T->sex);
        printf("%s\n",T->phone);
        printf("\n");
        goto temp;
        }
        default:break;
       }
        }
        }
        temp: ModifyInformationByNum(T->right,num);
    }
}


void ModifyInformationByName(Student *T,char n[]){
    int  k;
    if (T != NULL){
    ModifyInformationByName(T->left,n);
    if(strcmp(T->name,n) == 0){
    printf("\t%d\t", T->number);
    printf("%s\t", T->name);
    printf("%s\t", T->sex);
    printf("%s\n", T->phone);
   while(1){
   	printf("\n\n");
    printf(" 1:                姓名\n");
    printf(" 2:                性别\n");
    printf(" 3:                电话\n");
    printf("显示修改后的信息并且退出本次的循环\n");
    scanf("%d",&k);
    switch(k){
      case 1:{
      char n[M];
          printf("请输入名字:");
          scanf("%s",&n);
          strcpy(T->name,n);
      }break;
      case 2:{
      char sex[M];
         printf("请输入性别:");
         scanf("%s",&sex);
         strcpy(T->sex,sex);
        }break;
      case 3:{
        char phone[M];
        printf("请输入新号码:");
        scanf("%s",phone);
        strcpy(T->phone,phone);
        }break;
      case 0:{
        printf("\t学号\t姓名\t性别\t电话\n");
        printf("\t%d\t",T->number);
        printf("%s\t",T->name);
        printf("%s\t",T->sex);
        printf("%s\n",T->phone);
        printf("\n");
        goto temp;
        }
        default:break;
       }
        }
        }
        temp: ModifyInformationByName(T->right,n);
    }
}


void ModifyInformationBySex(Student *T,char sex[]){
 int  k;
    if (T != NULL){
    ModifyInformationBySex(T->left,sex);
    if(strcmp(T->sex,sex) == 0){
    printf("\t%d\t", T->number);
    printf("%s\t", T->name);
    printf("%s\t", T->sex);
    printf("%s\n", T->phone);
   while(1){
   	printf("\n\n");
    printf(" 1:                姓名\n");
    printf(" 2:                性别\n");
    printf(" 3:                电话\n");
    printf("显示修改后的信息并且退出本次的循环\n");
    scanf("%d",&k);
    switch(k){
      case 1:{
      char name[M];
          printf("请输入名字:");
          scanf("%s",&name);
          strcpy(T->name,name);
      }break;
      case 2:{
      char sex[M];
         printf("请输入性别:");
         scanf("%s",&sex);
         strcpy(T->sex,sex);
        }break;
      case 3:{
        char phone[M];
        printf("请输入新号码:");
        scanf("%s",phone);
        strcpy(T->phone,phone);
        }break;
      case 0:{
        printf("\t学号\t姓名\t性别\t电话\n");
        printf("\t%d\t",T->number);
        printf("%s\t",T->name);
        printf("%s\t",T->sex);
        printf("%s\n",T->phone);
        printf("\n");
        goto temp;
        }
        default:break;
       }
        }
        }
        temp: ModifyInformationBySex(T->right,sex);
    }
}




void ModifyInformationByPhone(Student *T,char p[]){
 int  k;
    if (T != NULL){
    ModifyInformationByPhone(T->left,p);
    if(strcmp(T->phone,p) == 0){
    printf("\t%d\t", T->number);
    printf("%s\t", T->name);
    printf("%s\t", T->sex);
    printf("%s\n", T->phone);
   while(1){
    printf("\n(\t1:姓名,\t2:性别 \t3:联系电话 \t 0:显示修改后的信息并退出本次修改\n");
    scanf("%d",&k);
    switch(k){
      case 1:{
      char name[M];
          printf("请输入名字:");
          scanf("%s",&name);
          strcpy(T->name,name);
      }break;
      case 2:{
      char sex[M];
         printf("请输入性别:");
         scanf("%s",&sex);
         strcpy(T->sex,sex);
        }break;
      case 3:{
        char phone[M];
        printf("请输入新号码:");
        scanf("%s",phone);
        strcpy(T->phone,phone);
        }break;
      case 0:{
        printf("\t学号\t姓名\t性别\t电话\n");
        printf("\t%d\t",T->number);
        printf("%s\t",T->name);
        printf("%s\t",T->sex);
        printf("%s\n",T->phone);
        printf("\n");
        goto temp;
        }
        default:break;
       }
        }
        }
        temp: ModifyInformationByPhone(T->right,p);
    }
}
/*------------------------------删除操作----------------------------------------------*/
void DeleteInformationByName(Student *T,Student *PRT, char n[]){    //采用递归的方式进行遍历删除
        if(T!=NULL){                                                //当遍历的根结点不为空指针时
            DeleteInformationByName(T->left,T,n);
            if(strcmp(T->name,n)==0){
                if(T->left&&T->right){                              //同时拥有左子树和右子树
                Student *student = FindMin(T->left);
                T->number= student->number;
                strcpy(T->name,student->name);
                strcpy(T->sex,student->sex);
                strcpy(T->phone,student->phone);
                DeleteInformationByName(T->right,T,student->name);
                }
                else{
                if(PRT == NULL){
                    if(T->left !=NULL){
                        T = T->left;
                    }
                    else{
                        T = T->right;
                     }
                    }
                else{
                    if(T == PRT->left){
						if(T->left != NULL){     //有右孩子结点无子结点
                    PRT->left = T->left;
                    }
                else{
                    PRT->left = T->right;
                }
                }
                else{
                if(T->left!= NULL){
                    PRT->right = T->left;
                }
                else{
                PRT->right = T->right;
                }
                }
            }
        }
    }
    DeleteInformationByName(T->right,T,n);
    }
}


void DeleteInformationByPhone(Student *T,Student *PRT, char p[]){     //采用递归的方式进行遍历删除
        if(T!=NULL){                                   //当遍历的根结点不为空指针时
            DeleteInformationByPhone(T->left,T,p);
            if(strcmp(T->phone,p)==0){
                if(T->left&&T->right){                  //同时拥有左子树和右子树
                Student *student = FindMin(T->left);
                T->number= student->number;
                strcpy(T->name,student->name);
                strcpy(T->sex,student->sex);
                strcpy(T->phone,student->phone);
                DeleteInformationByPhone(T->right,T,student->phone);
                }
                else{
                if(PRT == NULL){
                    if(T->left !=NULL){
                        T = T->left;
                    }
                    else{
                        T = T->right;
                     }
                    }
                else{
                    if(T == PRT->left){
                    if(T->left != NULL){
                    PRT->left = T->left;
                    }
                else{
                    PRT->left = T->right;
                }
                }
                else{
                if(T->left!= NULL){
                    PRT->right = T->left;
                }
                else{
                PRT->right = T->right;
                }
                }
            }
        }
    }
    DeleteInformationByPhone(T->right,T,p);
    }
}


/*----------------------------------视图层---------------------------------*/
Student *menu(Student *Stu){
        printf("\t                 同学录管理系统\n");
        printf("   ---------------------------------------------------------\n");
        printf("\t                1:显示原来的预置数组里同学的信息       \n\n");
        printf("\t                2:查找同学的信息                       \n\n");
        printf("\t                3:修改同学录信息                       \n\n");
        printf("\t                4:添加新同学信息                       \n\n");
        printf("\t                5:删除同学录信息                       \n\n");
        printf("\t                6:退出该系统\n\n");
        printf("    ----------------------------------------------------------\n");
        printf("\t选择您要进行的操作前的序号(1~6):");
}
Student *showInformation(Student *Stu){
        printf("\n\n");
        printf("\t                  显示原来的预置数组里同学的信息 \n");
        printf("    ----------------------------------------------------------\n");
        printf("\t               (1)先序遍历预置数组里同学的信息\n");printf("\n");
        printf("\t               (2)中序遍历预置数组里同学的信息\n");printf("\n");
        printf("\t               (3)后序遍历预置数组里同学的信息\n");printf("\n");
        printf("\t               (0)返回主菜单\n");
        printf("    ----------------------------------------------------------\n");


}
Student *SearchInformation(Student *Stu){
        printf("\n\n");
        printf("\t                 查找同学的信息 \n");
        printf("    ----------------------------------------------------------\n");
        printf("\t               (1)按照学号查找同学的信息\n");printf("\n");
        printf("\t               (2)按照姓名查找同学的信息\n");printf("\n");
        printf("\t               (3)按照性别查找同学的信息\n");printf("\n");
        printf("\t               (4)按照电话查找同学的信息\n");printf("\n");
        printf("\t               (0)返回主菜单\n");
        printf("    ----------------------------------------------------------\n");


}
Student *ModifyInformation(Student *Stu){
        printf("\n\n");
        printf("\t                  修改同学的信息 \n");
        printf("    ----------------------------------------------------------\n");
        printf("\t               (1)根据学号做为索引修改同学的信息\n");printf("\n");
        printf("\t               (2)根据姓名作为索引修改同学的信息\n");printf("\n");
        printf("\t               (3)根据电话做为索引修改同学的信息\n");printf("\n");
        printf("\t               (0)返回主菜单\n");
        printf("    ----------------------------------------------------------\n");


}
Student *DeleteInformation(Student *Stu){
        printf("\n\n");
        printf("\t                  删除同学的信息 \n");
        printf("    ----------------------------------------------------------\n");
        printf("\t               (1)根据姓名做为索引删除同学的信息\n");printf("\n");
        printf("\t               (2)根据电话删做为索引除同学的信息\n");printf("\n");
        printf("\t               (0)返回主菜单\n");
        printf("    ----------------------------------------------------------\n");
}


//打印函数
void print(Student*Stu){
    printf("\t学号\t\t姓名\t\t性别\t\t电话\n");
}




//测试函数
int main(){
    int a;
    void init();
    if(Stu==NULL){
       // printf("初始化工作完成\n\n");
    }
    //预置数组的信息
    Stu=Add(1,"Bob","male","15538569845",Stu);
    Stu=Add(2,"lisi","female","13846825852",Stu);
    Stu=Add(3,"yang","male","18837235685",Stu);
    l:
    while(1)
    {
        int n;
        menu(Stu);                  //主菜单
        scanf("%d",&n);
    switch(n)
        {
        case 1:{                  //显示操作
        system("cls");
		showInformation(Stu);
      while(1)
      {
        printf("\t请输入你的选择(0~3):");
        scanf(" %d",&a);
        switch(a){
        case 1:print(Stu);PerOrderTraverse(Stu);break;  //先序遍历二叉树
        case 2:print(Stu);InOrderTraverse(Stu);break;   //中序遍历二叉树
        case 3:print(Stu);PostOrderTraverse(Stu);break; //后序遍历二叉树
        case 0:goto l;break;
        default :break;
        }
       }
		  }break;


        case 2:{
        system("cls");
        SearchInformation(Stu);                       //2:查找菜单函数
      while(1)
      {
        printf("请输入你的选择(0-4):");
        scanf(" %d",&a);
        switch(a){


        case 1:{
         int num;
         printf("请输入要查找的同学学号:");
         scanf("%d",&num);
         print(Stu);
         SearchInformationByNum(Stu,num);        //按照学号进行查找
         printf("\n");
        }break;
        case 2:{
         char name[M];
         printf("请输入要查找的同学名字:");       //按照同学的姓名进行查找
         scanf("%s",&name);
         print(Stu);
         SearchInformationByName(Stu,name);
         printf("\n");
        }break;
        case 3:{
            char sex[M];
            Student *ptr=(Student*)malloc(len);
            printf("请输入要查找的同学的性别:");      //按照性别进行查找
            scanf("%s",&sex);
            print(Stu);
            SearchInformationBySex(Stu,sex);
            printf("\n");
        }break;
        case 4:{
            char phone[M];
            printf("请输入要查找的同学的电话:");
            scanf("%s",&phone);
            print(Stu);
            SearchInformationByPhone(Stu,phone);
            printf("\n");
        }break;
        case 0:goto l;break;
        default :break;
        }
       }
       }break;


      //修改操作
       case 3:{
       	system("cls");
        ModifyInformation(Stu);            //修改操作
      while(1)
      {
        printf("\t请输入你的选择(0~4):");
        scanf("%d",&a);
        switch(a){
        case 1:{
         int num;
         printf("\t请输入要修改的同学的学号:");     //按照同学的姓名进行查找
         scanf("%d",&num);
         print(Stu);
         ModifyInformationByNum(Stu,num);
         printf("\n");
        }break;


        case 2:{
         char name[M];
         printf("\t请输入要修改的同学的名字:");     //按照同学的姓名进行修改
         scanf("%s",&name);
         //Student *ptr = (Student *)malloc(len);
         print(Stu);
         ModifyInformationByName(Stu,name);
         printf("\n");
        }break;
       case 3:{
         char sex[M];
         printf("\t请输入要修改的同学的性别:");     //按照同学的姓名进行修改
         scanf("%s",&sex);
         //Student *ptr = (Student *)malloc(len);
         print(Stu);
         ModifyInformationByName(Stu,sex);
         printf("\n");
        }break;
      case 4:{
            char phone[M];
            //Student *ptr=(Student*)malloc(len);
            printf("\t请输入要修改的同学的电话号码:");   //按照性别进行修改
            scanf("%s",&phone);
            print(Stu);
            ModifyInformationByPhone(Stu,phone);
            printf("\n");
        }break;
        case 0:goto l;break;
        default :break;
        }
       }
       }break;


    //4:添加新的结点信息
        case 4:{
        system("cls");
        int  addnumber;
        char addname[M];
        char addsex[M];
        char addphone[M];
        printf("\t ---------------添加操作!---------\n\n");
        printf("请依次录入同学的\n\t学号\t姓名\t性别\t电话\n");
        scanf("%d",&addnumber);
        scanf("%s",&addname);
        scanf("%s",&addsex);
        scanf("%s",&addphone);
        Stu=Add(addnumber,addname,addsex,addphone,Stu);
        }break;


    //删除操作
        case 5:{
       	system("cls");
        DeleteInformation(Stu);          //删除操作
      while(1)
      {
        printf("请输入你的选择(0~2):");
        scanf("%d",&a);
        switch(a){
        case 1:{
            char name[M];
            Student *ptr=(Student*)malloc(len);
            printf("请输入要删除的同学的姓名:");   //按照姓名进行删除
            scanf("%s",&name);
            DeleteInformationByName(Stu,NULL,name);
            printf("\n");
        }break;
        case 2:{
            char phone[M];
            Student *ptr=(Student*)malloc(len);
            printf("请输入要删除的同学的电话:");   //按照姓名进行删除
            scanf("%s",&phone);
            DeleteInformationByPhone(Stu,NULL,phone);
            printf("\n");
        }break;
        case 0:goto l;break;
        default :break;
        }
       }
       }break;
    }
}
}




 
 
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值