acwing算法提高之数据结构--平衡树Treap

devtools/2024/10/18 14:25:29/

目录

  • 1 介绍
  • 2 训练

1 介绍

本博客用来记录使用平衡树求解的题目。

插入、删除、查询操作的时间复杂度都是O(logN)

动态维护一个有序序列。

2 训练

题目1:253普通平衡树

C++代码如下,

#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>using namespace std;const int N = 100010, INF = 1e8;int n;
struct Node {int l, r;int key, val;int cnt, size;
}tr[N];int root, idx;void pushup(int p) {tr[p].size = tr[tr[p].l].size + tr[tr[p].r].size + tr[p].cnt;
}int get_node(int key) {tr[++idx].key = key;tr[idx].val = rand();tr[idx].cnt = tr[idx].size = 1;return idx;
}void zig(int &p) {int q = tr[p].l;tr[p].l = tr[q].r, tr[q].r = p, p = q;pushup(tr[p].r), pushup(p);
}void zag(int &p) {int q = tr[p].r;tr[p].r = tr[q].l, tr[q].l = p, p = q;pushup(tr[p].l), pushup(p);
}void build() {get_node(-INF), get_node(INF);root = 1, tr[1].r = 2;pushup(root);if (tr[1].val < tr[2].val) zag(root);
}void insert(int &p, int key) {if (!p) p = get_node(key);else if (tr[p].key == key) tr[p].cnt++;else if (tr[p].key > key) {insert(tr[p].l, key);if (tr[tr[p].l].val > tr[p].val) zig(p);} else {insert(tr[p].r, key);if (tr[tr[p].r].val > tr[p].val) zag(p);}pushup(p);
}void remove(int &p, int key) {if (!p) return;if (tr[p].key == key) {if (tr[p].cnt > 1) tr[p].cnt --;else if (tr[p].l || tr[p].r) {if (!tr[p].r || tr[tr[p].l].val > tr[tr[p].r].val) {zig(p);remove(tr[p].r, key);} else {zag(p);remove(tr[p].l, key);}} else {p = 0;}} else if (tr[p].key > key) remove(tr[p].l, key);else remove(tr[p].r, key);pushup(p);
}int get_rank_by_key(int p, int key) {if (!p) return 0;if (tr[p].key == key) return tr[tr[p].l].size + 1;if (tr[p].key > key) return get_rank_by_key(tr[p].l, key);return tr[tr[p].l].size + tr[p].cnt + get_rank_by_key(tr[p].r, key);
}int get_key_by_rank(int p, int rank) {if (!p) return INF;if (tr[tr[p].l].size >= rank) return get_key_by_rank(tr[p].l, rank);if (tr[tr[p].l].size + tr[p].cnt >= rank) return tr[p].key;return get_key_by_rank(tr[p].r, rank - tr[tr[p].l].size - tr[p].cnt);
}int get_prev(int p, int key) {if (!p) return -INF;if (tr[p].key >= key) return get_prev(tr[p].l, key);return max(tr[p].key, get_prev(tr[p].r, key));
}int get_next(int p, int key) {if (!p) return INF;if (tr[p].key <= key) return get_next(tr[p].r, key);return min(tr[p].key, get_next(tr[p].l, key));
}int main() {build();scanf("%d", &n);while (n--) {int opt, x;scanf("%d%d", &opt, &x);if (opt == 1) insert(root, x);else if (opt == 2) remove(root, x);else if (opt == 3) printf("%d\n", get_rank_by_key(root, x) - 1);else if (opt == 4) printf("%d\n", get_key_by_rank(root, x + 1));else if (opt == 5) printf("%d\n", get_prev(root, x));else printf("%d\n", get_next(root, x));        }return 0;
}

使用C++的multiset,超时了,通过了 6/11个数据

#include <iostream>
#include <cstring>
#include <algorithm>
#include <set>using namespace std;int main() {multiset<int> s;int m;cin >> m;while (m--) {int op, x;cin >> op >> x;if (op == 1) {s.insert(x);} else if (op == 2) {s.extract(x);} else if (op == 3) {int idx = distance(s.begin(), s.lower_bound(x));cout << 1 + idx << endl;} else if (op == 4) {x -= 1; //下标从0开始auto it = s.begin();advance(it, x);cout << *it << endl;} else if (op == 5) {auto it = s.lower_bound(x);it--; //保证一定有解cout << *it << endl;} else if (op == 6) {auto it = s.upper_bound(x);cout << *it << endl;}}return 0;
}

使用python3的库sortedcontainers,发现acwing报错没有该模块,

在这里插入图片描述

from sortedcontainers import SortedList
import bisectsl = SortedList()m = int(input())
for i in range(m):line = intput()ls_line = line.split()ls_line = [int(x) for x in ls_line]op = ls_line[0]x = ls_line[1]if op == 1:sl.add(x)elif op == 2:sl.remove(x)elif op == 3:i = bisect.bisect_left(sl, x)print(i + 1)elif op == 4:print(sl[x-1])elif op == 5:it = bisect.bisect_left(ls, x)it -= 1print(sl[it])elif op == 6:it = biset.bisect_right(ls, x)print(sl[it])

题目2:265营业额统计

使用C++的multiset

#include <iostream>
#include <cstring>
#include <algorithm>
#include <set>
#include <climits>using namespace std;int main() {int n;cin >> n;long long res = 0;multiset<int> s;for (int i = 0; i < n; ++i) {int t;cin >> t;if (i == 0) {res += t;s.insert(t); //插入数tcontinue;}int ans = 0x3f3f3f3f;//求大于等于t的迭代器auto iter1 = s.lower_bound(t);if (iter1 != s.end()) {int x = *iter1;ans = min(ans, abs(x - t));}if (iter1 != s.begin()) {auto iter2 = iter1;iter2--;int x = *iter2;ans = min(ans, abs(x - t));}res += ans;s.insert(t); //插入数t}cout << res << endl;return 0;
}

C++代码如下,

#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>using namespace std;typedef long long LL;const int N = 33010, INF = 1e7;int n;
struct Node {int l, r;int key, val;
}tr[N];int root, idx;int get_node(int key) {tr[++idx].key = key;tr[idx].val = rand();return idx;
}void build() {get_node(-INF), get_node(INF);root = 1, tr[1].r = 2;
}void zig(int &p) {int q = tr[p].l;tr[p].l = tr[q].r, tr[q].r = p, p = q;
}void zag(int &p) {int q = tr[p].r;tr[p].r = tr[q].l, tr[q].l = p, p = q;
}void insert(int &p, int key) {if (!p) p = get_node(key);else if (tr[p].key == key) return;else if (tr[p].key > key) {insert(tr[p].l, key);if (tr[tr[p].l].val > tr[p].val) zig(p);} else {insert(tr[p].r, key);if (tr[tr[p].r].val > tr[p].val) zag(p);}
}int get_prev(int p, int key) {if (!p) return -INF;if (tr[p].key > key) return get_prev(tr[p].l, key);return max(tr[p].key, get_prev(tr[p].r, key));
}int get_next(int p, int key) {if (!p) return INF;if (tr[p].key < key) return get_next(tr[p].r, key);return min(tr[p].key, get_next(tr[p].l, key));
}int main() {build();scanf("%d", &n);LL res = 0;for (int i = 1; i <= n; ++i) {int x;scanf("%d", &x);if (i == 1) res += x;else res += min(x - get_prev(root, x), get_next(root, x) - x);insert(root, x);}printf("%lld\n", res);return 0;
}

http://www.ppmy.cn/devtools/38218.html

相关文章

ubuntu20中ros与anaconda的python版本冲突问题

系统环境 原本系统是ubuntu20 noetic&#xff0c;python都在/usr/bin中&#xff0c;一共是两个版本的python&#xff0c;一个是python3.8&#xff0c;另一个是python2.7。 问题发现 当安装anaconda后&#xff0c;并且将anaconda的bin目录加入到系统环境中时候&#xff0c;…

牛客NC97 字符串出现次数的TopK问题【中等 哈希+优先级队列 Java/Go】

题目 题目链接&#xff1a; https://www.nowcoder.com/practice/fd711bdfa0e840b381d7e1b82183b3ee 核心 哈希&#xff0c;优先级队列Java代码 import java.util.*;public class Solution {/*** 代码中的类名、方法名、参数名已经指定&#xff0c;请勿修改&#xff0c;直接返…

计算方法实验2(补充):列主元消元法解线性方程组

C源代码 #include<bits/stdc.h> using namespace std;// 列主元消去法求解线性方程组 vector<long double> Column_Elimination(vector<vector<long double>> A, vector<long double> b);int main() {vector<vector<long double>> …

Ansible——playbook编写

一、简介 1.什么是playbook Ansible Playbook 是设定自动化任务的一种蓝图&#xff0c;可在无需人工干预或有限干预的前提下执行复杂的 IT 操作。Ansible Playbook 对一组或一类共同构成 Ansible 清单的主机执行。 Ansible Playbook 本质上是一些框架&#xff0c;是一些预先编…

VxTerm使用教程:连接SSH服务端设备,什么是SSH

一、什么是SSH&#xff1f; <摘自百度> 安全外壳协议 SSH&#xff0c;即安全外壳协议&#xff08;Secure Shell&#xff09;&#xff0c;是一种网络协议&#xff0c;用于在计算机网络上提供安全的远程登录和命令执行功能。 SSH通过加密通信通道来保护数据传输&#xff0c…

STM32中的Systick的使用

SysTick&#xff0c;全称System Tick Timer&#xff0c;是Cortex-M microcontrollers内核中提供的一个简单而有效的系统定时器&#xff0c;设计用来给操作系统提供时间基准&#xff0c;或用于生成周期性的中断。STM32系列微控制器&#xff0c;作为基于ARM Cortex-M内核的设备&a…

Java | Leetcode Java题解之第66题加一

题目&#xff1a; 题解&#xff1a; class Solution {public int[] plusOne(int[] digits) {int n digits.length;for (int i n - 1; i > 0; --i) {if (digits[i] ! 9) {digits[i];for (int j i 1; j < n; j) {digits[j] 0;}return digits;}}// digits 中所有的元素…

Docker 中快速构建 Redis Cluster 集群

Docker 中快速构建 Redis Cluster 集群 目录 前言环境准备 所需软件配置网络 构建 Redis Cluster 镜像 创建自定义 Dockerfile构建镜像 启动 Redis 节点容器 启动命令 配置 Redis Cluster 集群 创建 Redis 集群验证集群状态 总结 前言 Redis 是一个高性能的键值对数据库&am…