《C语言及程序设计》实践参考——改造链表

简介: 返回:贺老师课程教学链接【项目3 - 改造链表】 下面是一个建立动态链表的程序。阅读程序,然后按要求改造程序。#include <iostream> using namespace std; #include <stdio.h>#include <malloc.h>#define N 5typedef struct

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

【项目3 - 改造链表】
下面是一个建立动态链表的程序。阅读程序,然后按要求改造程序。

#include  <iostream>  
using namespace std;  
#include <stdio.h>
#include <malloc.h>
#define N 5
typedef struct NODE
{
    int data;            //结点的数据
    struct NODE *next;  //指向下一结点
} Node;
Node *head=NULL;    //将链表头定义为全局变量,以便于后面操作
void make_list();   //建立链表
void out_list();    //输出链表

int main( )
{
    make_list();
    out_list();
    return 0;
}
void make_list()
{
    int n;
    Node *p;
    printf("输入若干正数(以0或一个负数结束)建立链表:" );
    scanf("%d", &n);
    while(n>0)   //输入若干正数建立链表,输入非正数时,建立过程结束
    {
        p=(Node*)malloc(sizeof(Node));  //新建结点
        p->data=n;
        p->next=head;  //新建的结点指向原先的链表头
        head=p;    //链表头赋值为新建的节点,这样,新结点总是链表头
        scanf("%d", &n);      //输入下一个数,准备建立下一个结点
    }
    return;
}
void out_list()
{
    Node *p=head;
    printf("链表中的数据为:\n");
    while(p!=NULL)
    {
        printf("%d ", p->data);
        p=p->next;
    }
    printf("\n");
    return;
}

在上面的程序基础上定义下面的函数,实现相应的功能。
为简便起见,每编写一个函数,立刻在main函数中调用进行测试。
(1)编写make_list2()函数建立链表,使建立链表时,后输入的数据,将新输入的数字对应的结点放在链表末尾。若输入为3 5 2 9 4 7 0,建立的链表为:

[参考解答]

#include <stdio.h>
#include <malloc.h>
#define N 5
typedef struct NODE
{
    int data;            //结点的数据
    struct NODE *next;  //指向下一结点
} Node;
Node *head=NULL;
void make_list2();  //新结点始终在链表尾部
void out_list();    //输出链表

int main( )
{
    make_list2();
    out_list();
    return 0;
}
void make_list2()
{
    int n;
    Node *p,*q;  //p用于指向新建立的结点, q指向链表尾部
    printf("输入若干正数(以0或一个负数结束)建立链表:\n");
    scanf("%d", &n);
    while(n>0)
    {
        p=(Node*)malloc(sizeof(Node));
        p->data=n;
        p->next=NULL;
        if(head==NULL)
            head=p;
        else
            q->next=p;
        q=p;
        scanf("%d", &n);
    }
    return;
}
void out_list()
{
    Node *p=head;
    printf("链表中的数据为:\n");
    while(p!=NULL)
    {
        printf("%d ", p->data);
        p=p->next;
    }
    printf("\n");
    return;
}

(2)编写函数void search(int x),输出链表中是否有值为x的结点。
[参考解答]

#include <stdio.h>
#include <malloc.h>
#define N 5
typedef struct NODE
{
    int data;            //结点的数据
    struct NODE *next;  //指向下一结点
} Node;
Node *head=NULL;
void make_list2();  //新结点始终在链表尾部
void out_list();    //输出链表
void search(int x);  //查找是否有值为x的结点

int main( )
{
    int x;
    make_list2();
    out_list();
    scanf("%d", &x);  //测试中输入一个链表中有的数字
    search(x);
    scanf("%d", &x);  //测试中输入一个链表中没有的数字
    search(x);
    return 0;
}

void make_list2()
{
    int n;
    Node *p,*q;  //p用于指向新建立的结点, q指向链表尾部
    printf("输入若干正数(以0或一个负数结束)建立链表:\n");
    scanf("%d", &n);
    while(n>0)
    {
        p=(Node*)malloc(sizeof(Node));
        p->data=n;
        p->next=NULL;
        if(head==NULL)
            head=p;
        else
            q->next=p;
        q=p;
        scanf("%d", &n);
    }
    return;
}

void out_list()
{
    Node *p=head;
    printf("链表中的数据为:\n");
    while(p!=NULL)
    {
        printf("%d ", p->data);
        p=p->next;
    }
    printf("\n");
    return;
}

void search(int x)
{
    //查找是否有值为x的结点
    Node *p=head;
    while(p!=NULL&&p->data!=x)
    {
        p=p->next;
    }
    if(p!=NULL)  //退出上一层循环一定是因为p->data==x
        printf("在链表中有值为%d的结点\n", x);
    else
        printf("在链表中没有值为%d的结点\n", x);
    return;
}

补充:

这个版本的search,只判断是否找到,有兴趣的同学,可以实现能输出有几个的版本
void search(int x)还可以重定义为bool search(int x),让main函数根据返回值输出结果
对应能输出有几个的版本,重定义为int search(int x),返回值是存在的个数,为0没有找到


(3)编写函数delete_first_node(),删除链表中的第一个结点。
[参考解答]

#include <stdio.h>
#include <malloc.h>
#define N 5
typedef struct NODE
{
    int data;            //结点的数据
    struct NODE *next;  //指向下一结点
} Node;
Node *head=NULL;
void make_list2();  //新结点始终在链表尾部
void out_list();    //输出链表
void delete_first_node();  //删除第一个结点

int main( )
{
    make_list2();
    out_list();
    delete_first_node();
    out_list();
    return 0;
}

void make_list2()
{
    int n;
    Node *p,*q;  //p用于指向新建立的结点, q指向链表尾部
    printf("输入若干正数(以0或一个负数结束)建立链表:\n");
    scanf("%d", &n);
    while(n>0)
    {
        p=(Node*)malloc(sizeof(Node));
        p->data=n;
        p->next=NULL;
        if(head==NULL)
            head=p;
        else
            q->next=p;
        q=p;
        scanf("%d", &n);
    }
    return;
}

void out_list()
{
    Node *p=head;
    printf("链表中的数据为:\n");
    while(p!=NULL)
    {
        printf("%d ", p->data);
        p=p->next;
    }
    printf("\n");
    return;
}

void delete_first_node()
{
    Node *p=head;
    if (p!=NULL)
    {
        head = p->next;
        free(p);
        printf("删除了首结点.\n");
    }
    else
    {
        printf("空链表,不能删除.\n");
    }
    return;
}

(4)编写函数delete_node(int x),删除结点值为x的结点。
[参考解答]

#include <stdio.h>
#include <malloc.h>
#define N 5
typedef struct NODE
{
    int data;            //结点的数据
    struct NODE *next;  //指向下一结点
} Node;
Node *head=NULL;
void make_list2();  //新结点始终在链表尾部
void out_list();    //输出链表
void delete_node(int x);  //删除第一个结点

int main( )
{
    make_list2();
    out_list();
    delete_node(3);
    out_list();
    return 0;
}

void make_list2()
{
    int n;
    Node *p,*q;  //p用于指向新建立的结点, q指向链表尾部
    printf("输入若干正数(以0或一个负数结束)建立链表:\n");
    scanf("%d", &n);
    while(n>0)
    {
        p=(Node*)malloc(sizeof(Node));
        p->data=n;
        p->next=NULL;
        if(head==NULL)
            head=p;
        else
            q->next=p;
        q=p;
        scanf("%d", &n);
    }
    return;
}

void out_list()
{
    Node *p=head;
    printf("链表中的数据为:\n");
    while(p!=NULL)
    {
        printf("%d ", p->data);
        p=p->next;
    }
    printf("\n");
    return;
}

void delete_node(int x)
{
    Node *p,*q;
    if (head==NULL)
        printf("空链表,不能删除\n");
    else
    {
        //要删除的恰是首结点(不排除连续有若干个结点值为x),
        while(head!=NULL&&head->data==x)
        {
            p=head;
            head=head->next;
            free(p);
        }
        if(head!=NULL)
        {
            p=head;
            q=p->next;
            while(q!=NULL)
            {
                if(q->data==x)//q就是该删除的结点
                {
                    p->next=q->next;
                    free(q);
                }
                else     //q不该删除,继续考察下一个
                {
                    p=q;
                }
                q=p->next;  //总是p的下一个结点
            }
        }
    }
    return;
}

为充分测试,该程序运行了4次,所用的测试输入数据分别为:
5 2 9 9 7 11 3 0
3 5 2 9 9 7 11 0
3 5 2 9 3 9 7 11 0
3 3 3 3 3 3 3 0


(5)编写make_list3()函数建立链表,使建立链表时,使结点中的数据呈现升序。若输入为3 5 2 9 4 7 0,建立的链表为:

[参考解答]

#include <stdio.h>
#include <malloc.h>
#define N 5
typedef struct NODE
{
    int data;            //结点的数据
    struct NODE *next;  //指向下一结点
} Node;
Node *head=NULL;
void make_list3();  //建立有序链表,各结点由小到大
void out_list();    //输出链表

int main( )
{
    make_list3();
    out_list();
    return 0;
}

void make_list3()
{
    int n;
    Node *t,*p,*q;  //p用于指向新建立的结点, q指向链表尾部
    printf("输入若干正数(以0或一个负数结束)建立链表:\n");
    scanf("%d", &n);
    while(n>0)
    {
        t=(Node*)malloc(sizeof(Node));
        t->data=n;
        t->next=NULL;
        if(head==NULL)   //是空链表,p作为第一个结点即可
            head=t;
        else             //插入p结点后,各结点应该保持有序
        {
            if(n<=head->data)  //新加入的结点应该为首结点
            {
                t->next=head;
                head=t;
            }
            //应该找到合适的位置后,将结点插入
            //此时,链表中至少已经有一个结点,且插入结点不是首结点
            else
            {
                p=head;
                q=p->next;   //p与q相邻,p更靠近q,插入位置将在p和q之间
                while(q!=NULL&&n>q->data)  //链表没有完且p结点比n小,一直往后找
                {
                    p=q;
                    q=p->next;
                }
                if(q==NULL) //q为null,作为最后一个结点直接插入到p后即可
                {
                    p->next = t;
                }
                else   //t插入到p和q之间
                {
                    t->next=q;
                    p->next=t;
                }
            }
        }
        scanf("%d", &n);
    }
    return;
}

void out_list()
{
    Node *p=head;
    printf("链表中的数据为:\n");
    while(p!=NULL)
    {
        printf("%d ", p->data);
        p=p->next;
    }
    printf("\n");
    return;
}

(6)编写函数void insert(int x),将值为x的结点插入到由make_list3建立起来的有序链表中。
[参考解答]

#include <stdio.h>
#include <malloc.h>
#define N 5
typedef struct NODE
{
    int data;            //结点的数据
    struct NODE *next;  //指向下一结点
} Node;
Node *head=NULL;
void make_list3();  //建立有序链表,各结点由小到大
void out_list();    //输出链表
void insert(int x); //将值为x的结点插入到有序链表中,使仍有序

int main( )
{
    make_list3();
    out_list();
    insert(5);
    out_list();
    return 0;
}

void make_list3()
{
    int n;
    Node *t,*p,*q;  //p用于指向新建立的结点, q指向链表尾部
    printf("输入若干正数(以0或一个负数结束)建立链表:\n");
    scanf("%d", &n);
    while(n>0)
    {
        t=(Node*)malloc(sizeof(Node));
        t->data=n;
        t->next=NULL;
        if(head==NULL)   //是空链表,p作为第一个结点即可
            head=t;
        else             //插入p结点后,各结点应该保持有序
        {
            if(n<=head->data)  //新加入的结点应该为首结点
            {
                t->next=head;
                head=t;
            }
            //应该找到合适的位置后,将结点插入
            //此时,链表中至少已经有一个结点,且插入结点不是首结点
            else
            {
                p=head;
                q=p->next;   //p与q相邻,p更靠近q,插入位置将在p和q之间
                while(q!=NULL&&n>q->data)  //链表没有完且p结点比n小,一直往后找
                {
                    p=q;
                    q=p->next;
                }
                if(q==NULL) //q为null,作为最后一个结点直接插入到p后即可
                {
                    p->next = t;
                }
                else   //t插入到p和q之间
                {
                    t->next=q;
                    p->next=t;
                }
            }
        }
        scanf("%d", &n);
    }
    return;
}

void insert(int x) //将值为x的结点插入到有序链表中,使仍有序
{
    Node *t,*p,*q;  //p用于指向新建立的结点, q指向链表尾部
    t=(Node*)malloc(sizeof(Node));;
    t->data=x;
    t->next=NULL;
    if(head==NULL)   //是空链表,p作为第一个结点即可
        head=t;
    else             //插入p结点后,各结点应该保持有序
    {
        if(x<=head->data)  //新加入的结点应该为首结点
        {
            t->next=head;
            head=t;
        }
        //应该找到合适的位置后,将结点插入
        //此时,链表中至少已经有一个结点,且插入结点不是首结点
        else
        {
            p=head;
            q=p->next;   //p与q相邻,p更靠近q,插入位置将在p和q之间
            while(q!=NULL&&x>q->data)  //链表没有完且p结点比n小,一直往后找
            {
                p=q;
                q=p->next;
            }
            if(q==NULL) //q为null,作为最后一个结点直接插入到p后即可
            {
                p->next = t;
            }
            else   //t插入到p和q之间
            {
                t->next=q;
                p->next=t;
            }
        }
    }
    return;
}

void out_list()
{
    Node *p=head;
    printf("链表中的数据为:\n");
    while(p!=NULL)
    {
        printf("%d ", p->data);
        p=p->next;
    }
    printf("\n");
    return;
}

(6)另解
实际上,从程序整体上看,makelist3()可以直接调用insert(int x)实现,这样写出的程序合乎工程上的原则。这启示我们,用函数组织程序结构,应该成为一种意识。

#include <stdio.h>
#include <malloc.h>
#define N 5
typedef struct NODE
{
    int data;            //结点的数据
    struct NODE *next;  //指向下一结点
} Node;
Node *head=NULL;
void make_list3();  //建立有序链表,各结点由小到大
void out_list();    //输出链表
void insert(int x); //将值为x的结点插入到有序链表中,使仍有序

int main( )
{
    make_list3();
    out_list();
    insert(5);
    out_list();
    return 0;
}

void make_list3()
{
    int n;
    printf("输入若干正数(以0或一个负数结束)建立链表:\n");
    scanf("%d", &n);
    while(n>0)
    {
        insert(n);
        scanf("%d", &n);
    }
    return;
}

void insert(int x) //将值为x的结点插入到有序链表中,使仍有序
{
    Node *t,*p,*q;  //p用于指向新建立的结点, q指向链表尾部
    t=(Node*)malloc(sizeof(Node));;
    t->data=x;
    t->next=NULL;
    if(head==NULL)   //是空链表,p作为第一个结点即可
        head=t;
    else             //插入p结点后,各结点应该保持有序
    {
        if(x<=head->data)  //新加入的结点应该为首结点
        {
            t->next=head;
            head=t;
        }
        //应该找到合适的位置后,将结点插入
        //此时,链表中至少已经有一个结点,且插入结点不是首结点
        else
        {
            p=head;
            q=p->next;   //p与q相邻,p更靠近q,插入位置将在p和q之间
            while(q!=NULL&&x>q->data)  //链表没有完且p结点比n小,一直往后找
            {
                p=q;
                q=p->next;
            }
            if(q==NULL) //q为null,作为最后一个结点直接插入到p后即可
            {
                p->next = t;
            }
            else   //t插入到p和q之间
            {
                t->next=q;
                p->next=t;
            }
        }
    }
    return;
}

void out_list()
{
    Node *p=head;
    printf("链表中的数据为:\n");
    while(p!=NULL)
    {
        printf("%d ", p->data);
        p=p->next;
    }
    printf("\n");
    return;
}
目录
相关文章
|
12天前
|
存储 缓存 C语言
C语言:链表和数组有什么区别
C语言中,链表和数组是两种常用的数据结构。数组是一种线性结构,元素在内存中连续存储,通过下标访问,适合随机访问且大小固定的情况。链表由一系列不连续的节点组成,每个节点存储数据和指向下一个节点的指针,适用于频繁插入和删除操作的场景,链表的大小可以动态变化。
|
13天前
|
C语言
无头链表再封装方式实现 (C语言描述)
如何在C语言中实现无头链表的再封装,包括创建节点和链表、插入和删除操作、查找和打印链表以及销毁链表的函数。
22 0
|
13天前
|
C语言
C语言链式结构之有头单链表再封装写法
本文介绍了如何使用C语言对有头单链表进行封装,包括节点的创建、链表的初始化、数据的插入和删除,以及链表的打印等功能。
13 1
|
13天前
|
C语言
C语言结构体链式结构之有头单链表
文章提供了一个C语言实现的有头单链表的完整代码,包括创建链表、插入、删除和打印等基本操作。
17 1
|
13天前
|
测试技术 C语言
单链表之无头链表(C语言版)
本文详细介绍了使用C语言实现无头单链表的方法,包括节点和链表结构的定义、链表的创建与销毁、节点的插入与删除,以及链表的打印等功能。文章通过具体的代码示例,展示了如何在无头链表中进行头插法、尾插法、自定义位置插入和删除,以及如何清空和销毁链表。
17 0
单链表之无头链表(C语言版)
|
13天前
|
C语言
无头链表二级指针方式实现(C语言描述)
本文介绍了如何在C语言中使用二级指针实现无头链表,并提供了创建节点、插入、删除、查找、销毁链表等操作的函数实现,以及一个示例程序来演示这些操作。
15 0
|
1月前
|
存储 C语言
C语言程序设计核心详解 第十章:位运算和c语言文件操作详解_文件操作函数
本文详细介绍了C语言中的位运算和文件操作。位运算包括按位与、或、异或、取反、左移和右移等六种运算符及其复合赋值运算符,每种运算符的功能和应用场景都有具体说明。文件操作部分则涵盖了文件的概念、分类、文件类型指针、文件的打开与关闭、读写操作及当前读写位置的调整等内容,提供了丰富的示例帮助理解。通过对本文的学习,读者可以全面掌握C语言中的位运算和文件处理技术。
|
1月前
|
存储 人工智能 C语言
C语言程序设计核心详解 第八章 指针超详细讲解_指针变量_二维数组指针_指向字符串指针
本文详细讲解了C语言中的指针,包括指针变量的定义与引用、指向数组及字符串的指针变量等。首先介绍了指针变量的基本概念和定义格式,随后通过多个示例展示了如何使用指针变量来操作普通变量、数组和字符串。文章还深入探讨了指向函数的指针变量以及指针数组的概念,并解释了空指针的意义和使用场景。通过丰富的代码示例和图形化展示,帮助读者更好地理解和掌握C语言中的指针知识。
|
1月前
|
存储 C语言
C语言程序设计核心详解 第九章 结构体与链表概要详解
本文档详细介绍了C语言中的结构体与链表。首先,讲解了结构体的定义、初始化及使用方法,并演示了如何通过不同方式定义结构体变量。接着,介绍了指向结构体的指针及其应用,包括结构体变量和结构体数组的指针操作。随后,概述了链表的概念与定义,解释了链表的基本操作如动态分配、插入和删除。最后,简述了共用体类型及其变量定义与引用方法。通过本文档,读者可以全面了解结构体与链表的基础知识及实际应用技巧。
|
1月前
|
存储 算法 C语言
C语言手撕实战代码_循环单链表和循环双链表
本文档详细介绍了用C语言实现循环单链表和循环双链表的相关算法。包括循环单链表的建立、逆转、左移、拆分及合并等操作;以及双链表的建立、遍历、排序和循环双链表的重组。通过具体示例和代码片段,展示了每种算法的实现思路与步骤,帮助读者深入理解并掌握这些数据结构的基本操作方法。