数据结构与算法2—链表

news2024/12/25 9:18:22

1. 链表

  • 线性表的链式存储结构就是用一组任意的存储单元(可以是不连续的)存储线性表的数据元素。
  • 采用链式存储结构的表示的线性表简称链表。
  • 链式存储方式可用于表示线性结构,也可用于表示非线性结构。

链表通常有两个域

  • data域——存放结点值的数据域
  • next域——存放结点的直接后继的地址,需要指针类型表示

2.单链表的表示方式

3.链表的存储结构

  • 由于线性表中各元素间存在着线性关系,每一个元素有一个直接前驱和一个直接后继。
  • 用链式存储结构表示线性表中的一个元素时至少需要两部分信息,一部分用于存放数据元素值,称为数据域;另一部分用于存放直接前驱或直接后继结点的地址(指针),称为指针域,称这种存储单元为结点。

 4.链表的分类

  •  单链表:只设置一个指向后继结点地址的指针域;
  • 循环链表:链表首尾相接构成一个环状结构;
  • 双向链表:单链表中增加一个指向前驱的指针。

 5.单链表的基本运算与实现示例

#include"stdio.h"
#include"malloc.h"

typedef struct
{
    int no;
    int score;
}DataType;

typedef struct node
{
    DataType data;
    struct node *next;
}ListNode;

//线性表的创建  //头插法
ListNode * CreatList()
{
    ListNode *L,*q;
    DataType x;  //x为dataType类型的结构体变量
    L=(ListNode *)malloc(sizeof(ListNode));  //头结点
    L->next=NULL;
    printf("请输入学号和成绩,以学号-1为结束:\n");
    scanf("%d",&x.no);
    while(x.no!=-1)
    {
        scanf("%d",&x.score);
        q=(ListNode *)malloc(sizeof(ListNode));
        q->data=x;
        q->next=L->next;  //头结点所存地址保存于新建结点的指针域中//
        L->next=q;  //新建结点的地址保存于头结点的指针域中
        scanf("%d",&x.no);
    }
    return L;
}
//初始化
ListNode * InitList()
{  
    ListNode *L; 
    L=(ListNode*)malloc(sizeof(ListNode));
    L->next=NULL;
    return L;
}

void PrintList(ListNode * L)
{
    ListNode *p;
    p=L->next;
    while(p!=NULL)
    {
        printf("[%d,%d]\n",p->data.no,p->data.score);
        p=p->next;
    }    
    printf("\n");
}

int GetLength(ListNode *L)
{ 
    int num=0;
    ListNode *p;
    p=L->next;
    while(p!=NULL)
    {  num++;
    p=p->next;
    }
    return(num);
}

void InsertList(ListNode *L,int i,DataType x)
{ 
    ListNode *p,*q,*s;
    int j=1; 
    p=L;
    if(i<1||i>GetLength(L)+1)
        printf("error!\n");
    s=(ListNode *)malloc(sizeof(ListNode));
    s->data=x;
    while(j<=i)
    {   
        q=p;  
        p=p->next;
        j++;
    }    /*找到插入位置*/
    s->next=q->next;//=p
    q->next=s;   
}

//按序号取元素
ListNode *GetNode(ListNode *L,int i)
{ 
    ListNode *p;
    int j=1;
    if(i<1 || i>GetLength(L))
    {
        printf("error!\n");
    }
    p=L->next;
    while(p!=NULL&&j<i)
    {
        p=p->next;
        j++;
    }
       return p;
}
//查找运算
int LocateList(ListNode *L,DataType x)
{ 
    int k=1;
    ListNode *p;
    p=L->next;
    while(p&&p->data.no!=x.no)
    {
        p=p->next;
        k++;
    }
    if(p==NULL) 
        return 0;
    else
        return k;
}
//修改第i个元素
void EditList(ListNode *p,int i,DataType e)
{
    int k=1;
    if(i<1 ||i>GetLength(p))     
    { 
        printf("position error\n");
    }
    while(k<=i)
    {
        p=p->next;
        k++;
    }
    p->data=e;
}

void DeleteList(ListNode *L,int i)
{ 
    ListNode *p,*q;  
    int j=1;
    p=L;
    if(i<1 || i>GetLength(L))
    {
        printf("error!\n");
    }
    while(j<i)
    {
        p=p->next;
        j++;
    }
    q=p->next; 
    p->next=q->next;
    free(q);
}

//排序
void SortList(ListNode *L)
{
    ListNode *p,*q,*pmin;
    DataType e;
    for(p=L->next;p->next!=NULL;p=p->next)  //选择排序
    {
        pmin=p;
        for(q=p->next;q!=NULL;q=q->next)
            if(q->data.score>pmin->data.score)
                pmin=q;
        if(pmin!=p)
        {
            e=p->data;
            p->data=pmin->data;
            pmin->data=e;
        }
    }
}

void main()
{
    ListNode *head,*p;
    DataType e;
    
    //    head=InitList();
    //创建
    head=CreatList();
    PrintList(head);
    
    printf("The length of linklist is %d\n",GetLength(head));
    
    //插入
    e.no=9;  e.score=80;
    InsertList(head,GetLength(head)+1,e);
    printf("插入后:\n");
    PrintList(head);
    printf("The length of linklist is %d\n",GetLength(head));
    //查询     
    e.no=3;
    int k=LocateList(head,e);
    p=GetNode(head,k);
    if(k>0)
        printf("学号为3的记录:[%d %d]\n",p->data.no,p->data.score);
    else
        printf("不存在的\n");
    
    //修改
    e.no=3;  e.score=100;
    int m=LocateList(head,e);
    EditList(head,m,e);
    printf("修改后:\n");
    PrintList(head);
    //删除
    e.no=2;
    int n=LocateList(head,e);
    DeleteList(head,n);
    printf("删除学号为2的记录后:\n");
    PrintList(head);
    printf("The length of linklist is %d\n",GetLength(head));
    //排序
    printf("排序后:\n");
    SortList(head);
    PrintList(head);
}

 6.循环链表

在单链表中,最后一个结点的指针域为空。访问单链表中任何数据只能从链表头开始顺序访问,而不能进行任何位置的随机查询访问。如要查询的结点在链表的尾部则需遍历整个链表。所以单链表的应用受到一定的限制。对单链表进行改进:

它将单链表中最后一个结点的指针指向链表的头结点,使整个链表头尾相接形成一个环形。

 

 7.双向链表

 双向链表用两个指针表示结点间的逻辑关系。 其增加了一个指向直接前驱的指针域,这样形成的链表有两条不同方向的链,前驱和后继,因此称为双链表。

双向链表结点的结构:

 双向链表结点的定义如下:

typedef struct dlistnode{
    DataType data;
    struct dlistnode *prior,*next;
   }DListNode;

双向链表结构示意图:

 

双向链表的插入操作:

关键语句指针操作序列既不是唯一也不是任意的。操作①必须在操作③之前完成,否则*p的前驱结点就丢掉了。

双向链表的删除操作:

另一种写法:

void DDeleteNode(DListNode *p)
{
    p->prior->next = p->next;
    p->next->prior = p->prior;
    free(p);
}

补:循环双链表

循环链表+双向链表的结合

带头结点且有n个结点的循环双链表

 

链式存储结构的特点:

优点:

  • 结点空间可以动态申请和释放;
  • 它的数据元素的逻辑次序靠结点的指针来指示,进行数据插入或删除时不需要移动数据元素。

不足:

  • 每个结点中的指针域需额外占用存储空间,当每个结点的数据域所占字节不多时,指针域所占存储空间的比重就显得很大;
  • 链式存储结构是一种非随机存储结构。对任一结点的操作都要从指针链查找到该结点,这增加了算法的复杂度。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/157707.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

北大硕士LeetCode算法专题课--递归和回溯

算法专题课 北大硕士LeetCode算法专题课-栈、队列相关问题_骨灰级收藏家的博客-CSDN博客 北大硕士LeetCode算法专题课--链表相关问题_骨灰级收藏家的博客-CSDN博客 北大硕士LeetCode算法专题课-查找相关问题_骨灰级收藏家的博客-CSDN博客 北大硕士LeetCode算法专题课-字符串…

一.【linux】常用操作命令

目录 1 . 查看当前登录系统用户信息 【who】 2 . 查看终端用户whoami 3 . 查看日期date 4 . 查看系统信息uname 5 . 日历cal 6 . 计算器bc 8 . 显示或修改计算机主机名的命令hostname 9 . 在显示器上显示文字的命令echo 10 . 清除计算机屏幕上的信息命令clear 11 . 查…

【尚硅谷】Java数据结构与算法笔记04 - 栈

文章目录一、实际需求二、栈的介绍三、应用场景四、数组模拟栈4.1 思路分析4.2 Java代码实现五、栈实现综合计算器5.1 中缀表达式5.2 后缀表达式一、实际需求 二、栈的介绍 栈的英文为(stack)栈是一个先入后出(FILO-First In Last Out)的有序列表。栈(stack)是限制线性表中元素…

nacos 2.1.0集群生产环境多节点部署

nacos 2.1.0集群生产环境多节点部署 版本 2.1.0版本发布日期 2022-04-29官网 集群部署说明GitHub GitHub - alibaba/nacos: an easy-to-use dynamic service discovery, configuration and service management platform for building cloud native applications. 下载地址&…

大数据面试题(七):Flume核心高频面试题

文章目录 Flume核心高频面试题 一、Flume有哪些组件&#xff0c;flume的source、channel、sink具体是做什么的 二、你是如何实现flume数据传输的监控的 三、Flume的source&#xff0c;sink&#xff0c;channel的作用&#xff1f;你们source是什么类型&#xff1f; 四、Flu…

2023年浙江建筑施工电工(建筑特种作业)模拟试题及答案

百分百题库提供特种工&#xff08;电工&#xff09;考试试题、特种工&#xff08;电工&#xff09;考试真题、特种工&#xff08;电工&#xff09;证考试题库等,提供在线做题刷题&#xff0c;在线模拟考试&#xff0c;助你考试轻松过关。 159.临时⽤电⼯程验收应填写&#xff0…

基于Vue 全家桶实现网易云音乐 WebApp

基于 Vue(2.5) vuex vue-router vue-axios better-scroll Scss ES6 等开发一款移动端音乐 WebApp&#xff0c;UI 界面参考了安卓版的网易云音乐、flex 布局适配常见移动端。因为服务器的原因&#xff0c;所以可能多人访问的时候有些东西会加载不出来&#xff0c;如果可以的…

P2069 松鼠吃果子

题目描述 输入输出样例 输入 #1 10 4 输出 #1 9 说明/提示 注&#xff1a;吃掉的果子依次为 3&#xff0c;8&#xff0c;5&#xff08;回到下面重做第 3 跳&#xff09;&#xff0c;9&#xff08;回到下面重做第 4 跳&#xff09;。 写着感觉好别扭 感觉和字符串没啥关…

接口管理工具YApi怎么用?颜值高、易管理、超好用!

众多接口管理工具如雨后春笋搬冒出。让人欣慰的是&#xff0c;有许多优秀作品来自国内&#xff0c;包含YApi和rap。 看着中文的官网&#xff0c;熟悉的汉语&#xff0c;不禁让人暗爽。当然这也就带来另一个弊端&#xff0c;因为使用基数少&#xff0c;所以参考资料少。我们想学…

发送tcp/udp请求

本文章介绍下通过nc工具和iperf工具&#xff0c;发送tcp/udp请求一、nc工具&#xff08;netcat工具&#xff09;这个工具linux系统默认是自带的&#xff0c;以下是命令的常用参数1.1 发送tcp请求在服务端监听端口nc -l port客户端连接并发送请求nc -v host port在服务端收到了信…

自定义动态IRule进行路由转发

gateway的负载均衡以及feign服务之间的调用或者RestTemplate请求&#xff0c;都可以使用自定义IRule规则进行路由转发。 自定义IRule 固定IRule&#xff1a;将IRule的一个实现类注入到spring容器中 Configuration public class MyIRuleConfig { Beanpublic IRule myIRule() {…

力扣sql基础篇(五)

力扣sql基础篇(五) 1 页面推荐 1.1 题目内容 1.1.1 基本题目信息 1.1.2 示例输入输出 a 输入示例 b 输出示例 1.2 示例sql语句 # Write your MySQL query statement below SELECT distinct page_id recommended_page FROM Likes WHERE (user_id in (SELECT user2_idFROM …

字符串查找、错误信息、字符分类函数

1、寻找子串strstr char* my_strstr(const char* str1, const char* str2) {assert(str1 && str2);const char* cp str1;const char* s1 str1;const char* s2 str2;while (*cp){s2 str2;s1 cp;while (*s1!\0 && *s2!\0 && * s1 *s2){s1;s2;}if (…

一步一步学爬虫(5)Ajax分析方法

一步一步学爬虫&#xff08;5&#xff09;Ajax分析方法5.2.1. 分析案例5.2.2. 过滤请求5.2.3. 总结5.2.1. 分析案例 这里还需要借助浏览器的开发者工具&#xff0c;下面以 Chrome 浏览器为例来介绍。 首先&#xff0c;用 Chrome 浏览器打开微博的链接 https://m.weibo.cn/u/283…

排查开机启动项

防止被控机器失联部分木马会添加开机启动项作为复活的手段 /etc/rc.local /etc/rc.local 是 /etc/rc.d/rc.local的软链接 [rootVM-4-11-centos etc]# ll rc.local lrwxrwxrwx 1 root root 13 Apr 6 2022 rc.local -> rc.d/rc.localrc.local的原始内容&#xff1a; #!/…

中文文案排版指北

格式规范千千万&#xff0c;内容创作 No.1 但是当下我们专注与文案排版。优秀的排版、正确使用标点符号能让阅读者赏心悦目&#xff0c;提升阅读体验。这篇文章将 GitHub 上的有关排版的规范摘抄部分供大家查阅&#xff0c;更多详情还请访问原文所在&#xff08;底部引用中的地…

APSIM练习:小麦苜蓿轮作

在本练习中&#xff0c;您将创建一个小麦苜蓿轮作&#xff0c;其中草料阶段持续数年。 基于“Rotation Sample.apsim”创建一个新的模拟。这是小麦鹰嘴豆轮作。在 Manager 文件夹中&#xff0c;Rotation Management 组件用于表示旋转&#xff0c;使用带有节点&#xff08;圆圈…

网站需要怎么优化比较好(网站优化有哪些技巧)

怎样让网站优化在短时间内看到效果&#xff1f; 在企业推广营销推广中&#xff0c;网站优化占据了很大一部分&#xff0c;对于企业推广营销有着决定性影响。如果想让您的企业在诸多品牌中脱颖而出&#xff0c;甚至在互联网中有强大的竞争力&#xff0c;你必须要做好网站优化&a…

基于微信小程序云开发的投票小程序源码,图文投票微信小程序源码

功能介绍 投票活动十分火&#xff0c;商家&#xff0c;企业&#xff0c;机构偶尔都会来一场投票活动评选&#xff0c;本小程序支持图文投票&#xff0c;简单方便、随时随地完成投票&#xff0c;可以方便设定投票模式&#xff08;按天按全程&#xff0c;投票数限定&#xff09;…

MySQL基础——DQL语句

select 聚合函数(字段列表) from 表名&#xff1b; Mysql的相关知识 好处 &#xff1a; 1.持久化数据到本地 2.可以实现结构化查询&#xff0c;方便管理 数据库主要分为: DB(数据库)&#xff0c;保存一系列有组织的数据容器。 DBMS(数据库管理系统)&#xff0c;用于对DB获得数…