洛谷 [P3224] 永无乡
Treap 的合并
首先感谢 @Capella 的DeBug
其次,这是由一个 & 号引发的血案
注意对于所有修改操作都要 &
Treap的合并, 启发式合并,对于每一个节点都 insert ,注意垃圾回收
#include <iostream>
#include <cstdio>
#include <cstring>
#include <cmath>
#include <algorithm>
#define ll long long
#define inf 0x3f3f3f3f
using namespace std;
const int MAXN = 500005;
int init() {
int rv = 0, fh = 1;
char c = getchar();
while(c < '0' || c > '9') {
if(c == '-') fh = -1;
c = getchar();
}
while(c >= '0' && c <= '9') {
rv = (rv<<1) + (rv<<3) + c - '0';
c = getchar();
}
return fh * rv;
}
int pool[MAXN], top, n, m, id[MAXN], q;
struct UFS {
int fa[MAXN];
void unit() {
for(int i = 1; i <= n; i++) fa[i] = i;
}
int find(int x) {
if(x != fa[x]) return fa[x] = find(fa[x]);
return fa[x];
}
}ufs;
struct node{
int val, dat, num, size, l, r;
}a[MAXN];
struct Treap{
int New(int x, int num) {
int ret = pool[++top];
a[ret].val = x;
a[ret].num = num;
a[ret].dat = rand();
a[ret].size = 1;
a[ret].l = a[ret].r = 0;
return ret;
}
void del(int &rt) {pool[top--] = rt;rt = 0;}
void PushUp(int rt) {
a[rt].size = a[a[rt].l].size + a[a[rt].r].size + 1;
}
void zig(int &rt) {
int nxt = a[rt].l;
a[rt].l = a[nxt].r; a[nxt].r = rt; rt = nxt;
PushUp(a[rt].r);
PushUp(rt);
}
void zag(int &rt) {
int nxt = a[rt].r;
a[rt].r = a[nxt].l; a[nxt].l = rt; rt = nxt;
PushUp(a[rt].l);
PushUp(rt);
}
void Insert(int &rt, int x, int num) {
if(!rt) {
rt = New(x, num);
return;
}
if(x < a[rt].val) {
Insert(a[rt].l, x, num);
if(a[a[rt].l].dat > a[rt].dat) zig(rt);
}else {
Insert(a[rt].r, x, num);
if(a[a[rt].r].dat > a[rt].dat) zag(rt);
}
PushUp(rt);
}
int GetNum(int rt, int x) {
if(!rt) return -1;
if(a[a[rt].l].size >= x) return GetNum(a[rt].l, x);
if(a[a[rt].l].size + 1 >=x) return a[rt].num;
return GetNum(a[rt].r,x - 1 - a[a[rt].l].size);
}
void merge(int /*x*/&x, int &y) {
if(!y) return;
if(a[y].l)
merge(x, a[y].l);
if(a[y].r)
merge(x, a[y].r);
Insert(x, a[y].val, a[y].num);del(y);
}
}bst;
void merge(int u, int v) {
int r1 = ufs.find(u), r2 = ufs.find(v);
if(r1 != r2) {
if(a[id[r1]].size < a[id[r2]].size) {
ufs.fa[r1] = r2;
bst.merge(id[r2], id[r1]);
}else {
ufs.fa[r2] = r1;
bst.merge(id[r1], id[r2]);
}
}
}
int main() {
n = init(); m = init();
ufs.unit();
for(int i = 0; i < MAXN; i++) pool[i] = i;
for(int i = 1; i <= n; i++) {
int t = init();
id[i] = bst.New(t, i);
}
for(int i = 1; i <= m; i++) {
int u = init(), v = init();
if(!u || !v) continue;
merge(u, v);
}
q = init();
for(int i = 1; i <= q; i++) {
char t;
scanf(" %c ", &t);
if(t == 'Q') {
int x = init(), k = init();
printf("%d\n", bst.GetNum(id[ufs.find(x)], k));
}else {
int u = init(), v = init();
if(!u || !v) continue;
merge(u, v);
}
}
return 0;
}
洛谷 [P3224] 永无乡的更多相关文章
- 洛谷P3224 永无乡 [HNOI2012] 线段树/splay/treap
正解:线段树合并 解题报告: 传送门! 这题也是有很多解法,eg:splay,treap,... 然而我都不会我会学的QAQ! 反正今天就只讲下线段树合并怎么做QAQ 首先看到这样子的说第k重要的是什 ...
- 【Luogu】P3224永无乡(splay)
题目链接 splay模板,启发式合并(其实就是暴力插入)即可. 顺便吐槽时限,带垃圾回收而已……不至于最后一个点死活不让过吧? #include<cstdio> #include<c ...
- 洛谷 P3224 [HNOI2012]永无乡 解题报告
P3224 [HNOI2012]永无乡 题目描述 永无乡包含 \(n\) 座岛,编号从 \(1\) 到 \(n\) ,每座岛都有自己的独一无二的重要度,按照重要度可以将这 \(n\) 座岛排名,名次用 ...
- 洛谷P3224 [HNOI2012]永无乡(线段树合并+并查集)
题目描述 永无乡包含 nnn 座岛,编号从 111 到 nnn ,每座岛都有自己的独一无二的重要度,按照重要度可以将这 nnn 座岛排名,名次用 111 到 nnn 来表示.某些岛之间由巨大的桥连接, ...
- 洛谷 P3224 [HNOI2012]永无乡
题面 永无乡包含 \(n\) 座岛,编号从 \(1\) 到 \(n\) ,每座岛都有自己的独一无二的重要度,按照重要度可以将这 \(n\) 座岛排名,名次用 \(1\) 到 \(n\) 来表示.某些岛 ...
- 线段树合并+并查集 || BZOJ 2733: [HNOI2012]永无乡 || Luogu P3224 [HNOI2012]永无乡
题面:P3224 [HNOI2012]永无乡 题解: 随便写写 代码: #include<cstdio> #include<cstring> #include<iostr ...
- 【题解】永无乡 [HNOI2012] [BZOJ2733] [P3224]
[题解]永无乡 [HNOI2012] [BZOJ2733] [P3224] [题目描述] 永无乡包含 \(n\) 座岛,编号从 \(1\) 到 \(n\) ,每座岛都有自己的独一无二的重要度,按照重要 ...
- P3224 [HNOI2012]永无乡 题解
P3224 [HNOI2012]永无乡 题解 题意概括 有若干集合,每个集合最初包含一个值,和一个编号1~n.两个操作:合并两个集合,查询包含值x的集合中第k大值最初的集合编号. 思路 维护集合之间关 ...
- bzoj2733 / P3224 [HNOI2012]永无乡(并查集+线段树合并)
[HNOI2012]永无乡 每个联通块的点集用动态开点线段树维护 并查集维护图 合并时把线段树也合并就好了. #include<iostream> #include<cstdio&g ...
随机推荐
- charles连接手机抓包--------最详细的步骤
首先确保电脑和手机连接到同一个热点上 电脑连接热点以后,首先打开Charles设置Charles的setting port一般都默认8888 Enable transparent HTTP proxy ...
- k8s学习目录
目录 K8S基础部分 基础部分 5 秒创建 k8s 集群[转] k8s 核心功能[转] k8s 重要概念[转] 部署 k8s Cluster(上)[转] 部署 k8s Cluster(下)[转] Ku ...
- 判断一个链表是否为回文结构 【题目】 给定一个链表的头节点head,请判断该链表是否为回 文结构。 例如: 1->2->1,返回true。 1->2->2->1,返回true。 15->6->15,返回true。 1->2->3,返回false。 进阶: 如果链表长度为N,时间复杂度达到O(N),额外空间复杂 度达到O(1)。
方式1:借助栈 空间辅助度是O(N) 方式2: 借助栈 空间复杂度是 O(n/2).只存后半个链表 方式3: 反转后半个链表 最后再反转回来 package my_basic.class_3; im ...
- jExcelAPI导入导出excel
MS的电子表格(Excel)是Office的重要成员,是保存统计数据的一种常用格式.作为办公文档,势必要涉及到的电子文档的交换,Excel是一种在企业中非常通用的文件格式,打印和管理也比较方便.在 ...
- Bootstrap历练实例:带有下拉菜单的标签和胶囊导航
<!DOCTYPE html><html><head><meta http-equiv="Content-Type" content=&q ...
- javase(1)_基础语法
一.java概述 1.Java语言特点:纯面向对象(一切皆对象),平台无关(JVM屏蔽底层运行平台的差异),不同的平台有不同的JVM,JVM将程序翻译成当前操作系统能执行的程序,一次编译到处运行),健 ...
- CF-1072-C. Cram Time(贪心,数学)
CF-1072-C. Cram Time http://codeforces.com/contest/1072/problem/C 题意: 第一天有 a 小时,第二天有 b 小时.第 k 个任务需要 ...
- 【OS_Linux】三大文本处理工具之grep命令
grep(global search regular expression(RE) and print out the line,整行搜索并打印匹配成功的行 语法:grep [选项] 搜索词 ...
- shell 管道导致的变量重置问题
测试脚本: #!/bin/sh flag= func() { flag= } main() { func | echo "flag=$flag" } 输出显示的flag=0! 参考 ...
- Vue—事件修饰符
Vue事件修饰符 Vue.js 为 v-on 提供了事件修饰符来处理 DOM 事件细节,如:event.preventDefault() 或 event.stopPropagation(). Vue. ...