从0开始回顾数据结构---链表与堆

简介: #include <iostream>#include <algorithm>#include <string.h>using namespace std;const int N = 100010;int h[N], ph[N], hp[N], cnt;void heap_swap(int a, int b){ swap(ph[hp[a]],ph[hp[b]]); swap(hp[a], hp[b]); swap(h[a], h[b]);}void down(int u){ int t = u; if (u * 2 <= cnt &&

链表

1、链表的建立

#include <cstdio>
using namespace std;
struct ListNode {
    int val;
    ListNode *next;
    ListNode(int x) : val(x), next(nullptr) {}
  //ListNode() : val(0), next(nullptr) {}
    //ListNode(int x, ListNode *next) : val(x), next(next) {}
};
int main(){
}

2、链表反转

给定一个链表的头节点,让我们反转该链表并输出反转后链表的头节点。

样例:

如样例所示,原始链表为1->2->3->4->5->NULL,我们将其翻转输出5->4->3->2->1->NULL。下面我们来讲解双指针的做法。


将一个链表翻转,即将该链表所有节点的next指针指向它的前驱节点。由于是单链表,我们在遍历时并不能直接找到其前驱节点,因此我们需要定义一个指针保存其前驱节点。


每次翻转时,我们都需要修改当前节点的next指针。如果不在改变当前节点的next指针前保存其后继节点,那么我们就失去了当前节点和后序节点的联系,因此还需要额外定义一个指针用于保存当前节点的后继节点。


具体过程如下:

  1. 定义一个前驱指针precur指针,pre指针用来指向前驱节点,cur指针用来遍历整个链表,初始化pre = nullcur = head


  1. 我们首先保存cur指针指向节点的后继节点,然后让cur指针指向节点的next指针指向其前驱节点,即cur->next = pre
  2. pre指针和cur指针分别后移一位,重复上述过程,直到cur指向空节点。
  3. 最后我们返回pre节点。

图示过程如下:


时间复杂度分析: 只遍历一次链表,时间复杂度是

#include <cstdio>
#include <iostream> 
using namespace std;
struct ListNode {
    int val;
    ListNode *next;
    ListNode(int x) : val(x), next(nullptr) {}
  //ListNode() : val(0), next(nullptr) {}
    //ListNode(int x, ListNode *next) : val(x), next(next) {}
};
ListNode* head =  nullptr;
// 头插法 
void add(int x)
{
  ListNode* node = new ListNode(x);
    node->next = head;
    head = node;
}
void print()
{
    ListNode* p = head;
    while (p)
    {
        printf("%d ", p->val);
        p = p->next;
    }
}
ListNode* reverseList(ListNode* head) {
    ListNode* pre = nullptr;    //前驱指针
    ListNode* cur = head;
    while(cur){
        ListNode* t = cur->next;  //先保存后继节点
        cur->next = pre;
        pre = cur, cur = t;
    }
    return pre;
}
int main()
{
  for(int i = 0; i < 5; i++)  add(i);
  head = reverseList(head);
  print();
  return 0;
}

1、什么是堆?

堆(Heap)是一类特殊的数据结构,是最高效的优先级队列。堆通常是一个可以被看做一棵完全二叉树的数组对象。

堆也有不同种类的堆,分别为小根堆与大根堆:

  • 对于一棵二叉树,若树中的任意一个节点的权值都小于等于其父节点的权值,则称该二叉树满足"大根堆性质"。满足"大根堆性质"的完全二叉树就是"大根堆"
  • 若树中任意一个节点的权值都大于等于其父节点的权值,则称该二叉树满足"小根堆性质"。满足"小根堆性质"的二叉树就是"小根堆"

如图所示:


适用范围:海量数据求前n大或者前n小,并且n比较小,堆可以放入内存

基本原理及要点:最大堆求前n小,最小堆求前n大。比如:最大堆求前n小:

  • 第一步:将前n个数据都放入堆中,维护一个大小为n的最大堆;
  • 第二步:遍历之后的数据,比较当前元素与最大堆里的最大元素(堆顶元素),如果它小于最大元素,则应该替换那个最大元素;
  • 第三步:将整个最大堆扫描一遍即可得到所有的前n元素。

扩展:双堆,一个最大堆与一个最小堆结合,可以用来维护中位数。

2、手写堆(小根堆)

小根堆:父节点比左右儿子小。

一个堆的基本功能:

  1. 插入一个数;
  2. 求集合当中的最小值;
  3. 删除最小值;
  4. 删除任意一个元素;
  5. 修改任意一个元素。

如何存贮堆?

考虑用数组来存贮二叉树,根结点下标为1 (不建议从0开始),假设父节点编号为x,那么x的左儿子为:2x,x的右儿子为2x + 1。

2个基本操作:

  • down(x)操作:下沉一个节点,自上往下维护这个堆。将当前点x跟它的儿子们相比,把最小的作为新的父结点,也就是继续一次交换,然后递归下去,直到不需要进行交换,就能保持小根堆为止。
  • up(x)操作:上升一个节点,自下往上维护这个堆。up(x)只需要把x和它的父结点x / 2进行比较就行,如果父结点较大,那就交换他们俩,然后递归下去,直到不需要交换为止。

如何实现一个堆的基本功能?

  1. 插入一个数x:heap[++size] = x; up(size);
  2. 求集合中的最小值 : heap[1];
  3. 删除最小值: heap[1] = heap[size]; size--; down(1);
  4. 删除集合中任意一个数: heap[k] = heap[size]; size--; down(k); up(k);
  5. 修改集合中任意一个数: heap[k] = x; down(k); up(k);


down()和up()函数代码模版:

// 值变化,往下维护
void down(int u) {
    int t = u;   // t记录最小值
    if (2 * u <= size_ && h[2 * u] < h[t]) t = 2 * u; // 左儿子存在,且值比父亲小
    if (2 * u + 1 <= size_ && h[2 * u + 1] < h[t]) t = 2 * u + 1; // // 右儿子存在,且值比父亲小
    if (t != u) {
        swap(h[t], h[u]);
        down(t); 
    }
    return;
}
// 值变化,往上维护
void up(int u) {
    if (u / 2 > 0 && h[u / 2] > h[u]) {
        swap(h[u / 2], h[u]);
        up(u / 2);
    }
    return;
}

题目要求:

维护一个集合,初始时集合为空,支持如下几种操作:

  1. I x,插入一个数 x;
  2. PM,输出当前集合中的最小值;
  3. DM,删除当前集合中的最小值(数据保证此时的最小值唯一);
  4. D k,删除第 k 个插入的数;
  5. C k x,修改第 k 个插入的数,将其变为 x;

输入格式:

  1. 第一行包含整数 N。
  2. 接下来 N 行,每行包含一个操作指令,操作指令为 I xPMDMD kC k x 中的一种。

输入样例:

8
I -10
PM
I -10
D 1
C 2 8
I 6
PM
DM

输出样例:

-10
6

C++完整代码

#include <iostream>
#include <algorithm>
#include <string.h>
using namespace std;
const int N = 100010;
int h[N], ph[N], hp[N], cnt;
void heap_swap(int a, int b)
{
    swap(ph[hp[a]],ph[hp[b]]);
    swap(hp[a], hp[b]);
    swap(h[a], h[b]);
}
void down(int u)
{
    int t = u;
    if (u * 2 <= cnt && h[u * 2] < h[t]) t = u * 2;
    if (u * 2 + 1 <= cnt && h[u * 2 + 1] < h[t]) t = u * 2 + 1;
    if (u != t)
    {
        heap_swap(u, t);
        down(t);
    }
}
void up(int u)
{
    while (u / 2 && h[u] < h[u / 2])
    {
        heap_swap(u, u / 2);
        u >>= 1;
    }
}
int main()
{
    int n, m = 0;
    scanf("%d", &n);
    while (n -- )
    {
        char op[5];
        int k, x;
        scanf("%s", op);
        if (!strcmp(op, "I"))
        {
            scanf("%d", &x);
            cnt ++ ;
            m ++ ;
            ph[m] = cnt, hp[cnt] = m;
            h[cnt] = x;
            up(cnt);
        }
        else if (!strcmp(op, "PM")) printf("%d\n", h[1]);
        else if (!strcmp(op, "DM"))
        {
            heap_swap(1, cnt);
            cnt -- ;
            down(1);
        }
        else if (!strcmp(op, "D"))
        {
            scanf("%d", &k);
            k = ph[k];
            heap_swap(k, cnt);
            cnt -- ;
            up(k);
            down(k);
        }
        else
        {
            scanf("%d%d", &k, &x);
            k = ph[k];
            h[k] = x;
            up(k);
            down(k);
        }
    }
    return 0;
}


相关文章
|
2月前
|
存储 算法 Java
散列表的数据结构以及对象在JVM堆中的存储过程
本文介绍了散列表的基本概念及其在JVM中的应用,详细讲解了散列表的结构、对象存储过程、Hashtable的扩容机制及与HashMap的区别。通过实例和图解,帮助读者理解散列表的工作原理和优化策略。
43 1
散列表的数据结构以及对象在JVM堆中的存储过程
|
2月前
|
存储 算法 Perl
数据结构实验之链表
本实验旨在掌握线性表中元素的前驱、后续概念及链表的建立、插入、删除等算法,并分析时间复杂度,理解链表特点。实验内容包括循环链表应用(约瑟夫回环问题)、删除单链表中重复节点及双向循环链表的设计与实现。通过编程实践,加深对链表数据结构的理解和应用能力。
64 4
|
8天前
|
数据库
数据结构中二叉树,哈希表,顺序表,链表的比较补充
二叉搜索树,哈希表,顺序表,链表的特点的比较
数据结构中二叉树,哈希表,顺序表,链表的比较补充
|
2月前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
68 5
|
2月前
|
存储 搜索推荐 算法
【数据结构】树型结构详解 + 堆的实现(c语言)(附源码)
本文介绍了树和二叉树的基本概念及结构,重点讲解了堆这一重要的数据结构。堆是一种特殊的完全二叉树,常用于实现优先队列和高效的排序算法(如堆排序)。文章详细描述了堆的性质、存储方式及其实现方法,包括插入、删除和取堆顶数据等操作的具体实现。通过这些内容,读者可以全面了解堆的原理和应用。
104 16
|
2月前
|
存储 C语言
【数据结构】手把手教你单链表(c语言)(附源码)
本文介绍了单链表的基本概念、结构定义及其实现方法。单链表是一种内存地址不连续但逻辑顺序连续的数据结构,每个节点包含数据域和指针域。文章详细讲解了单链表的常见操作,如头插、尾插、头删、尾删、查找、指定位置插入和删除等,并提供了完整的C语言代码示例。通过学习单链表,可以更好地理解数据结构的底层逻辑,提高编程能力。
114 4
|
2月前
|
算法 安全 搜索推荐
2024重生之回溯数据结构与算法系列学习之单双链表精题详解(9)【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构王道第2.3章之IKUN和I原达人之数据结构与算法系列学习x单双链表精题详解、数据结构、C++、排序算法、java、动态规划你个小黑子;这都学不会;能不能不要给我家鸽鸽丢脸啊~除了会黑我家鸽鸽还会干嘛?!!!
|
2月前
|
存储 Web App开发 算法
2024重生之回溯数据结构与算法系列学习之单双链表【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构之单双链表按位、值查找;[前后]插入;删除指定节点;求表长、静态链表等代码及具体思路详解步骤;举例说明、注意点及常见报错问题所对应的解决方法
|
2月前
|
算法
数据结构之购物车系统(链表和栈)
本文介绍了基于链表和栈的购物车系统的设计与实现。该系统通过命令行界面提供商品管理、购物车查看、结算等功能,支持用户便捷地管理购物清单。核心代码定义了商品、购物车商品节点和购物车的数据结构,并实现了添加、删除商品、查看购物车内容及结算等操作。算法分析显示,系统在处理小规模购物车时表现良好,但在大规模购物车操作下可能存在性能瓶颈。
53 0
|
3月前
|
存储 Java
数据结构第三篇【链表的相关知识点一及在线OJ习题】
数据结构第三篇【链表的相关知识点一及在线OJ习题】
34 7