《C语言及程序设计》实践参考——链表版通信录

返回:贺老师课程教学链接

【项目4-链表版通信录】
利用链表存储数据,写一个通信录程序,能够记录多个联系人的编号、姓名、性别、联系电话、地址,完成数据的录入、添加、删除、修改以及查询功能。

[参考解答]

本解答自网络:链接

#include<stdio.h>
#include<string.h>
#include<stdlib.h>
typedef struct             //通讯录结点类型
{
    char num[5];        //编号
    char name[9];        //姓名
    char sex[3];        //性别
    char phone[13];        //电话
    char addr[31];        //地址
} DataType;

typedef struct node     //结点类型定义
{
    DataType data;        //结点数据域
    struct node * next;    //结点指针域
} ListNode;

typedef ListNode * LinkList;
LinkList head;
ListNode *p;

//函数说明
int menu_select();
LinkList CreateList();
void InsertNode(LinkList head,ListNode *p);
ListNode *ListFind(LinkList head);
void DelNode(LinkList head);
void PrintList(LinkList head);

//主函数
int main()
{
    for(;;)
    {
        switch(menu_select())
        {
        case 1:
            printf("****************************************\n");
            printf("*          通讯录链表的建立            *\n");
            printf("****************************************\n");
            head=CreateList();
            break;
        case 2:
            printf("****************************************\n");
            printf("*            通讯者信息的添加          *\n");
            printf("****************************************\n");
            printf("*编号(4) 姓名(8) 性别 电话(11) 地址(31)*\n");
            printf("****************************************\n");
            p=(ListNode *)malloc(sizeof(ListNode));    //申请新结点
            scanf("%s%s%s%s%s",p->data.num,p->data.name,
                  p->data.sex,p->data.phone,p->data.addr);
            InsertNode(head,p);
            break;
        case 3:
            printf("****************************************\n");
            printf("*          通讯录信息的查询            *\n");
            printf("****************************************\n");
            p=ListFind(head);
            if(p!=NULL)
            {
                printf("*编 号  姓 名  性 别  联系电话  地 址  *\n");
                printf("----------------------------------------\n");
                printf("%s\t%s\t%s\t%s\t%s\n",p->data.num,p->data.name,
                       p->data.sex,p->data.phone,p->data.addr);
                printf("----------------------------------------\n");
            }
            else
                printf("没有查到要查询的通读者!\n");
            break;
        case 4:
            printf("****************************************\n");
            printf("*          通讯录信息的删除            *\n");
            printf("****************************************\n");
            DelNode(head);    //删除结点
            break;
        case 5:
            printf("****************************************\n");
            printf("*          通讯录链表的输出            *\n");
            printf("****************************************\n");
            PrintList(head);
            break;
        case 0:
            printf("\t 再  见! \n");
            return 0;
        }
    }
}

/**********************************/
/*       菜单选择函数程序         */
/**********************************/
int menu_select()
{
    int sn;
    printf("==============================\n");
    printf("   通讯录管理系统\n");
    printf("==============================\n");
    printf("   1.通讯录链表的建立\n");
    printf("   2.通讯者结点的插入\n");
    printf("   3.通讯者结点的查询\n");
    printf("   4.通讯者结点的删除\n");
    printf("   5.通讯录链表的输出\n");
    printf("   0.退出管理系统\n");
    printf("==============================\n");
    for(;;)
    {
        scanf(" %d",&sn);
        if(sn<0||sn>5)
            printf("\n\t输入错误,重选0-5");
        else
            break;
    }
    return sn;
}

/***********************************/
/*  用尾插法建立通讯录链表函数     */
/***********************************/
LinkList CreateList()
{
    //尾插法建立带头结点的通讯录链表算法
    LinkList head=(ListNode*)malloc(sizeof(ListNode)); //申请表头结点
    ListNode *p,*rear;
    int flag=0;        //结束标志置0
    rear=head;         //尾指针初始指向头结点
    while(flag==0)
    {
        p=(ListNode*)malloc(sizeof(ListNode));    //申请新结点
        printf("编号(4) 姓名(8) 性别  电话(11)  地址(31) \n");
        printf("-----------------------------------------\n");
        scanf(" %s%s%s%s%s",p->data.num,p->data.name,p->data.sex,
              p->data.phone,p->data.addr);
        rear->next=p;            //新结点连接到尾结点之后
        rear=p;                    //尾指针指向新结点
        printf("结束建表吗? (1结束/0继续):");
        scanf(" %d",&flag);        //读入一个标志数据
    }
    rear->next=NULL;            //终端结点指针域置空
    return head;
}

/*****************************************/
/*   在通讯录(顺序)链表head中插入结点    */
/*****************************************/
void InsertNode(LinkList head,ListNode *p)
{
    ListNode *p1,*p2;
    p1=head;
    p2=p1->next;
    while(p2!=NULL && strcmp(p2->data.num,p->data.num)<0)
    {
        p1=p2;               //p1指向刚访问过的结点
        p2=p2->next;         //p2指向表的下一个结点
    }
    p1->next=p;               //插入p所指向的结点
    p->next=p2;               //连接表中剩余部分
}

/*****************************************/
/*        有序通讯录链表上的查找         */
/*****************************************/
ListNode * ListFind(LinkList head)
{
    //有序通讯录链表上的查找
    ListNode *p;
    char num[5];
    char name[9];
    int xz;
    printf("=================\n");
    printf("  1.按编号查询   \n");
    printf("  2.按姓名查询   \n");
    printf("=================\n");
    printf("请选择:");
    p=head->next;    //假定通讯录表带关结点
    scanf("%d",&xz);
    fflush(stdin);
    if(xz==1)
    {
        printf("请输入要查找者的编号:");
        scanf("%s",num);
        fflush(stdin);
        while(p&&strcmp(p->data.num,num)<0)
        {
            p=p->next;
        }
        if(p==NULL||strcmpi(p->data.num,num)>0)
        {
            p=NULL;        //没有查到要查找的通讯者
        }
    }
    else
    {
        if(xz==2)
            printf("请输入要查找者的姓名:");
        scanf("%s",name);
        fflush(stdin);
        while(p&&strcmp(p->data.name,name)!=0)
            p=p->next;
    }
    return p;
}

/*****************************************/
/*        通讯录链表上结点的删除         */
/*****************************************/
void DelNode(LinkList head)
{
    char jx;
    ListNode *p,*q;
    p=ListFind(head);    //调用查找函数
    if(p==NULL)
    {
        printf("没有查到要删除的通讯者");
        return;
    }
    printf("真的要删除该结点吗?(y/n):");
    scanf("%c",&jx);    //注意在%c前加上一空格可以处理掉输入缓冲区
    fflush(stdin);        //用这个函数也可以清除输入缓冲区
    if(jx=='y'||jx=='Y')
    {
        q=head;
        while(q!=NULL && q->next!=p)
            q=q->next;
        q->next=p->next;    //删除结点
        free(p);            //释放被删除的结点空间
        printf("通讯者已被删除\n");
    }
}

/*****************************************/
/*        通讯录链表上输出函数         */
/*****************************************/
void PrintList(LinkList head)
{
    ListNode *p;
    p=head->next;            //因为链表带头结点,使p指向链表开始结点
    printf("编号  姓名  性别  联系电话  地址\n");
    printf("------------------------------------------\n");
    while(p!=NULL)
    {
        printf("%s%s%s%s%s\n",p->data.num,p->data.name,
               p->data.sex,p->data.phone,p->data.addr);
        printf("------------------------------------------\n");
        p=p->next;            //后移一个结点
    }
}
时间: 2024-10-26 17:19:14

《C语言及程序设计》实践参考——链表版通信录的相关文章

《C++语言基础》实践参考—— 链表类

返回:贺老师课程教学链接  项目要求 [项目 - 链表类]动态链表也是程序设计中的一种非常有用的数据结构.可以说,是否能够理解有关操作的原理,决定了你是否有资格称为"科班"出身.在后续的专业基础课中,相关的内容还会从不同的角度,反复地认识,反复地实践.不过,在现阶段多些体验,也是很有必要的了.(1)阅读下面的程序,回顾一下动态链表,阅读程序过程中,请用笔画一画形成链表的过程中指针值的变化. #include <iostream> using namespace std; s

《C语言及程序设计》实践参考——函数版星号图

返回:贺老师课程教学链接  项目要求 [项目1-函数版星号图]这一组的练习意在通过调用函数输出星号图,体会与理解函数的工作过程,并为其后编制自定义函数实现特定功能.(1)补充完下面的程序,使程序输出星号下面的星号图: #include <stdio.h> void printstars(int m) //定义能输出一行m个星号的函数 { int j; for (j=1; j<=m; ++j) printf("*"); } int main( ) { int n=6;

《C语言及程序设计》实践参考——链表的合并

返回:贺老师课程教学链接 [项目1-链表的合并] 输入一个整数m,表示A链表的长度,再输入m个数作为A链表中的m个数据元素,建立链表A,其头指针为heada.输入一个整数n,表示B链表的长度,再输入n个数表示B链表中的n个数据元素,建立链表B,其头指针为headb.输入i.len.j,将要从单链表A中删除自第i个元素起的共len个元素,然后将单链表A插入到单链表B的第j个元素之前.最后输出操作后的链表B. 例如,输入: 11 13 5 14 62 3 43 71 5 72 34 5 (11个数构

《C++语言基础》实践项目——链表类及其应用

返回:贺老师课程教学链接 [项目 - 链表类]动态链表也是程序设计中的一种非常有用的数据结构.可以说,是否能够理解有关操作的原理,决定了你是否有资格称为"科班"出身.在后续的专业基础课中,相关的内容还会从不同的角度,反复地认识,反复地实践.不过,在现阶段多些体验,也是很有必要的了.(1)阅读下面的程序,回顾一下动态链表,阅读程序过程中,请用笔画一画形成链表的过程中指针值的变化. #include <iostream> using namespace std; struct

C++实践参考——OOP版电子词典

[项目-OOP版电子词典] 做一个简单的电子词典.在文件dictionary.txt中,保存的是英汉对照的一个词典,词汇量近8000个,英文.中文释义与词性间用'\t'隔开.(1)编程序,由用户输入英文词,显示词性和中文释义.提示1:如果要用OOP完成这个词典(当然也可以用OO方法实现),可以定义一个Word类表示一个词条,其中的数据成员string english; 表示英文单词,string chinese;表示对应中文意思,string word_class;表示该词的词性:还可以定义一个

《C++语言基础》实践参考——Josephus(约瑟夫环)问题

返回:贺老师课程教学链接  项目要求 [项目-Josephus(约瑟夫环)问题]n个小孩子围成一圈,从第一个小孩子开始顺时针方向数数字,到第m个小孩子离开,这样反反复复,最终只剩下一个小孩子,求第几个小孩子留下?    提示:约瑟夫环即是一个首尾相连的链表,在建立好这个环以后,从头结点开始,每次间隔m孩子删除一个结点,直至只余下一个结点(删除了n-1个).     参考下面的代码,也可以自行设计类. //链表结点kid,其中number为这个人的编号 struct kid { int numbe

《C++语言基础》实践参考——旱冰场造价

返回:贺老师课程教学链接 [项目1 - 旱冰场造价] 有一个圆形的旱冰场地,场地内抹水泥,造价为每平方米M元,围栏用木条围成,每米造价N元,输入旱冰场半径,用程序计算出其造价. 一级提示: 先定义类,确定其数据成员和成员函数.在这里要定义的旱冰场类,实际我们只关心其面积和周长,根据面积计算场地内抹水泥的造价,根据周长计算围栏的造价.有了类的定义,在main函数中定义该类对象,调用成员函数即可以完成求解.二级提示: 实际上,对于旱冰场,我们只关心它就是一个圆.这就是设计中抓住了本质.于是,设计一个

《C++语言基础》实践参考——指向学生类的指针

返回:贺老师课程教学链接 [项目4-指向学生类的指针] 设计一个学生类Student,数据成员包括学号(num)和成绩(score),成员函数根据需要自行设计(建议配备需要的set.get函数,以及必要的输入或输出,给出的代码中也可以找到需要成员函数的线索).在main函数中,要做到: 建立一个对象数组,通过初始化,设置5个学生的数据,要求: 用指针指向数组首元素,输出第1.3.5个学生的信息: 设计一个函数int max(Student *arr);,用指向对象的指针作函数参数,在max函数中

《C++语言基础》实践参考——复数模板类

返回:贺老师课程教学链接 [项目6-复数模板类]    阅读教材例10.1.该例实现了一个复数类,但是美中不足的是,复数类的实部和虚部都固定只能是double型的.可以通过模板类的技术手段,设计Complex,使实部和虚部的类型为定义对象时指定的实际类型.    (1)要求类成员函数在类外定义.    (2)在此基础上,再实现减法.乘法和除法    你可以使用的main()函数如下. int main( ) { Complex<int> c1(3,4),c2(5,-10),c3; //实部和虚