• 数据结构(二)——线性表(单链表)


     2.3线性表的链式表示

    顺序表的优缺点:
    优点:可随机存储,存储密度高
    缺点:要求大片连续空间,且改变容量不方便

    2.3.1 单链表的基本概念

    单链表:用链式存储实现了线性结构。一个结点存储一个数据元素,各结点间的前后关系用一个指针表示。
    优点:不要求大片连续空间,改变容量方便。
    缺点:不可随机存取,要耗费一定空间存放指针。

    1. //用代码定义一个单链表
    2. struct LNode{ //LNode是结点
    3. ElemType data; //data为数据域 定义单链表结点类型
    4. struct LNode *next; //*next是指针域 指针指向下一个结点
    5. }
    6. struct LNode *p = (struct LNode *)malloc(sizeof(struct LNode));
    7. //增加一个新的结点:在内存中申请一个结点所需空间,并用指针n指向这个结点

    typedef关键字——数据类型重命名
    typedef <数据类型> <别名>
    typedef struct LNode LNode;        //这样使用就可以直接哟弄个LNode 不需要前面加一长串
    typedef struct LNode *LinkList;

    要表示一个单链表时,只需要表明一个头指针L,指向单链表的第一个结点
    LNode *L;        //声明一个指向单链表第一个结点的指针 

    LinkList L;        //声明一个指向单链表第一个结点的指针  代码可读性更强

    不带头结点的单链表

    1. typedef struct LNode{ //定义单链表结点类型
    2. ElemType data; //每个节点存放一个数据元素
    3. struct LNode *next; //指针指向下一个节点
    4. }LNode, *LinkList;
    5. //初始化一个空的单链表
    6. bool InitList(LinkList &L){
    7. L = NULL; //空表,暂时还没有任何结点 防止脏数据
    8. return true;
    9. }
    10. void test(){
    11. LinkList L; //声明一个指向单链表的头指针
    12. //初始化一个空表
    13. InitList(L);
    14. ...
    15. }
    16. //判断单链表是否为空
    17. bool Empty(LinkList L){
    18. return (L==NULL)
    19. }

    带头结点的单链表

    1. typedef struct LNode{
    2. ElemType data;
    3. struct LNode *next; //指针指向下一结点
    4. }LNode, *LinkList;
    5. // 初始化一个单链表(带头结点)
    6. bool InitList(LinkList &L){
    7. L = (LNode *)malloc(sizeof(LNode)); //分配一个头结点
    8. if (L == NULL) //内存不足,分配失败
    9. return false;
    10. L->next = NULL; //头结点之后暂时还没有结点
    11. return true;
    12. }
    13. void test(){
    14. LinkList L; //声明一个指向单链表的头指针
    15. //初始化一个空表
    16. InitList(L);
    17. ...
    18. }
    19. //判断单链表是否为空
    20. bool Empty(LinkList L){
    21. if (L->next == NULL)
    22. return true;
    23. else
    24. return false;
    25. }

    不带头结点,写代码更麻烦 对第一个数据结点和后续数据结点的 处理需要用不同的代码逻辑 对空表和非空表的处理需要用不同的 代码逻辑

    带头结点,写代码更方便

    2.3.2_1单链表的插入删除

    ListInsert(&L,i,e):插入操作。在表L中的第i个位置上插入指定元素e

    按位序插入(带头结点):

    1. typedef struct LNode{
    2. ElemType data;
    3. struct LNode *next;
    4. }LNode,*LinkList;
    5. //在第i个位置插入元素e(带头结点)
    6. bool ListInsert(LinkList &L,int i ElemType e){ //i是要插入的位序
    7. if(i<1) //i表示的是位序 位序最小是1
    8. return false;
    9. LNode *p; //指针p指向当前扫描到的结点
    10. int j=0; //当前p指向的是第几个结点
    11. p=L; //L指向头结点,头结点是第0个结点,不存数据
    12. while(p!=NULL && j-1){ //循环找到第i-1个结点 如果i>length p最后会等于NULL
    13. p=p->next;
    14. j++;
    15. }
    16. if(p==NULL) //i值不合法
    17. return false;
    18. LNode *s=(LNode*)malloc(sizeof(LNode)); //malloc开辟一块空间s
    19. s->data=e; //把要插入的元素存到新结点中
    20. s->next=p->next; //让s指向的next的指针等于p结点的next指针指向的位置
    21. p->next=s; //让p的结点的next指针指向新的结点s 即将结点s连到p之后
    22. return true; //插入成功
    23. }

    时间复杂度:这里问题规模n指的是表的长度
            最好时间复杂度:O(1)
            最坏时间复杂度:O(n)
            平均时间复杂度:O(n)

    按位序插入(不带头结点):
    由于不存在“第0个”结点,因此i=1时需要特殊处理

    1. typedef struct LNode{
    2. ElemType data;
    3. struct LNode *next;
    4. }LNode, *LinkList;
    5. //在第i个位置插入元素e
    6. bool ListInsert(LinkList &L, int i, ElemType e){
    7. //判断i的合法性
    8. if(i<1)
    9. return false;
    10. //需要判断插入的位置是否是第1个 对第一个结点单独写操作进行处理
    11. if(i==1){
    12. LNode *s = (LNode *)malloc(size of(LNode)); //malloc申请一个结点
    13. s->data =e; //把要插入的e放到新申请的结点中
    14. s->next =L; //让新结点的指针指向L
    15. L=s; //头指针指向新结点
    16. return true;
    17. }
    18. //i>1的情况与带头结点一样,唯一区别是j的初始值为1
    19. LNode *p; //指针p指向当前扫描到的结点
    20. int j=1; //当前p指向的是第几个结点
    21. p = L; //p指向的是第1个结点,注意不是头结点
    22. //循环找到第i-1个结点
    23. while(p!=NULL && j-1){ //如果i>lengh,p最后会等于NULL
    24. p = p->next;
    25. j++;
    26. }
    27. //p值为NULL说明i值不合法
    28. if (p==NULL)
    29. return false;
    30. //在第i-1个结点后插入新结点
    31. LNode *s = (LNode *)malloc(sizeof(LNode));
    32. s->data = e;
    33. s->next = p->next;
    34. p->next = s;
    35. return true;
    36. }

    结论:不带头结点写代码更不方便,推荐用带头结点注意

    指定结点的后插操作:

    1. typedef struct LNode{
    2. ElemType data;
    3. struct LNode *next;
    4. }LNode, *LinkList;
    5. // 在结点p后插入元素e
    6. bool InsertNextNode(LNode *p, ElemType e){
    7. if(p==NULL){
    8. return false;
    9. }
    10. LNode *s = (LNode *)malloc(sizeof(LNode));
    11. if(s==NULL) //内存分配失败
    12. return false;
    13. s->data = e; //用结点s保存数据元素e
    14. s->next = p->next;
    15. p->next = s; //将结点s连接到p后
    16. return true;
    17. }
    18. // 按位序插入的函数中可以直接调用后插操作
    19. bool ListInsert(LinkList &L, int i, ElemType e){
    20. if(i<1)
    21. return False;
    22. LNode *p;
    23. //指针p指向当前扫描到的结点
    24. int j=0;
    25. //当前p指向的是第几个结点
    26. p = L;
    27. //循环找到第i-1个结点
    28. while(p!=NULL && j-1){
    29. //如果i>lengh, p最后会等于NULL
    30. p = p->next;
    31. j++;
    32. }
    33. return InsertNextNode(p, e)
    34. }

    时间复杂度:O(1)

    指定结点的前插操作:

    1. typedef struct LNode{
    2. ElemType data;
    3. struct LNode *next;
    4. }LNode, *LinkList;
    5. //前插操作: 在结点p前插入元素e
    6. bool InsertPriorNode(LNode *p, ElemType e){
    7. if(p==NULL)
    8. return false;
    9. LNode *s = (LNode *)malloc(sizeof(LNode));
    10. if(s==NULL) // 内存分配失败
    11. return false;
    12. s->next = p->next;
    13. p->next = s; // 将新结点s连到结点p之后
    14. s->data = p->data; //将p中元素复制到s中
    15. p->data = e; //p中元素覆盖为e
    16. return true;
    17. }

    按位序删除(带头结点)

    ListDelete(&L,i,&e):删除操作。删除表L中第i个位置的元素,并用e返回删除元素的值
    找到第i-1个结点,将其指针指向第i+1个结点,并释放第i个结点,头结点可看做“第0个”结点

    1. typedef struct LNode{
    2. ElemType data;
    3. struct LNode *next;}LNode, *LinkList;
    4. // 删除第i个结点并将其所保存的数据存入e
    5. bool ListDelete(LinkList &L, int i, ElemType &e){
    6. if(i<1)
    7. return false;
    8. LNode *p; //指针p指向当前扫描到的结点
    9. int j=0; //当前p指向的是第几个结点
    10. p = L; //L指向头结点,头结点是第0个结点,不存数据
    11. while(p!=NULL && j-1){ //循环找到第i-1个结点
    12. p = p->next; //如果i>lengh,p和p的后继结点会等于NULL
    13. j++;
    14. }
    15. if(p==NULL) //i值不合法
    16. return false;
    17. if(p->next == NULL) //第i-1个结点之后已无其他结点
    18. return false;
    19. LNode *q = p->next; //令q指向被删除的结点
    20. e = q->data; //把删除的结点的值存到变量e
    21. p->next = q->next; //p的next指向q的next 就是null 相当于将*q结点从链中“断开”
    22. free(q); //释放结点的存储空间
    23. return true; //删除成功
    24. }

    指定结点的删除 
    删除结点p,需要修改其前驱 结点的 next 指针
    方法1:传入头指针,循环寻找 p 的前驱结点
    方法2:类似于结点前插的实现

    1. // 删除指定结点p
    2. bool DeleteNode(LNode *p){
    3. if(p==NULL)
    4. return false;
    5. LNode *q = p->next; // 令q指向p的后继结点
    6. // 如果p是最后一个结点,则q指向NULL,继续执行就会报错
    7. p->data = q->data; //把后继结点的数据复制到p结点的数据域中
    8. p->next = q->next; //让p结点的next指针指向q结点后的位置 相当于将*q结点从链中“断开”
    9. free(q); //释放q结点的空间
    10. return true;
    11. }

     

    2.3.2_2单链表的插入查找

    GetElem(L,i):按位查找操作。获取表L中第i个位置的元素的值。

    1. typedef struct LNode{
    2. ElemType data;
    3. struct LNode *next;
    4. }LNode, *LinkList;
    5. // 查找指定位序i的结点并返回
    6. LNode * GetElem(LinkList L, int i){
    7. if(i<0)
    8. return NULL;
    9. LNode *p; //指针p指向当前扫描到的结点
    10. int j=0; //当前p指向的是第几个结点
    11. p = L; //L指向头结点,头结点是第0个结点(不存数据)
    12. while(p!=NULL && j
    13. p = p->next;
    14. j++;
    15. }
    16. return p;
    17. }
    18. // 在结点p后插入元素e
    19. bool InsertNextNode(LNode *p, ElemType e){
    20. if(p==NULL){
    21. return false;
    22. }
    23. LNode *s = (LNode *)malloc(sizeof(LNode));
    24. if(s==NULL) //内存分配失败
    25. return false;
    26. s->data = e; //用结点s保存数据元素e
    27. s->next = p->next;
    28. p->next = s; //将结点s连接到p后
    29. return true;
    30. }
    31. // 封装后的插入操作,在第i个位置插入元素e
    32. bool ListInsert(LinkList &L, int i, ElemType e){
    33. if(i<1)
    34. return False;
    35. // 找到第i-1个元素
    36. LNode *p = GetElem(L, i-1);
    37. // 在p结点后插入元素e
    38. return InsertNextNode(p, e);//调用查询操作和后插操作
    39. }

    平均时间复杂度:O(n) 

    LocateElem(L,e):按值查找操作。在表L中查找具有给定关键字值的元素。

    1. // 查找数据域为e的结点指针,否则返回NULL
    2. LNode * LocateElem(LinkList L, ElemType e){
    3. LNode *p = L->next; //定义一个指针p,让他指向头结点的下一个结点
    4. // 从第一个结点开始查找数据域为e的结点
    5. while(p!=NULL && p->data != e){ //当p不为NULL且数据跟传入的e不相等时循环
    6. p = p->next; //让p指向下一个结点
    7. }
    8. return p; //跳出循环后返回p
    9. }

    平均时间复杂度:O(n)  只能依次往后查找

    求表的长度

    1. // 计算单链表的长度
    2. int Length(LinkList L){
    3. int len=0; //统计表长
    4. LNode *p = L;
    5. while(p->next != NULL){
    6. p = p->next;
    7. len++;
    8. }
    9. return len;
    10. }

    2.3.2_3单链表的插入建立

    尾插法

    1. // 使用尾插法建立单链表L
    2. LinkList List_TailInsert(LinkList &L){
    3. int x; //设ElemType为整型int
    4. L = (LinkList)malloc(sizeof(LNode)); //建立头结点(初始化空表)
    5. LNode *s, *r = L; //r为表尾指针
    6. scanf("%d", &x); //输入要插入的结点的值
    7. while(x!=9999){ //输入9999表示结束
    8. s = (LNode *)malloc(sizeof(LNode));
    9. s->data = x;
    10. r->next = s; //在r结点之后插入元素x
    11. r = s; //r指针指向新的表尾结点
    12. scanf("%d", &x);
    13. }
    14. r->next = NULL; //尾结点指针置空
    15. return L;
    16. }

    头插法

    1. LinkList List_HeadInsert(LinkList &L){ //逆向建立单链表
    2. LNode *s;
    3. int x;
    4. L = (LinkList)malloc(sizeof(LNode)); //建立头结点
    5. L->next = NULL; //初始为空链表,先把头指针指向NULL,这步很关键
    6. scanf("%d", &x); //输入要插入的结点的值
    7. while(x!=9999){ //输入9999表结束
    8. s = (LNode *)malloc(sizeof(LNode)); //创建新结点
    9. s->data = x;
    10. s->next = L->next;
    11. L->next = s; //将新结点插入表中,L为头指针
    12. scanf("%d", &x);
    13. }
    14. return L;
    15. }

     头插法实现链表的逆置:

    1. // 将链表L中的数据逆置并返回
    2. LNode *Inverse(LNode *L){
    3. LNode *p, *q;
    4. p = L->next; //p指针指向第一个结点
    5. L->next = NULL; //头结点置空
    6. // 依次判断p结点中的数据并采用头插法插到L链表中
    7. while (p != NULL){
    8. q = p;
    9. p = p->next;
    10. q->next = L->next;
    11. L->next = q;
    12. }
    13. return L;
    14. }

     头插法逆置详见【数据结构】单链表逆置:头插法图解

  • 相关阅读:
    房屋租赁管理系统
    <STL标准库中对stack、queue、priority_queue及反向迭代器的模拟实现>——《C++初阶》
    FutureTask和CompletableFuture的模拟使用
    springboot中如何在测试环境下进行web环境模拟测试
    nacos回顾+cloud家族的皇家卫士Sentinel
    Top 5 Cutting-edge technology examples 2023
    【机器学习】强化学习的概念及马尔科夫决策
    【直播笔记0629】 并发编程二:锁
    LeetCode100122. Separate Black and White Balls
    lerna publish时报错 E401 [UNAUTHORIZED] Login first
  • 原文地址:https://blog.csdn.net/a_rain2333/article/details/136572786