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

简介: 返回:贺老师课程教学链接【项目4-链表版通信录】 利用链表存储数据,写一个通信录程序,能够记录多个联系人的编号、姓名、性别、联系电话、地址,完成数据的录入、添加、删除、修改以及查询功能。[参考解答]本解答自网络:链接#include<stdio.h>#include<string.h>#include<stdlib.h&gt

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

【项目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;            //后移一个结点
    }
}
目录
相关文章
|
8月前
|
存储 C语言
【C语言程序设计——函数】递归求斐波那契数列的前n项(头歌实践教学平台习题)【合集】
本关任务是编写递归函数求斐波那契数列的前n项。主要内容包括: 1. **递归的概念**:递归是一种函数直接或间接调用自身的编程技巧,通过“俄罗斯套娃”的方式解决问题。 2. **边界条件的确定**:边界条件是递归停止的条件,确保递归不会无限进行。例如,计算阶乘时,当n为0或1时返回1。 3. **循环控制与跳转语句**:介绍`for`、`while`循环及`break`、`continue`语句的使用方法。 编程要求是在右侧编辑器Begin--End之间补充代码,测试输入分别为3和5,预期输出为斐波那契数列的前几项。通关代码已给出,需确保正确实现递归逻辑并处理好边界条件,以避免栈溢出或结果
358 16
|
8月前
|
算法 C语言
【C语言程序设计——循环程序设计】求解最大公约数(头歌实践教学平台习题)【合集】
采用欧几里得算法(EuclideanAlgorithm)求解两个正整数的最大公约数。的最大公约数,然后检查最大公约数是否大于1。如果是,就返回1,表示。根据提示,在右侧编辑器Begin--End之间的区域内补充必要的代码。作为新的参数传递进去。这个递归过程会不断进行,直到。有除1以外的公约数;变为0,此时就找到了最大公约数。开始你的任务吧,祝你成功!是否为0,如果是,那么。就是最大公约数,直接返回。
201 18
|
8月前
|
Serverless C语言
【C语言程序设计——循环程序设计】利用循环求数值 x 的平方根(头歌实践教学平台习题)【合集】
根据提示在右侧编辑器Begin--End之间的区域内补充必要的代码,求解出数值x的平方根;运用迭代公式,编写一个循环程序,求解出数值x的平方根。注意:不能直接用平方根公式/函数求解本题!开始你的任务吧,祝你成功!​ 相关知识 求平方根的迭代公式 绝对值函数fabs() 循环语句 一、求平方根的迭代公式 1.原理 在C语言中,求一个数的平方根可以使用牛顿迭代法。对于方程(为要求平方根的数),设是的第n次近似值,牛顿迭代公式为。 其基本思想是从一个初始近似值开始,通过不断迭代这个公式,使得越来越接近。
164 18
|
8月前
|
C语言
【C语言程序设计——循环程序设计】统计海军鸣放礼炮声数量(头歌实践教学平台习题)【合集】
有A、B、C三艘军舰同时开始鸣放礼炮各21响。已知A舰每隔5秒1次,B舰每隔6秒放1次,C舰每隔7秒放1次。编程计算观众总共听到几次礼炮声。根据提示,在右侧编辑器Begin--End之间的区域内补充必要的代码。开始你的任务吧,祝你成功!
165 13
|
8月前
|
存储 安全 C语言
【C语言程序设计——选择结构程序设计】预测你的身高(头歌实践教学平台习题)【合集】
分支的语句,这可能不是预期的行为,这种现象被称为“case穿透”,在某些特定情况下可以利用这一特性来简化代码,但在大多数情况下,需要谨慎使用。编写一个程序,该程序需输入个人数据,进而预测其成年后的身高。根据提示,在右侧编辑器补充代码,计算并输出最终预测的身高。分支下的语句,提示用户输入无效。常量的值必须是唯一的,且在同一个。语句的作用至关重要,如果遗漏。开始你的任务吧,祝你成功!,程序将会继续执行下一个。常量都不匹配,就会执行。来确保程序的正确性。
240 10
|
8月前
|
小程序 C语言
【C语言程序设计——基础】顺序结构程序设计(头歌实践教学平台习题)【合集】
目录 任务描述 相关知识 编程要求 测试说明 我的通关代码: 测试结果: 任务描述 相关知识 编程编写一个程序,从键盘输入3个变量的值,例如a=5,b=6,c=7,然后将3个变量的值进行交换,使得a=6,b=7,c=5。面积=sqrt(s(s−a)(s−b)(s−c)),s=(a+b+c)/2。使用输入函数获取半径,格式指示符与数据类型一致,实验一下,不一致会如何。根据提示,在右侧编辑器补充代码,计算并输出圆的周长和面积。
162 10
|
8月前
|
存储 编译器 C语言
【C语言程序设计——函数】分数数列求和2(头歌实践教学平台习题)【合集】
函数首部:按照 C 语言语法,函数的定义首部表明这是一个自定义函数,函数名为fun,它接收一个整型参数n,用于指定要求阶乘的那个数,并且函数的返回值类型为float(在实际中如果阶乘结果数值较大,用float可能会有精度损失,也可以考虑使用double等更合适的数据类型,这里以float为例)。例如:// 函数体代码将放在这里函数体内部变量定义:在函数体中,首先需要定义一些变量来辅助完成阶乘的计算。比如需要定义一个变量(通常为float或double类型,这里假设用float。
196 3
|
8月前
|
存储 C语言
【C语言程序设计——循环程序设计】利用数列的累加和求 sinx(头歌实践教学平台习题)【合集】
项的累加和,一般会使用循环结构,在每次循环中计算出当前项的值(可能基于通项公式或者递推关系),然后累加到一个用于存储累加和的变量中。在C语言中推导数列中的某一项,通常需要依据数列给定的通项公式或者前后项之间的递推关系来实现。例如,对于一个简单的等差数列,其通项公式为。的级数,其每一项之间存在特定的递推关系(后项的分子是其前项的分子乘上。,计算sinx的值,直到最后一项的绝对值小于。为项数),就可以通过代码来计算出指定项的值。对于更复杂的数列,像题目中涉及的用于近似计算。开始你的任务吧,祝你成功!
194 6
|
8月前
|
存储 算法 安全
【C语言程序设计——函数】分数数列求和1(头歌实践教学平台习题)【合集】
if 语句是最基础的形式,当条件为真时执行其内部的语句块;switch 语句则适用于针对一个表达式的多个固定值进行判断,根据表达式的值与各个 case 后的常量值匹配情况,执行相应 case 分支下的语句,直到遇到 break 语句跳出 switch 结构,若没有匹配值则执行 default 分支(可选)。例如,在判断一个数是否大于 10 的场景中,条件表达式为 “num> 10”,这里的 “num” 是程序中的变量,通过比较其值与 10 的大小关系来确定条件的真假。常量的值必须是唯一的,且在同一个。
167 2
|
8月前
|
存储 编译器 C语言
【C语言程序设计——函数】回文数判定(头歌实践教学平台习题)【合集】
算术运算于 C 语言仿若精密 “齿轮组”,驱动着数值处理流程。编写函数求区间[100,500]中所有的回文数,要求每行打印10个数。根据提示在右侧编辑器Begin--End之间的区域内补充必要的代码。如果操作数是浮点数,在 C 语言中是不允许直接进行。的结果是 -1,因为 -7 除以 3 商为 -2,余数为 -1;注意:每一个数据输出格式为 printf("%4d", i);的结果是 1,因为 7 除以 -3 商为 -2,余数为 1。取余运算要求两个操作数必须是整数类型,包括。开始你的任务吧,祝你成功!
141 1