当前位置: 首页 > backend >正文

数据结构(二)——线性表的链式表示和实现

一、单链表

1.单链表的定义

如图所示每个节点包含两个域:数据域和指针域。数据域存储数据元素,指针域存储下一个节点的地址,因此指针指向的类型也是节点类型。每个指针都指向下一个节点,都是朝一个方向的,这样的链表称为单向链表或单链表。

eg1: 若线性表采用链式存储,则表中各元素的存储地址(不一定是连续的)

2.单链表的操作

(1)创建(插入)

创建单链表分为头插法和尾插法两种,头插法是指每次把新节点插到头节点之后,其创建的单链表和数据输入顺序正好相反,因此也称为逆序建表。尾插法是指每次把新节点链接到链表的尾部,其创建的单链表和数据输入顺序一致,因此也称为正序建表。

头插法:

尾插法:

eg1: 在一个单链表中,已知q结点是p结点的前驱结点,若在q和p之间插入S结点则执行(C)

(2)查找

在一个单链表中查找是否存在元素e,可以定义一个p指针,指向第一个元素节点,比较p指向节点的数据域是否等于e。如果相等,查找成功,返回true;如果不等,则p指向p的下一个节点,继续比较,如果p为空,查找失败,返回false

eg: 从一个具有n个结点的单链表中查找其值等于x结点时,在查找成功的情况下,需比较(D)个结点

(3)删除

删除一个节点,实际上是把这个节点跳过去。根据单向链表向后操作的特性,要想跳过第i个节点,就必须先找到第i-1个节点,否则是无法跳过去的

eg:构造一个空的带表头结点的单链线性表L,并逆位序(头插法)输入n个元素的值;访问单链表L的第i个元素并打印输出;在单链表L的第i个元素之前插入新的元素e;在单链表表L中删除第i个元素,并用e返回其值;打印输出链表长度。

代码如下:

#include <stdio.h>
#include <stdlib.h>//定义链表结点结构体
typedef struct LNode{int data;  //数据域struct LNode *next;  //指针域
}LNode,*LinkList;//构造一个空的带表头结点的单链线性表L,并逆位序(头插法)输入n个元素的值
void CreateList(LinkList *L,int n){*L = (LinkList)malloc(sizeof(LNode));  //创建头节点,并返回头结点的指针(*L)->next = NULL;  //初始化头结点的next指针为NULLLinkList p;  //定义指向链表结点的指针int i;for(i = n;i>0;--i){  //循环n次,创建n个结点p  =(LinkList)malloc(sizeof(LNode));  //创建新结点scanf("%d",&p->data);  //输入数据p->next = (*L)->next;  //将新数据插入到表头,next指向当前头结点的下一个结点(*L)->next = p;  //头结点的next指向新结点}
}//访问单链表L的第i个元素并打印输出
int GetElem(LinkList L,int i){LNode *p = L->next;  //从头结点的下一结点开始查找int j = 1;  //从1开始计数while(p&&j<i){  //遍历链表,直到找到第i个元素p = p->next;j++;}if(!p||j>i){  //未找到则退出程序printf("第%d个元素不存在\n",i);exit(1);}else{return p->data;  //找到则返回第i个元素的值}
}//在单链表L的第i个元素之前插入新的元素e
void ListInsert(LinkList L,int i,int e){LinkList p = L;  //定义指向链表的指针,初始化为头结点int j = 0;  //计数器,初始值为0while(p&&j<i-1){  //遍历链表,寻找第i-1个节点p = p->next;j++;}if(!p||j>i-1){  //如果未找到第i-1个节点,表示插入位置无效,退出程序exit(1);}else{LinkList s = (LinkList)malloc(sizeof(LNode));  //创建一个新节点s->data = e;  //新节点数据域赋值为es->next = p->next;  //新节点的next指向第i个节点p->next = s;  //第i-1个节点的next指向新节点}
}//在单链表表L中删除第i个元素,并用e返回其值
int ListDelete(LinkList L,int i,int *e){LinkList p = L;  // 定义指向链表的指针,初始化为头节点int j = 0;  // 计数器,初始值为0while(p->next&&j<i-1){  // 遍历链表,寻找第i-1个节点p = p->next;j++;}if(!(p->next)||j>i-1){  // 如果未找到第i-1个节点,表示删除位置无效,退出程序exit(1);}else{LinkList q = p->next;  // 找到第i个节点*e = q->data;  // 返回删除节点的值p->next = q->next;  // 将第i-1个节点的next指向第i+1个节点free(q);  // 释放被删除节点的内存return 1;  // 返回1表示删除成功}
}//打印输出链表长度
int ListLength(LinkList L){LNode *p = L;  //从头结点开始遍历链表int count = 0;  //计数器while(p->next){  //遍历链表直到末尾p = p->next;count++;  //每经过一个节点,计数器加一}return count;  //返回链表长度
}//打印链表
void PrintList(LinkList L){LNode *p = L->next;  //从头结点的下一个节点开始遍历while(p){  //遍历链表直到末尾printf("%d",p->data);  //输出当前节点的data值p = p->next;  //移动到下一个节点}printf("");  //换行
}int main(){LinkList L;int n,i,e;printf("请输入链表的长度:");scanf("%d",&n);CreateList(&L,n);  //创建链表printf("链表创建成功,内容为:");PrintList(L);//访问第i个元素printf("请输入要访问的元素位置:");scanf("%d",&i);int elem = GetElem(L,i);printf("第%d个元素为:%d",i,elem);//插入新元素printf("请输入要插入的位置和新元素值:");scanf("%d %d",&i,&e);ListInsert(L,i,e);printf("插入新元素后链表内容为:");PrintList(L);//删除第i个元素printf("请输入要删除的元素位置:");scanf("%d",&i);if(ListDelete(L,i,&e)){printf("删除成功,删除的元素是: %d",e);printf("删除后链表内容为:");PrintList(L);}//打印链表长度int length = ListLength(L);printf("链表长度为:%d",length);return 0;
}

http://www.xdnf.cn/news/4849.html

相关文章:

  • Linux网络编程day7 线程池and UDP
  • PostgreSQL安装与升级cron插件
  • 如何使用极狐GitLab 软件包仓库功能托管 terraform?
  • MATLAB 在医疗行业的应用
  • ImGui 基础用法
  • 第5章 深度学习和卷积神经网络
  • 飞算 用到妙处 AI辅助编程 - 双击方法名,自动识别到上下文中很方便
  • macOS Python 环境配置指南
  • Java死锁问题全解析:从原理到实战解决方案
  • 6:点云处理—QT三维窗口显示
  • Screeps Arena基础入门
  • 碰一碰发视频一键成片功能开发实践与技术解析
  • 字符宽度介绍
  • 仿LISP运算 - 华为OD机试真题(A卷、JavaScript题解)
  • 特征工程概述
  • QT 文件选择对话框 QFileDialog
  • DL/T645-2007电表协议简介以及请求应答帧格式
  • RSAC 2025观察:零信任+AI=网络安全新范式
  • 【HCIP】----OSPF综合实验
  • Day 14 训练
  • 雷赛伺服电机
  • 山东安全员A证的考试科目有哪些?
  • MySQL中隔离级别那点事
  • 主备Smart Link + Monitor Link组网技术详细配置
  • 【LeetCode】删除排序数组中的重复项 II
  • 2018机械行业ERP软件发展趋势
  • 从 ImageNet 到产业革命:AlexNet 作为破局者的三大核心创新及其时代穿透力
  • SKNet、空间注意力介绍
  • 1.MySQL数据库初体验
  • Matlab 基于Hough变换的人眼虹膜定位方法