数据结构基础详解(C语言):单链表_定义_初始化_插入_删除_查找_建立操作_纯c语言代码注释讲解

简介: 本文详细介绍了单链表的理论知识,涵盖单链表的定义、优点与缺点,并通过示例代码讲解了单链表的初始化、插入、删除、查找等核心操作。文中还具体分析了按位序插入、指定节点前后插入、按位序删除及按值查找等算法实现,并提供了尾插法和头插法建立单链表的方法,帮助读者深入理解单链表的基本原理与应用技巧。

单链表理论知识详解

1.单链表的定义

线性表的链式存储.
优点:不要求大片连续空间,改变容量方便
缺点:不可随机存取,要耗费一定空间存放指针

typedef struct LNode{
   
    int data;
    struct LNode *next;
}LNode, *LinkList;
AI 代码解读

typedef 取别名
将struct LNode 取别名为别的,方便书写
比如我们要声明一个该结构体的时候
由原先的struct LNode a; 可以直接写为LNode a;
由原先的struct LNode *p; 可以直接写为LinkList a;

2.单链表的初始化

带头结点的初始化,头结点就是多一个结点,指向第一个存放数据的结点.
不带头结点,会使处理数据的逻辑更复杂,对==空表和非空表需要不同的代码逻辑==.
单链表的初始化本质:为头结点分配一个堆空间,将头结点指针域置为空,加上判断内存是否能分配

#include <stdio.h>
#include <stdlib.h>
//这是带有头结点的单链表初始化
void InitList() {
   
    LinkList L;//定义头指针变量 
    L=(LNode*)malloc(sizeof(LNode));//头指针指向分配的头结点内存空间 
    L->next=NULL;
    return true;
}
int main()
{
   
    InitList( );
}
AI 代码解读

3.单链表的插入和删除

3.1 单链表的插入

3.1.1 按位序插入

按位序插入,比如说有5个元素,插入到第三个元素的位置
注意在有头结点时,位序5,意味着是结点6
假如我们要插入的位序是3,意味着我们要寻找的是位序2,也就是结点3,当j=i-1时我们跳出循环,先操作,后j++,j代表当前结点值从0开始,也就是我们在j=3的时候应该跳出循环,所以先操作,后j++,就是j<i-1,j=i的时候就跳出循环

传入什么? 表+插入位置+插入的值
分为几步?
首先是非法操作的判断,是否合法.
第二步是,寻找要插入的位置,插入第几个位置,就找到他前一个位置即i-1,让此时的指针p落在该点处,即我们可以操作他的next域
第三步,先判断吐过p指向空,插入操作不合法,若合法,分配堆空间给一个新的结点s,s的数据域是传入值e,s的指针域指向原先的i(i-1的next域,即p当前的next域),然后将i-1的next域指向新的i

核心思想:先连后断
bool ListInsert(LinkList L,int i,int e)
{
   
    if(i<1)
        return false;
    LNode *p=L;  //为什么需要p指针,因为我们不能动表头指针
    int j=0;    //用来判断当前指针在第几个结点处,j=0,意思是在头结点处
    while(p!=NULL&&j<i-1) //P不能为空,为空咋插入啊,操作不合法,i=j的时候跳出循环
    {
   
        p=p->next; 
        j++; 
    }    //通过这个循环,我们就能找到A的指向B的next域,在他俩中间插入C
    if(p==NULL)
        return false;  //上个循环判断它是否为空,为空不执行,为空的具体操作写在了这,为空就结束
    LNode *s=(LNode *)malloc(sizeof(LNode));
    s->data=e;
    s->next=p->next;
    p->next=s;
    return true; 
}
AI 代码解读

3.1.2 在指定结点的前后插入

一.后插操作

分两步
判断操作是否合法(p指针是否为空+s是否能分配)
插入元素操作

InsertNextNode(LNode *p,int e)
{
   
    if(p==NULL)
        return false;
    LNode *s=(LNode *)malloc(sizeof(LNode));    
    if(s==NULL)
        return false;
    s->data=e;
    s->next=p->next;
    p->next=s;
    return true; 
}
AI 代码解读

二.前插操作

前插操作我们这里不讨论从前遍历一遍,到最后的那种方法
而是考虑,用后插法再交换他们的数据域这种形式可以将时间复杂度降低到o(1)

bool InsertPriorNode(LNode *p,int e)
{
   
    if(p==NULL)
        return false;
    LNode *s=(LNode *)malloc(sizeof(LNode));    
    if(s==NULL)
        return false;
    s->next=p->next;
    p->next=s;
    s->data=p->data;
    p->data=e;
    return true; 
}
AI 代码解读

4.单链表的删除

4.1 按位序删除

第一步与之前的查找的相同的,现查找位序-1的点
然后再进行删除操作

bool ListDelete(LinkList L,int i,int &e)
{
   
    if(i<1)
        return false;
    LNode *p=L;  //为什么需要p指针,因为我们不能动L头指针
    int j=0;  //用来判断当前指针在第几个结点处,j=0,意思是在头结点处
    while(p!=NULL&&j<i-1) //P不能为空,为空咋插入啊,操作不合法,i=j的时候跳出循环
    {
   
        p=p->next; 
        j++; 
    } 
    if(p==NULL) 
        return false;
    if(p->next==NULL) 
        return false;
    LNode *q=p->next;  //方便操作搞出了一个q,直接用p也行,就是写起来不直观
    e=q->data;
    p->next=q->next;
    free(q);
    return true; 
}
AI 代码解读

4.2 指定结点的删除

指定删除结点p,我们思考,给你结点p删除它,需要找到前一个结点,但是那样做太麻烦了,不如交换指定结点和后一个结点的数据域,再删除新的后继结点

bool DeleteNode(LNode *p)
{
   
    if(p==NULL)
        return false;
    LNode *q=p->next;
    p->data=q->data;
    p->next=q->next;
    free(q);
    return true;
}
AI 代码解读

5.单链表的查找

5.1 按位序查找

返回值是位序结点的指针

LNode * GetElem(LinkList L,int i)
{
   
    if(i<0)
        return NULL;
    LNode *p=L;
    int j=0;
    while(p!=NULL&&j<i)
    {
   
        p=p->next;
        j++;
    }
    return p;
 }
AI 代码解读

5.2 按值查找

LNode * LocateElem(LinkList L,int e)
{
   
    LNode *p=L->next; //从第一个结点处开始查值
    while(p!=NULL&&p->data!=e)
    {
   
        p=p->next;
    }
    return p; 
}
AI 代码解读

补充一个:求表的长度

int Length(LinkList L){
   
    int len=0;  //不包括头结点 
    LNode *p=L;
    while(p->next!=NULL)
    {
   
        p=p->next;
        len++;
    }
    return len;
}
AI 代码解读

6. 单链表的建立(带头结点的建立)

单链表的建立包括了头结点的建立(初始化)

6.1 尾插法建立单链表

- 在尾插法中,LNode *s,*r=L;这个写法,其实是为了简化代码,实际上*s不需要赋值,
- 因为在接下来的代码中会给结点s分配堆空间,结点s的位置就会变成随机的,
- 实际上,我们只需要让r=L就行,声明一个s即可
AI 代码解读
  • 声明输入值x,分配头结点,声明s和r指针
  • 循环分配s结点再把它加入链表,再循环的输入x值
  • 链表尾指针置空
LinkList List_Tailnsert(LinkList &L)
{
   
    int x;
    L=(LinkList)malloc(sizeof(LNode)); //初始化头结点
    LNode *s,*r=L;                     //定义上表尾指针和待随机分配的结点指针
    scanf("%d",&x);
    while(x!=9999) //输出9999表示结束
    {
   
        s=(LNode *)malloc(sizeof(LNode));
        s->data=x;
        r->next=s;
        r=s;
        scanf("%d",&x);
     }
     r->next=NULL;
     return L; 
}
AI 代码解读

6.2 头插法建立单链表

  • 头插法相比于尾插法,我们要把头指针置空,因为分配的头指针很可能指向神秘的空间有脏数据
LinkList List_Headlnsert(LinkList L)
{
   
    int x;
    L=(LinkList)malloc(sizeof(LNode));
    L->next=NULL; //初始链表头指针指向NULL
    LNode *s;
    scanf("%d",&x);
    while(x!=9999) //输出9999表示结束
    {
   
        s=(LNode *)malloc(sizeof(LNode));
        s->data=x;
        s->next=L->next;
        L->next=s;
        scanf("%d",&x);
     }
     return L; 
}
AI 代码解读

本文完整代码

#include <stdio.h>
#include <stdlib.h>
typedef struct LNode{
   
    int data;
    struct LNode *next;
}LNode ,*LinkList;

void InitList() {
   
    LinkList L;//定义头指针变量 
    L=(LNode*)malloc(sizeof(LNode));//头指针指向分配的头结点内存空间 
    L->next=NULL;
    return true;
}
bool ListInsert(LinkList L,int i,int e)
{
   
    if(i<1)
        return false;
    LNode *p=L;  //为什么需要p指针,因为我们不能动L头指针
    int j=0;  //用来判断当前指针在第几个结点处,j=0,意思是在头结点处
    while(p!=NULL&&j<i-1) //P不能为空,为空咋插入啊,操作不合法,i=j的时候跳出循环
    {
   
        p=p->next; 
        j++; 
    } 
    if(p==NULL)
        return false;
    LNode *s=(LNode *)malloc(sizeof(LNode));
    s->data=e;
    s->next=p->next;
    p->next=s;
    return true; 
}

InsertNextNode(LNode *p,int e)
{
   
    if(p==NULL)
        return false;
    LNode *s=(LNode *)malloc(sizeof(LNode));    
    if(s==NULL)
        return false;
    s->data=e;
    s->next=p->next;
    p->next=s;
    return true; 
}
bool InsertPriorNode(LNode *p,int e)
{
   
    if(p==NULL)
        return false;
    LNode *s=(LNode *)malloc(sizeof(LNode));    
    if(s==NULL)
        return false;
    s->next=p->next;
    p->next=s;
    s->data=p->data;
    p->data=e;
    return true; 
}

bool ListDelete(LinkList L,int i,int &e)
{
   
    if(i<1)
        return false;
    LNode *p=L;  //为什么需要p指针,因为我们不能动L头指针
    int j=0;  //用来判断当前指针在第几个结点处,j=0,意思是在头结点处
    while(p!=NULL&&j<i-1) //P不能为空,为空咋插入啊,操作不合法,i=j的时候跳出循环
    {
   
        p=p->next; 
        j++; 
    } 
    if(p==NULL) 
        return false;
    if(p->next==NULL) 
        return false;
    LNode *q=p->next;  //方便操作搞出了一个q,直接用p也行,就是写起来不直观
    e=q->data;
    p->next=q->next;
    free(q);
    return true; 
}

bool DeleteNode(LNode *p)
{
   
    if(p==NULL)
        return false;
    LNode *q=p->next;
    p->data=q->data;
    p->next=q->next;
    free(q);
    return true;
}

LNode * GetElem(LinkList L,int i)
{
   
    if(i<0)
        return NULL;
    LNode *p=L;
    int j=0;
    while(p!=NULL&&j<i)
    {
   
        p=p->next;
        j++;
    }
    return p;
} 
LNode * LocateElem(LinkList L,int e)
{
   
    LNode *p=L->next; //从第一个结点处开始查值
    while(p!=NULL&&p->data!=e)
    {
   
        p=p->next;
    }
    return p; 
}

int Length(LinkList L){
   
    int len=0;  //不包括头结点 
    LNode *p=L;
    while(p->next!=NULL)
    {
   
        p=p->next;
        len++;
    }
    return len;
}
LinkList List_Tailnsert(LinkList L)
{
   
    int x;
    L=(LinkList)malloc(sizeof(LNode));
    LNode *s,*r=L;
    scanf("%d",&x);
    while(x!=9999) //输出9999表示结束
    {
   
        s=(LNode *)malloc(sizeof(LNode));
        s->data=x;
        r->next=s;
        r=s;
        scanf("%d",&x);
     }
     r->next=NULL;
     return L; 
}

LinkList List_Headlnsert(LinkList L)
{
   
    int x;
    L=(LinkList)malloc(sizeof(LNode));
    LNode *s;
    scanf("%d",&x);
    while(x!=9999) //输出9999表示结束
    {
   
        s=(LNode *)malloc(sizeof(LNode));
        s->data=x;
        s->next=L->next;
        L->next=s;
        scanf("%d",&x);
     }
     return L; 
}

int main()
{
   
    InitList();
    int e=-1;
}
AI 代码解读
目录
打赏
0
4
6
0
118
分享
相关文章
C语言中的位运算技巧,涵盖基本概念、应用场景、实用技巧及示例代码,并讨论了位运算的性能优势及其与其他数据结构和算法的结合
本文深入解析了C语言中的位运算技巧,涵盖基本概念、应用场景、实用技巧及示例代码,并讨论了位运算的性能优势及其与其他数据结构和算法的结合,旨在帮助读者掌握这一高效的数据处理方法。
107 1
c语言及数据结构实现简单贪吃蛇小游戏
c语言及数据结构实现简单贪吃蛇小游戏
数据结构(C语言)之对归并排序的介绍与理解
归并排序是一种基于分治策略的排序算法,通过递归将数组不断分割为子数组,直到每个子数组仅剩一个元素,再逐步合并这些有序的子数组以得到最终的有序数组。递归版本中,每次分割区间为[left, mid]和[mid+1, right],确保每两个区间内数据有序后进行合并。非递归版本则通过逐步增加gap值(初始为1),先对单个元素排序,再逐步扩大到更大的区间进行合并,直至整个数组有序。归并排序的时间复杂度为O(n*logn),空间复杂度为O(n),且具有稳定性,适用于普通排序及大文件排序场景。
C 语言递归算法:以简洁代码驾驭复杂逻辑
C语言递归算法简介:通过简洁的代码实现复杂的逻辑处理,递归函数自我调用解决分层问题,高效而优雅。适用于树形结构遍历、数学计算等领域。
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面,旨在通过综合策略提升程序性能,满足实际需求。
101 1
|
3月前
|
【数据结构】栈和队列(c语言实现)(附源码)
本文介绍了栈和队列两种数据结构。栈是一种只能在一端进行插入和删除操作的线性表,遵循“先进后出”原则;队列则在一端插入、另一端删除,遵循“先进先出”原则。文章详细讲解了栈和队列的结构定义、方法声明及实现,并提供了完整的代码示例。栈和队列在实际应用中非常广泛,如二叉树的层序遍历和快速排序的非递归实现等。
364 9
|
3月前
|
非递归实现后序遍历时,如何避免栈溢出?
后序遍历的递归实现和非递归实现各有优缺点,在实际应用中需要根据具体的问题需求、二叉树的特点以及性能和空间的限制等因素来选择合适的实现方式。
58 1
|
1月前
|
【C++数据结构——栈与队列】顺序栈的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现顺序栈的基本运算。开始你的任务吧,祝你成功!​ 相关知识 初始化栈 销毁栈 判断栈是否为空 进栈 出栈 取栈顶元素 1.初始化栈 概念:初始化栈是为栈的使用做准备,包括分配内存空间(如果是动态分配)和设置栈的初始状态。栈有顺序栈和链式栈两种常见形式。对于顺序栈,通常需要定义一个数组来存储栈元素,并设置一个变量来记录栈顶位置;对于链式栈,需要定义节点结构,包含数据域和指针域,同时初始化栈顶指针。 示例(顺序栈): 以下是一个简单的顺序栈初始化示例,假设用C语言实现,栈中存储
150 77
☀☀☀☀☀☀☀有关栈和队列应用的oj题讲解☼☼☼☼☼☼☼
### 简介 本文介绍了三种数据结构的实现方法:用两个队列实现栈、用两个栈实现队列以及设计循环队列。具体思路如下: 1. **用两个队列实现栈**: - 插入元素时,选择非空队列进行插入。 - 移除栈顶元素时,将非空队列中的元素依次转移到另一个队列,直到只剩下一个元素,然后弹出该元素。 - 判空条件为两个队列均为空。 2. **用两个栈实现队列**: - 插入元素时,选择非空栈进行插入。 - 移除队首元素时,将非空栈中的元素依次转移到另一个栈,再将这些元素重新放回原栈以保持顺序。 - 判空条件为两个栈均为空。
|
1月前
|
【C++数据结构——栈与队列】环形队列的基本运算(头歌实践教学平台习题)【合集】
【数据结构——栈与队列】环形队列的基本运算(头歌实践教学平台习题)【合集】初始化队列、销毁队列、判断队列是否为空、进队列、出队列等。本关任务:编写一个程序实现环形队列的基本运算。(6)出队列序列:yzopq2*(5)依次进队列元素:opq2*(6)出队列序列:bcdef。(2)依次进队列元素:abc。(5)依次进队列元素:def。(2)依次进队列元素:xyz。开始你的任务吧,祝你成功!(4)出队一个元素a。(4)出队一个元素x。
50 13
【C++数据结构——栈与队列】环形队列的基本运算(头歌实践教学平台习题)【合集】

热门文章

最新文章