bzoj4399 魔法少女LJJ 线段树合并
只看题面绝对做不出系列....
注意到\(c \leqslant 7\),因此不会有删边操作(那样例删边干嘛)
注意到\(2, 5\)操作十分的有趣,启示我们拿线段树合并来做
操作\(7\)很好处理
操作\(6\),维护对数的和即可
操作\(3, 4\),乍看不好处理,然而势能分析一下就可以得出暴力的复杂度是\(O(n \log n)\)的
然而我好像写了个稳定的\(\log\)维护
然后好像就没了诶......
空间直接动态开点是开不下的....
需要预先离散化权值
复杂度\(O(n \log n)\)
#include <cmath>
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;
#define de double
#define ri register int
#define rep(io, st, ed) for(ri io = st; io <= ed; io ++)
#define gc getchar
inline int read() {
int p = 0, w = 1; char c = gc();
while(c > '9' || c < '0') { if(c == '-') w = -1; c = gc(); }
while(c >= '0' && c <= '9') p = p * 10 + c - '0', c = gc();
return p * w;
}
const int sid = 5e5 + 5;
const int eid = 5e6 + 5;
de mul[eid];
int n, m, id, nc;
int rt[sid], fa[sid];
int ls[eid], rs[eid], sz[eid];
int opt[sid], c1[sid], c2[sid], T[sid];
inline int find(int o) { return fa[o] = (o == fa[o]) ? o : find(fa[o]); }
inline void upd(int o) {
int lc = ls[o], rc = rs[o];
sz[o] = sz[lc] + sz[rc];
mul[o] = mul[lc] + mul[rc];
}
inline int merge(int x, int y) {
if(!x || !y) return x + y;
ls[x] = merge(ls[x], ls[y]);
rs[x] = merge(rs[x], rs[y]);
sz[x] = sz[x] + sz[y];
mul[x] = mul[x] + mul[y];
return x;
}
inline void mdf(int &o, int l, int r, int c, int v) {
if(!o) o = ++ id;
if(l == r) { sz[o] = v; mul[o] = (de)v * (de)log(T[c]); return; }
int mid = (l + r) >> 1;
if(c <= mid) mdf(ls[o], l, mid, c, v);
else mdf(rs[o], mid + 1, r, c, v);
upd(o);
}
inline int dfs(int &o, int l, int r, int ml, int mr) {
if(!o || ml > r || mr < l) return 0;
if(ml <= l && mr >= r) {
int tmp = sz[o]; o = 0;
return tmp;
}
int mid = (l + r) >> 1;
int ret = dfs(ls[o], l, mid, ml, mr) + dfs(rs[o], mid + 1, r, ml, mr);
upd(o); return ret;
}
inline int qry(int o, int l, int r, int k) {
if(l == r) return T[l];
int mid = (l + r) >> 1;
if(sz[ls[o]] >= k) return qry(ls[o], l, mid, k);
else return qry(rs[o], mid + 1, r, k - sz[ls[o]]);
}
void calc() {
rep(i, 1, m) {
opt[i] = read(); c1[i] = read();
if(opt[i] != 1 && opt[i] != 7) c2[i] = read();
if(opt[i] == 1) T[++ nc] = c1[i];
if(opt[i] == 3 || opt[i] == 4) T[++ nc] = c2[i];
}
sort(T + 1, T + nc + 1);
nc = unique(T + 1, T + nc + 1) - T - 1;
rep(i, 1, m) {
if(opt[i] == 1)
c1[i] = lower_bound(T + 1, T + nc + 1, c1[i]) - T;
if(opt[i] == 3 || opt[i] == 4)
c2[i] = lower_bound(T + 1, T + nc + 1, c2[i]) - T;
}
rep(i, 1, m) {
int u, v, w, num;
switch(opt[i]) {
case 1 :
n ++; fa[n] = n;
mdf(rt[n], 1, nc, c1[i], 1); break;
case 2 :
u = find(c1[i]); v = find(c2[i]);
if(u == v) break;
fa[v] = u; rt[u] = merge(rt[u], rt[v]); break;
case 3 :
u = find(c1[i]); w = c2[i];
num = dfs(rt[u], 1, nc, 1, w);
mdf(rt[u], 1, nc, w, num); break;
case 4 :
u = find(c1[i]); w = c2[i];
num = dfs(rt[u], 1, nc, w, nc);
mdf(rt[u], 1, nc, w, num); break;
case 5 :
u = find(c1[i]); w = c2[i];
printf("%d\n", qry(rt[u], 1, nc, w)); break;
case 6 :
u = find(c1[i]); v = find(c2[i]);
if(mul[rt[u]] > mul[rt[v]]) puts("1");
else puts("0"); break;
case 7 :
u = find(c1[i]);
printf("%d\n", sz[rt[u]]); break;
default : break;
}
}
}
int main() {
//freopen("4399.in", "r", stdin);
//freopen("4399.out", "w", stdout);
m = read();
calc();
return 0;
}
bzoj4399 魔法少女LJJ 线段树合并的更多相关文章
- BZOJ4399魔法少女LJJ——线段树合并+并查集
题目描述 在森林中见过会动的树,在沙漠中见过会动的仙人掌过后,魔法少女LJJ已经觉得自己见过世界上的所有稀奇古怪的事情了LJJ感叹道“这里真是个迷人的绿色世界,空气清新.淡雅,到处散发着醉人的奶浆味: ...
- bzoj4399 魔法少女LJJ 线段树合并+线段树二分+并查集
题目传送门 https://lydsy.com/JudgeOnline/problem.php?id=4399 题解 毒瘤题 \(9\) 种操作还有支持动态图的连通性 仔细读题 $ c<=7$. ...
- 【BZOJ4399】魔法少女LJJ 线段树合并
[BZOJ4399]魔法少女LJJ Description 在森林中见过会动的树,在沙漠中见过会动的仙人掌过后,魔法少女LJJ已经觉得自己见过世界上的所有稀奇古怪的事情了LJJ感叹道“这里真是个迷人的 ...
- BZOJ 4399: 魔法少女LJJ 线段树合并 + 对数
Description 在森林中见过会动的树,在沙漠中见过会动的仙人掌过后,魔法少女LJJ已经觉得自己见过世界上的所有稀奇古怪的事情了LJJ感叹道“这里真是个迷人的绿色世界,空气清新.淡雅,到处散发着 ...
- BZOJ.4399.魔法少女LJJ(线段树合并)
BZOJ 注意\(c\leq7\)→_→ 然后就是裸的权值线段树+线段树合并了. 对于取\(\max/\min\)操作可以直接区间修改清空超出范围的值,然后更新到对应位置上就行了(比如对\(v\)取\ ...
- 魔法少女 LJJ——线段树
题目 [题目描述] 在森林中见过会动的树,在沙漠中见过会动的仙人掌过后,魔法少女 LJJ 已经觉得自己见过世界上的所有稀奇古怪的事情了. LJJ 感叹道“这里真是个迷人的绿色世界,空气清新.淡雅,到处 ...
- BZOJ 4399: 魔法少女LJJ(线段树)
传送门 解题思路 出题人真会玩..操作\(2\)线段树合并,然后每棵线段树维护元素个数和.对于\(6\)这个询问,因为乘积太大,所以要用对数.时间复杂度\(O(nlogn)\) 代码 #include ...
- BZOJ4399 魔法少女LJJ【线段树合并】【并查集】
Description 在森林中见过会动的树,在沙漠中见过会动的仙人掌过后,魔法少女LJJ已经觉得自己见过世界上的所有稀奇古怪的事情了 LJJ感叹道"这里真是个迷人的绿色世界,空气清新.淡雅 ...
- BZOJ4399 魔法少女LJJ(线段树合并)
注意到只有增加点/合并的操作.这些操作都可以用线段树完成,于是线段树合并一发就好了.注意乘积大小直接比较肯定会炸,取个对数即可.数据中存在重边. #include<iostream> #i ...
随机推荐
- 【译】第十二篇 Integration Services:高级日志记录
本篇文章是Integration Services系列的第十二篇,详细内容请参考原文. 简介在前一篇文章我们配置了SSIS内置日志记录,演示了简单和高级日志配置,保存并查看日志配置,生成自定义日志消息 ...
- 新电脑重新安装win10+python3.6+anaconda+tensorflow1.12(gpu版)
安装了一天的软件,遇到了很多坑,在快泪崩的时候,始终以磨刀不误砍柴工鼓励自己,坚持安好了,话不多说,上干货: 前言: TensorFlow 有两个版本:CPU 版本和 GPU 版本.GP ...
- 图片压缩之 PNG
作者:程志达链接:https://zhuanlan.zhihu.com/p/19570424来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非商业转载请注明出处. PNG(Portable N ...
- CentOS_5.5_安装GCC编译LiME
1 概述 近期遇到个使用CentOS 5.5的系统,生产环境没有GCC.GDB.要对这台机器抓取关键内存回去用volatility分析. 思路1:使用工具Dump某个进程的内存.使用cat /proc ...
- python操作mysql(pymysql + sqlalchemy)
pymysql pymsql是Python中操作MySQL的模块,其使用方法和MySQLdb几乎相同. 下载安装 pip3 install pymysql 使用操作 1.执行sql #!/usr/bi ...
- java 二叉树遍历
package com.lever; import java.util.LinkedList;import java.util.Queue; /** * 二叉树遍历 * @author lckxxy ...
- 区间DP小结
也写了好几天的区间DP了,这里稍微总结一下(感觉还是不怎么会啊!). 但是多多少少也有了点感悟: 一.在有了一点思路之后,一定要先确定好dp数组的含义,不要模糊不清地就去写状态转移方程. 二.还么想好 ...
- HDU 2476 String painter(区间DP+思维)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2476 题目大意:给你字符串A.B,每次操作可以将一段区间刷成任意字符,问最少需要几次操作可以使得字符串 ...
- 更新svn的客户端TortoiseSVN后 ,之前使用svn管理的文件的关联图标消失了
说明:下面的解决方法及图片来自博客:装了SVN,你的关联图标变了没有? 解决办法:在同步的文件点击右键如下图 ... 现则Settings,出现的界面如下 ... ...
- HTML5练习1
制作简历 主要代码: <!doctype html> <html> <head> <meta charset="utf-8"> &l ...