文章目录

  • 前言
  • 一、链表
    • 1、链表的概念及结构
    • 2、链表的分类
    • 3、最常用的两种链表
  • 二、单链表的实现
    • 1、结构的定义
    • 2、创建新节点
    • 3、在头部插入数据
    • 4、在尾部插入数据
    • 5、查找数据
    • 6、在pos位置前插入数据
    • 7、在pos位置后插入数据
    • 8、在头部删除数据
    • 9、在尾部删除数据
    • 10、删除pos位置前的数据
    • 11、删除pos位置后的数据
    • 12、修改pos位置处的数据
    • 13、打印链表中的数据
    • 14、销毁链表
  • 三、完整代码
    • 1、SList.h
    • 2、SList.c
    • 3、test.c

前言

在上一节中我们提到了顺序表有如下缺陷:

在头部/中间的插入与删除需要挪动数据,时间复杂度为O(N),效率低;

增容需要申请新空间,可能会拷贝数据,释放旧空间,会有不小的消耗;

增容一般是呈2倍的增长,势必会有一定的空间浪费。例如当前容量为100,满了以后增容到 200,如果我们再继续插入了5个数据,后面没有数据插入了,那么会浪费95个数据空间;

基于顺序表的这些不足,我们设计出了链表。


一、链表

1、链表的概念及结构

链表是一种物理存储结构上非连续、非顺序的存储结构,数据元素的逻辑顺序是通过链表中的指针链接次序实现的 。

链表和顺序表的不同之处在于:顺序表不仅要求逻辑结构上也连续,还要求物理结构上连续;而链表只要求逻辑结构上连续,物理结构上可以不连续;

所谓的逻辑结构指的是数据在逻辑上是如何存储的,这是由人们主观想象出来的;而物理结构则是数据在物理内存中实际存储的方式,不随人们的主观意志而改变。

链表的结构图示如下:

从上面的图中我们也可以看出:链表在逻辑结构上连续指的是链表的每一个节点都记录着下一个节点的地址,我们可以根据此地址来找到链表的下一个节点,就好像它们被一根线连起来了一样;

而实际上链表的每一个节点都是在堆区上随机申请的,前一个节点的地址可能比后一个节点大,也可能比后一个节点小,二者之前其实并没有物理结构上的关系。

2、链表的分类

在实际应用中,链表根据带头/不带头、循环/不循环、双向/单向这三种选择一共可以组合出8种结构。

单向或者双向:双向链表对比单向链表来说,其结构体中会多一个结构体指针变量,用来指向前一个节点的地址。

带头或者不带头:带头与不带头其实区别就是链表最开始的时候会有一个节点,这个节点不用来存储数据,仅仅作为链表的头部使用,还是一个节点都没有。

循环或者非循环:非循环链表的最后一个节点的next指向NULL,而循环链表的最后一个节点的next指向链表的第一个节点。

3、最常用的两种链表

虽然链表有这么多中结构,但是我们实际中最常用还是以下两种结构:无头单向非循环链表和双向带头循环链表。

无头单向非循环链表

无头单向非循环链表结构最简单,一般不会单独用来存数据,实际中更多是作为其他数据结构的子结构,如哈希桶、图的邻接表等等;另外这种结构在笔试面试中出现很多;其实如果不做特殊声明,一般情况下无头单向非循环链表指的就是我们的单链表

带头双向循环链表

带头双向循环链表结构最复杂,一般用于单独存储数据;实际中我们使用的链表数据结构,都是带头双向循环链表;另外它虽然结构复杂,但是使用代码实现后会有很多优势,所以反而是链表中使用起来最简单的。


二、单链表的实现

由于单链表是其他结构链表学习的基础,且经常被用做其他数据结构的子结构,在笔试题中也最常被考到,所以下面我们用C原因来手动实现一个单链表,以此来加强我们对单链表的理解。

1、结构的定义

实现,与顺序表一样,单链表也需要一个变量来data来记录数据,且我们应该对data的类型重命名,使得我们的链表可以管理不同类型的数据;其次,由于单链表中需要存储下一个节点的地址,所以我们应该有一个指向结构体的指针。

//符号和结构的声明
typedef int SLTDataType;  //数据类型重命名
typedef struct SListNode   //链表的一个节点
{SLTDataType data;struct SListNode* next;  //存放下一个节点的地址
}SLTNode;

2、创建新节点

由于单链表的每一个节点都需要单独开辟,所以我们可以把创建节点封装成一个函数,避免在头插、尾插、任意位置插入这些位置重复实现。

需要注意的是,由于我们这里实现的单链表是不带头的,即单链表一开始就是空的,所以我们并不需要对其进行初始化操作,只需要定义一个指向NULL的节点指针 plist 即可。

//创建新节点
SLTNode* BuySLTNode(SLTDataType x)
{SLTNode* newNode = (SLTNode*)malloc(sizeof(SLTNode));if (newNode == NULL){perror("malloc fail");return NULL;  //如果malloc失败,返回NULL}newNode->data = x;newNode->next = NULL;return newNode;
}

3、在头部插入数据

特别注意:不管我们在什么地方插入数据,我们都需要传递二级指针,因为链表一开始是空的,所以我们在插入第一个数据的时候需要让 plist 指向我们新开辟的这一个节点,即头结点;而我们知道,要改变 int,需要传递 int*,要改变 int*,需要传递 int**,类比过来,这里的 plist 是一个结构体指针变量,我们想要改变它,让它从 NULL 变为第一个节点的地址,就需要传递结构体指针的地址,即二级指针才能实现。

其次,我们在改变节点中的next指针的时候使用的是结构体指针,即一级指针,而并没有用到二级指针,这是因为我们修改节点中的next是对结构体进行操作,而要改变结构体我们只需要使用结构体指针即可,而不用像上面修改结构体指针一样使用二级指针。

同时,结构体指针的地址是一定不为空的,因为即使是链表为空即 plis == NULL 的时候,&plist 也不等于空,所以我们需要对 pphead 进行断言,来保证代码的鲁棒性;而链表又是可能为空的,所以我们不需要对 *pphead (即 plist) 进行断言。

如果我们使用带头节点的单链表就不需要传递二级指针,因为不管我们如何对链表进行操作,头结点都始终不会改变。

//在头部插入数据
void SListPushFront(SLTNode** pphead, SLTDataType x)  //要改变plist,所以用二级指针来接收plist的地址
{assert(pphead);  //pphead为plist的地址,一定不为空//assert(*pphead);  //error:*pphead得到plist,而链表可能没有节点,所以plist可以为空,不用断言SLTNode* newNode = BuySLTNode(x);  //开辟新节点newNode->next = *pphead;*pphead = newNode;
}

4、在尾部插入数据

在尾部插入数据我们需要先找到的尾结点的前一个节点,因为我们需要让前一个节点的next指针指向新开辟的节点,然后让新开辟的节点的next指向尾结点,这样才能让我们的链表链接起来。

而我们的单链表只能找到下一个节点的地址,想要找到前一个节点需要从头开始遍历,所以单链表尾插的效率是比较低的,时间复杂度为O(N),我们可以通过设计双向链表来解决这个问题。

//在尾部插入数据
void SListPushBack(SLTNode** pphead, SLTDataType x)
{assert(pphead);  //plist的地址一定不为空SLTNode* newNode = BuySLTNode(x);if (*pphead == NULL)  //如果链表为空{newNode->next = *pphead;*pphead = newNode;return;}//如果链表不为空,我们需要找到链表的尾SLTNode* tail = *pphead;while (tail->next != NULL){tail = tail->next;}tail->next = newNode;
}

5、查找数据

查找数据不会改变头结点,所以我们只需要传递一级指针。

//查找数据
SLTNode* SListFind(SLTNode* phead, SLTDataType x)
{assert(phead);  //链表为空查找直接报错SLTNode* cur = phead;while (cur != NULL){if (cur->data == x)return cur;  //找到返回节点地址cur = cur->next;}return NULL;  //找不到返回空
}

6、在pos位置前插入数据

和尾插一样,我们需要从头遍历链表,找到 pos 节点的前一个节点,让该节点的next指向新开辟的节点,使得链表成功链接。

//在pos位置前插入数据
void SListInsert(SLTNode** pphead, SLTNode* pos, SLTDataType x)
{assert(pphead);assert(pos);if (pos == *pphead)  //如果pos等于*pphead,相当于头插{SListPushFront(pphead, x);return;}SLTNode* newNode = BuySLTNode(x);//找到pos位置的前一个节点SLTNode* prev = *pphead;while (prev->next != pos){assert(prev);  //如果prev为空循环还没停止,说明在链表中找不到pos,直接报错prev = prev->next;}prev->next = newNode;newNode->next = pos;
}

7、在pos位置后插入数据

由于单链表在某一节点的前面插入数据时需要从头遍历寻找该节点的前一个节点,导致时间复杂度为O(N),所以人们为了提高单链表的效率,为单链表单独设计了在pos位置后插入数据的函数;除了单链表,其他数据结构插入数据都是在前面插入。

//在pos位置之后插入数据
void SListInsertAfter(SLTNode** pphead, SLTNode* pos, SLTDataType x)
{assert(pphead && pos);SLTNode* next = pos->next;SLTNode* newNode = BuySLTNode(x);pos->next = newNode;newNode->next = next;
}

8、在头部删除数据

特别注意: 和插入数据一样,因为我们删除的可能是链表中的最后一个数据,即可能会改变 plist 的指向 (让 plist 重新指向 NULL),所以不管我们在什么地方删除数据,都需要传递二级指针

其次,由于我们这里是删除数据,所以函数调用者需要保证调用此函数时链表中至少是含有一个数据的;所以我们对 *pphead (等价于 plist) 进行断言,当调用者错误使用此函数时,我们直接报错并介绍程序。

//在头部删除数据
void SListPopFront(SLTNode** pphead)
{assert(pphead);assert(*pphead);  //当链表为空时,删除元素报错SLTNode* tmp = (*pphead)->next;  //注意:* 和 -> 优先级一样,要加括号free(*pphead);*pphead = tmp;
}

9、在尾部删除数据

在尾部删除数据面临着和尾插一样的问题,需要改变前一个节点的next指针,所以时间复杂度也为O(N)。

//在尾部删除数据
void SListPopBack(SLTNode** pphead)
{assert(pphead);assert(*pphead);  //当链表为空时,删除元素报错if ((*pphead)->next == NULL)  //当链表只有一个元素时,相当于头删{SListPopFront(pphead);return;}//先找到链表的尾及其前一个节点SLTNode* cur = *pphead;SLTNode* prev = cur;while (cur->next != NULL){prev = cur;cur = cur->next;}prev->next = NULL;free(cur);cur = NULL;
}

10、删除pos位置前的数据

//删除pos位置前的数据
void SListErase(SLTNode** pphead, SLTNode* pos)
{assert(pphead && pos);assert(*pphead);  //当链表为空时,删除元素报错if (pos == *pphead)  //pos等于*pphead时相当于头删{SListPopFront(pphead);return;}//找到pos的前一个节点SLTNode* prev = *pphead;while (prev->next != pos){assert(prev);  //如果prev为空循环还没停止,说明在链表中找不到pos,直接报错prev = prev->next;}SLTNode* tmp = pos->next;prev->next = tmp;free(pos);pos = NULL;
}

11、删除pos位置后的数据

和在pos位置后插入数据一样,为了提高效率,人们也设计了一个在pos位置后删除数据的函数。

//删除pos位置后的数据
void SListEraseAfter(SLTNode** pphead, SLTNode* pos)
{assert(pphead && pos);assert(*pphead);  //当链表为空时,删除元素报错assert((*pphead)->next);  //当链表只有一个元素时,也不能调用此函数SLTNode* tmp = pos->next->next;free(pos->next);pos->next = tmp;
}

12、修改pos位置处的数据

修改数据也不会改变头指针,所以这里传一级指针;同时,为了保证代码的鲁棒性,我们这里对 phead 和 pos 断言一下。

//修改pos位置处的数据
void SListModify(SLTNode* phead, SLTNode* pos, SLTDataType x)
{assert(phead && pos);SLTNode* cur = phead;while (cur != pos){assert(cur);  //如果cur为空循环还没停止,说明在链表中找不到pos,直接报错cur = cur->next;}cur->data = x;
}

13、打印链表中的数据

打印数据也不会改变头指针,所以这里传一级指针;但是这里和修改数据不一样的地方是,当链表为空的时候我们打印的逻辑也是正常的,只是说调用此函数什么都不打印而已,但是我们不能对其断言让其为空时报错。

//打印链表
void SListPrint(SLTNode* phead)  //打印不需要改变plist
{//不用对Phead进行断言,当链表为空时打印的逻辑是正常的SLTNode* cur = phead;while (cur != NULL){printf("%d->", cur->data);  //-> 和 NULL 是为了让我们的链表更形象化cur = cur->next;}printf("NULL\n");
}

14、销毁链表

销毁链表需要将 plist 值为空,所以这里我们传递二级指针。

//销毁链表
void SListDestory(SLTNode** pphead)
{assert(pphead);SLTNode* cur = *pphead;while (cur != NULL){SLTNode* tmp = cur->next;  //保存下一个节点的地方free(cur);  //释放cur = tmp;}*pphead = NULL;  //将plist置为空
}

三、完整代码

1、SList.h

#pragma once  //防止头文件重复包含//头文件的包含
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>//符号和结构的声明
typedef int SLTDataType;  //数据类型重命名
typedef struct SListNode   //链表的一个节点
{SLTDataType data;struct SListNode* next;  //存放下一个节点的地址
}SLTNode;//函数的声明
//创建新建节点
SLTNode* BuySLTNode(SLTDataType x);
//在头部插入数据
void SListPushFront(SLTNode** pphead, SLTDataType x);
//销毁链表
void SListDestory(SLTNode** pphead);
//在尾部插入数据
void SListPushBack(SLTNode** pphead, SLTDataType x);
//查找数据
SLTNode* SListFind(SLTNode* phead, SLTDataType x);
//在pos之前插入数据
void SListInsert(SLTNode** pphead, SLTNode* pos, SLTDataType x);
//在pos之后插入数据
void SListInsertAfter(SLTNode** pphead, SLTNode* pos, SLTDataType x);
//打印链表
void SListPrint(SLTNode* phead);
//在头部删除数据
void SListPopFront(SLTNode** pphead);
//在尾部删除数据
void SListPopBack(SLTNode** pphead);
//删除pos位置处的数据
void SListErase(SLTNode** pphead, SLTNode* pos);
//删除pos位置后的数据
void SListEraseAfter(SLTNode** pphead, SLTNode* pos);
//修改pos位置处的函数
void SListModify(SLTNode* phead, SLTNode* pos, SLTDataType x);

2、SList.c

#define _CRT_SECURE_NO_WARNINGS 1
#include "SList.h"//创建新节点
SLTNode* BuySLTNode(SLTDataType x)
{SLTNode* newNode = (SLTNode*)malloc(sizeof(SLTNode));if (newNode == NULL){perror("malloc fail");return NULL;  //如果malloc失败,返回NULL}newNode->data = x;newNode->next = NULL;return newNode;
}//销毁链表
void SListDestory(SLTNode** pphead)
{assert(pphead);SLTNode* cur = *pphead;while (cur != NULL){SLTNode* tmp = cur->next;  //保存下一个节点的地方free(cur);  //释放cur = tmp;}*pphead = NULL;  //将plist置为空
}//在头部插入数据
void SListPushFront(SLTNode** pphead, SLTDataType x)  //要改变plist,所以用二级指针来接收plist的地址
{assert(pphead);  //pphead为plist的地址,一定不为空//assert(*pphead);  //error:*pphead得到plist,而链表可能没有节点,所以plist可以为空,不用断言SLTNode* newNode = BuySLTNode(x);  //开辟新节点newNode->next = *pphead;*pphead = newNode;
}//打印链表
void SListPrint(SLTNode* phead)  //打印不需要改变plist
{//不用对Phead进行断言,当链表为空时打印的逻辑是正常的SLTNode* cur = phead;while (cur != NULL){printf("%d->", cur->data);  //-> 和 NULL 是为了让我们的链表更形象化cur = cur->next;}printf("NULL\n");
}//在尾部插入数据
void SListPushBack(SLTNode** pphead, SLTDataType x)
{assert(pphead);  //plist的地址一定不为空SLTNode* newNode = BuySLTNode(x);if (*pphead == NULL)  //如果链表为空{newNode->next = *pphead;*pphead = newNode;return;}//如果链表不为空,我们需要找到链表的尾SLTNode* tail = *pphead;while (tail->next != NULL){tail = tail->next;}tail->next = newNode;
}//查找数据
SLTNode* SListFind(SLTNode* phead, SLTDataType x)
{assert(phead);  //链表为空查找直接报错SLTNode* cur = phead;while (cur != NULL){if (cur->data == x)return cur;  //找到返回节点地址cur = cur->next;}return NULL;  //找不到返回空
}//在pos位置前插入数据
void SListInsert(SLTNode** pphead, SLTNode* pos, SLTDataType x)
{assert(pphead);assert(pos);if (pos == *pphead)  //如果pos等于*pphead,相当于头插{SListPushFront(pphead, x);return;}SLTNode* newNode = BuySLTNode(x);//找到pos位置的前一个节点SLTNode* prev = *pphead;while (prev->next != pos){assert(prev);  //如果prev为空循环还没停止,说明在链表中找不到pos,直接报错prev = prev->next;}prev->next = newNode;newNode->next = pos;
}//在pos位置之后插入数据
void SListInsertAfter(SLTNode** pphead, SLTNode* pos, SLTDataType x)
{assert(pphead && pos);SLTNode* next = pos->next;SLTNode* newNode = BuySLTNode(x);pos->next = newNode;newNode->next = next;
}//在头部删除数据
void SListPopFront(SLTNode** pphead)
{assert(pphead);assert(*pphead);  //当链表为空时,删除元素报错SLTNode* tmp = (*pphead)->next;  //注意:* 和 -> 优先级一样,要加括号free(*pphead);*pphead = tmp;
}//在尾部删除数据
void SListPopBack(SLTNode** pphead)
{assert(pphead);assert(*pphead);  //当链表为空时,删除元素报错if ((*pphead)->next == NULL)  //当链表只有一个元素时,相当于头删{SListPopFront(pphead);return;}//先找到链表的尾及其前一个节点SLTNode* cur = *pphead;SLTNode* prev = cur;while (cur->next != NULL){prev = cur;cur = cur->next;}prev->next = NULL;free(cur);cur = NULL;
}//删除pos位置处的数据
void SListErase(SLTNode** pphead, SLTNode* pos)
{assert(pphead && pos);assert(*pphead);  //当链表为空时,删除元素报错if (pos == *pphead)  //pos等于*pphead时相当于头删{SListPopFront(pphead);return;}//找到pos的前一个节点SLTNode* prev = *pphead;while (prev->next != pos){assert(prev);  //如果prev为空循环还没停止,说明在链表中找不到pos,直接报错prev = prev->next;}SLTNode* tmp = pos->next;prev->next = tmp;free(pos);pos = NULL;
}//删除pos位置后的数据
void SListEraseAfter(SLTNode** pphead, SLTNode* pos)
{assert(pphead && pos);assert(*pphead);  //当链表为空时,删除元素报错assert((*pphead)->next);  //当链表只有一个元素时,也不能调用此函数SLTNode* tmp = pos->next->next;free(pos->next);pos->next = tmp;
}//修改pos位置处的数据
void SListModify(SLTNode* phead, SLTNode* pos, SLTDataType x)
{assert(phead && pos);SLTNode* cur = phead;while (cur != pos){assert(cur);  //如果cur为空循环还没停止,说明在链表中找不到pos,直接报错cur = cur->next;}cur->data = x;
}

3、test.c

#define _CRT_SECURE_NO_WARNINGS 1
#include "SList.h"void test1()   //测试插入
{SLTNode* plist = NULL;  //指向链表第一个节点的指针,因为链表没有节点,所以初始化为NULL;//头插SListPushFront(&plist, 1);SListPushFront(&plist, 2);SListPushFront(&plist, 3);SListPrint(plist);//尾插SListPushBack(&plist, 4);SListPushBack(&plist, 5);SListPushBack(&plist, 6);SListPrint(plist);//在pos位置前插入SLTNode* pos = SListFind(plist, 5);if (pos != NULL){SListInsert(&plist, pos, 50);}pos = SListFind(plist, 3);if (pos != NULL){SListInsert(&plist, pos, 30);}SListPrint(plist);pos = SListFind(plist, 6);if (pos != NULL){SListInsert(&plist, pos, 60);}SListPrint(plist);//在pos位置后插入pos = SListFind(plist, 1);if (pos != NULL){SListInsertAfter(&plist, pos, 10);}pos = SListFind(plist, 3);if (pos != NULL){SListInsertAfter(&plist, pos, 30);}SListPrint(plist);//销毁SListDestory(&plist);
}void test2()  //测试删除
{SLTNode* plist = NULL;  //指向链表第一个节点的指针,因为链表没有节点,所以初始化为NULL;//头插SListPushFront(&plist, 1);SListPushFront(&plist, 2);SListPushFront(&plist, 3);SListPrint(plist);//在头部删除数据//SListPopFront(&plist);//SListPopFront(&plist);//SListPopFront(&plist);//SListPrint(plist);//在尾部删除数据//SListPopBack(&plist);//SListPopBack(&plist);//SListPopBack(&plist);//SListPrint(plist);//删除pos位置处的数据//SLTNode* pos = SListFind(plist, 1);//if (pos != NULL)//{//    SListErase(&plist, pos);//}//pos = SListFind(plist, 3);//if (pos != NULL)//{//    SListErase(&plist, pos);//}//SListPrint(plist);//pos = SListFind(plist, 2);//if (pos != NULL)//{//    SListErase(&plist, pos);//}//SListPrint(plist);//删除pos位置后的数据SLTNode* pos = SListFind(plist, 2);if (pos != NULL){SListEraseAfter(&plist, pos);}SListPrint(plist);pos = SListFind(plist, 3);if (pos != NULL){SListEraseAfter(&plist, pos);}SListPrint(plist);//销毁SListDestory(&plist);
}void test3()  //测试查和改
{SLTNode* plist = NULL;  //指向链表第一个节点的指针,因为链表没有节点,所以初始化为NULL;//头插SListPushFront(&plist, 1);SListPushFront(&plist, 2);SListPushFront(&plist, 3);SListPrint(plist);//查找并修改pos位置处的数据SLTNode* pos = SListFind(plist, 3);if (pos != NULL){SListModify(plist, pos, 30);}SListPrint(plist);pos = SListFind(plist, 1);if (pos != NULL){SListModify(plist, pos, 10);}SListPrint(plist);//销毁SListDestory(&plist);
}int main()
{//test1();  //测试插入//test2();  //测试删除test3();  //测试查和改
}

大家也可以去我的 Gitee 仓库中获取完整代码:SList/SList · 野猪佩奇/日常学习 - 码云 - 开源中国 (gitee.com)


【数据结构】单链表的增删查改(C语言实现)相关推荐

  1. 数据结构之单链表的增删查改等操作画图详解

    单链表 文章目录 单链表 链表的概念及其结构 概念 结构 链表的实现 开辟一个新结点 链表的销毁 打印链表 单链表的尾插 单链表的头插 单链表的头删 单链表的尾删 找到单链表中的一个结点 在pos位置 ...

  2. 单链表的增删查改等基本操作C++实现

    单链表的初始化.增删查改.遍历一次找中间结点.删除一个无头单链表的非尾结点(不给头结点) #include<stdio.h> #include<stdlib.h> #inclu ...

  3. c语言 单链表的增删查改

    文章目录 1.链表基本结构 2.链表的尾插 3.链表的尾删 4.链表的头插与头删 头插 头删 5.链表的查找与修改 查找 修改 6.链表的节点插入与删除 插入 删除 1.链表基本结构 链表由一个个的节 ...

  4. c语言 增删查 案例,C语言实现单链表的增删查改

    #include #include #include"linklist.h" #define Test_Header printf("\n-----%s-----\n&q ...

  5. 【JAVA数据结构】链表的增删查改(单向不带头非循环链表)

    月色与雪色之间,你是第三种绝色 大家好,这里是新一,请多关照

  6. Java——链表的增删查改

    [基础学习] 链表的增删查改 注意:所有链表都先需要判断是否为null <1>链表的增 有三种方法,分别为:头插法,尾插法,任意节点位置插法 1.头插法 2.尾插法 3.任意节点位置插法 ...

  7. 【数据结构】顺序表的增删查改 (C语言实现)

    文章目录 一.线性表 二.顺序表 1.什么是顺序表 2.顺序表的分类 三.动态顺序表的实现 1.结构的定义 2.顺序表的初始化 3.检查容量 4.在头部插入数据 5.在尾部插入数据 6.在指定位置插入 ...

  8. 数据结构----单链表增删改查

    单链表的增删改查 一.链表(Linked List) 链表是有序列表,以节点的方式来存储的,链式存储: 每个节点包含data域,next域:指向下一节点: 链表的各个节点不一定是连续存储: 链表分为带 ...

  9. 无头结点单链表的逆置_单链表的增删查 逆置 倒数第k个节点等问题

    对于单链表而言,它没有双链表那么复杂,它只有头节点,尾节点,节点数据,后继指针.在下面本人实现了 单链表的 增   删   插  查  改. #include #include #include #i ...

最新文章

  1. 7分钟了解科大讯飞开发者节:AI红利期来临,全新1024计划发布(未完待续)
  2. [register]-ARMV8-aarch64-通用寄存器介绍
  3. 宣布 Windows Azure 通过 PCI DSS 合规性验证并且 ISO 认证范围扩大,同时正式发布 Windows Azure Hyper-V 恢复管理器和其他更新功能...
  4. 无重复字符的最长子串—leetcode3
  5. 计算机试题dddd,数据库系统概论试题及答案dddd_Image_Marked.pdf
  6. web语义化方便了谁?
  7. 关于redis的db的使用几号db的问题
  8. 编写一个Java应用程序,从键盘读取用户输入两个字符串,并重载3个函数分别实现这两个字符串的拼接、整数相加和浮点数相加。要进行异常处理,对输入的不符合要求的字符串提示给用户,不能使程序崩溃。
  9. MySql免安装版绿化版安装配置,附MySQL服务无法启动解决方案
  10. H264数据格式解析
  11. 学习笔记:强化学习之A3C代码详解
  12. 「CTS2019」氪金手游
  13. python哪些城市好发展_我用Python爬了各大城市的旅游数据后发现,最好玩的还是这里......
  14. c语言 乘法运算符,C 乘法运算符
  15. GNU/Linux 初學之旅
  16. NVR-网络视频录像机
  17. python读取word内容复制粘贴,Python读取word文本操作详解
  18. 新建、重命名、删除文件夹必须要通过刷新才能显示解决办法
  19. 看完这篇文章,保你学会C语言switch case 语句
  20. Word中已经取消了西文在单词中间换行,但是英文依然部分不连续

热门文章

  1. java创建出现module_Eclipse创建Maven多模块工程Module开发(图文教程)
  2. ObjectARX学习
  3. amber教程5.3:带非标准残基的绿色荧光蛋白的MD
  4. rabbitmq消息阻塞情况分析;自动重试处理异常
  5. 项目落地 - 湖北省城建设计院智慧海绵城市示范工程
  6. 杂文 | 如何在演讲中讲个好故事
  7. python的复数四则运算代码_复数四则运算源代码
  8. python batch_size_神经网络中batch_size参数的含义及设置方法
  9. sssssssssssssss
  10. Java中的字符集编码入门-增补字符