魔法少女 LJJ——线段树
题目
【题目描述】
在森林中见过会动的树,在沙漠中见过会动的仙人掌过后,魔法少女 LJJ 已经觉得自己见过世界上的所有稀奇古怪的事情了。
LJJ 感叹道“这里真是个迷人的绿色世界,空气清新、淡雅,到处散发着醉人的奶浆味;小猴在枝头悠来荡去,好不自在;各式各样的鲜花争相开放,各种树枝的枝头挂满沉甸甸的野果;鸟儿的歌声婉转动听,小河里飘着落下的花瓣真是人间仙境”。
SHY 觉得 LJJ 还是太 naive,一天, SHY 带着自己心爱的图找到 LJJ,对 LJJ 说:“既然你已经见识过动态树,动态仙人掌了,那么今天就来见识一下动态图吧”。
LJJ:“要支持什么操作?”
SHY:
1. 新建一个节点,权值为 $x$。
2. 连接两个节点。
3. 将一个节点 $a$ 所属于的联通快内权值小于 $x$ 的所有节点权值变成 $x$。
4. 将一个节点 $a$ 所属于的联通快内权值大于 $x$ 的所有节点权值变成 $x$。
5. 询问一个节点 $a$ 所属于的联通块内的第 $k$ 小的权值是多少。
6. 询问一个节点 $a$ 所属联通快内所有节点权值之积与另一个节点 $b$ 所属联通快内所有节点权值之积的大小。
7. 询问 $a$ 所在联通块内节点的数量
8. 若两个节点 $a$,$b$ 直接相连,将这条边断开。
9. 若节点 $a$ 存在,将这个点删去。
LJJ:“我可以离线吗?”
SHY:“可以,每次操作是不加密的。”
LJJ:“我可以暴力吗?”
SHY:“自重。”
LJJ很郁闷,你能帮帮他吗?
【输入格式】
第一行一个整数 $n$ ,代表岛屿数量。
接下来 $n-1$ 行,每行三个整数 $u,v,w$ ,代表 $u$ 号岛屿和 $v$ 号岛屿由一条代价为 $c$ 的桥梁直接相连,保证 $1 \le u,v \le n$ 且 $1 \le c \le 100000 $ 。
第 $n+1$ 行,一个整数 $m$ ,代表敌方机器能使用的次数。
接下来 $m$ 行,每行一个整数 $k_i$ ,代表第 $i$ 次后,有 $k_i$ 个岛屿资源丰富,接下来 $k$ 个整数$h_1,h_2,…h_k$ ,表示资源丰富岛屿的编号。
【输出格式】
具体输出格式见样例 。
【样例输入】
12
1 2
1 3
1 4
1 5
1 6
2 1 2
2 2 3
2 3 4
2 4 5
9 1
3 2 5
5 3 4
【样例输出】
6
【数据范围与提示】
对 $100\%$ 的数据 $0\le m \le 400000,c \le 7$,所有出现的数均 $\le 1000000000$,所有出现的点保证存在.
题解
题目中 $ c \leq 7 $,即说明仅有前 $ 7 $ 个操作
对于操作一,新建一棵线段树,并插入这个点的值
对于操作二,合并两颗线段树,用并查集维护
对于操作三,查询个数并删除小于 $ x $ 的点,插入 $ x $
对于操作四,类似三
对于操作五,线段树上二分即可
对于操作六,可以发现 $ \log(a\times b)=\log a+\log b $,所以转化成 $ log x $ 比较和即可
对于操作七,查询整个线段树点的个数即可
于是这又是一道大力题
代码
#include<bits/stdc++.h>
#define LL long long
#define db double
#define _(d) while(d(isdigit(ch=getchar())))
using namespace std;
int R(){
int x;bool f=;char ch;_(!)if(ch=='-')f=;x=ch^;
_()x=(x<<)+(x<<)+(ch^);return f?x:-x;}
const int N=2e6+;
int n,m,tmp,cnt,fa[N],d[N],rot[N];
struct node{int op,x,y;}q[N];
struct seg{int ls,rs,num;bool tag;db val;}tr[N*];
int ef(int x){return lower_bound(d+,d++n,x)-d;}
int find(int x){return fa[x]==x?x:fa[x]=find(fa[x]);}
#define Ls(x) tr[x].ls
#define Rs(x) tr[x].rs
void pushup(int rt){
tr[rt].num=tr[Ls(rt)].num+tr[Rs(rt)].num;
tr[rt].val=tr[Ls(rt)].val+tr[Rs(rt)].val;
return;
}
void pushdown(int rt){
tr[rt].tag=;
tr[Ls(rt)].num=tr[Ls(rt)].val=,tr[Ls(rt)].tag=;
tr[Rs(rt)].num=tr[Rs(rt)].val=,tr[Rs(rt)].tag=;
return;
}
void insert(int &rt,int l,int r,int k,int x,db y){
if(!rt)rt=++cnt;
if(l==r){tr[rt].num+=x,tr[rt].val+=y;return;}
int mid=(l+r)>>;
if(tr[rt].tag)pushdown(rt);
if(k<=mid)insert(Ls(rt),l,mid,k,x,y);
else insert(Rs(rt),mid+,r,k,x,y);
pushup(rt);
return;
}
void merge(int &o1,int o2,int l,int r){
if(!o1||!o2){o1+=o2;return;}
if(l==r){
tr[o1].num+=tr[o2].num,tr[o1].val+=tr[o2].val;
return;
}
int mid=(l+r)>>;
if(tr[o1].tag)pushdown(o1);
if(tr[o2].tag)pushdown(o2);
merge(Ls(o1),Ls(o2),l,mid),merge(Rs(o1),Rs(o2),mid+,r);
pushup(o1);
return;
}
int query(int rt,int l,int r,int ql,int qr){
if(!rt||qr<ql)return ;
if(ql<=l&&qr>=r)return tr[rt].num;
int mid=(l+r)>>,res=;
if(tr[rt].tag)pushdown(rt);
if(ql<=mid)res=query(Ls(rt),l,mid,ql,qr);
if(qr>mid)res+=query(Rs(rt),mid+,r,ql,qr);
return res;
}
void clean(int rt,int l,int r,int ql,int qr){
if(!rt||qr<ql)return;
if(ql<=l&&qr>=r){
tr[rt].num=tr[rt].val=,tr[rt].tag=;
return;
}
int mid=(l+r)>>;
if(ql<=mid)clean(Ls(rt),l,mid,ql,qr);
if(qr>mid)clean(Rs(rt),mid+,r,ql,qr);
pushup(rt);
return;
}
int ask(int rt,int l,int r,int k){
if(l==r)return d[l];
int mid=(l+r)>>;
if(tr[rt].tag)pushdown(rt);
if(k<=tr[Ls(rt)].num)return ask(Ls(rt),l,mid,k);
else return ask(Rs(rt),mid+,r,k-tr[Ls(rt)].num);
}
int main(){
m=R();
for(int i=;i<=m;i++){
int op=R(),x=R(),y=;
if(op!=&&op!=)y=R();
q[i]=(node){op,x,y};
if(op==||op==)d[++n]=y;
if(op==)d[++n]=x;
}
sort(d+,d++n);
n=unique(d+,d++n)-d-;
for(int i=;i<=m;i++){
int op=q[i].op,x=q[i].x,y=q[i].y;
if(op==)insert(rot[++tmp],,n,ef(x),,(db)log(x)),fa[tmp]=tmp;
if(op==){
int fx=find(x),fy=find(y);
if(fx!=fy)fa[fy]=fx,merge(rot[fx],rot[fy],,n);
}
if(op==){
int fx=find(x),k=ef(y),sum=query(rot[fx],,n,,k-);
clean(rot[fx],,n,,k-),insert(rot[fx],,n,k,sum,sum*log(y));
}
if(op==){
int fx=find(x),k=ef(y),sum=query(rot[fx],,n,k+,n);
clean(rot[fx],,n,k+,n),insert(rot[fx],,n,k,sum,sum*log(y));
}
if(op==)printf("%d\n",ask(rot[find(x)],,n,y));
if(op==)puts(tr[rot[find(x)]].val>tr[rot[find(y)]].val?"":"");
if(op==)printf("%d\n",tr[rot[find(x)]].num);
}
return ;
}
魔法少女 LJJ——线段树的更多相关文章
- BZOJ4399魔法少女LJJ——线段树合并+并查集
题目描述 在森林中见过会动的树,在沙漠中见过会动的仙人掌过后,魔法少女LJJ已经觉得自己见过世界上的所有稀奇古怪的事情了LJJ感叹道“这里真是个迷人的绿色世界,空气清新.淡雅,到处散发着醉人的奶浆味: ...
- 【BZOJ4399】魔法少女LJJ 线段树合并
[BZOJ4399]魔法少女LJJ Description 在森林中见过会动的树,在沙漠中见过会动的仙人掌过后,魔法少女LJJ已经觉得自己见过世界上的所有稀奇古怪的事情了LJJ感叹道“这里真是个迷人的 ...
- BZOJ 4399: 魔法少女LJJ 线段树合并 + 对数
Description 在森林中见过会动的树,在沙漠中见过会动的仙人掌过后,魔法少女LJJ已经觉得自己见过世界上的所有稀奇古怪的事情了LJJ感叹道“这里真是个迷人的绿色世界,空气清新.淡雅,到处散发着 ...
- BZOJ.4399.魔法少女LJJ(线段树合并)
BZOJ 注意\(c\leq7\)→_→ 然后就是裸的权值线段树+线段树合并了. 对于取\(\max/\min\)操作可以直接区间修改清空超出范围的值,然后更新到对应位置上就行了(比如对\(v\)取\ ...
- bzoj4399 魔法少女LJJ 线段树合并
只看题面绝对做不出系列.... 注意到\(c \leqslant 7\),因此不会有删边操作(那样例删边干嘛) 注意到\(2, 5\)操作十分的有趣,启示我们拿线段树合并来做 操作\(7\)很好处理 ...
- bzoj4399 魔法少女LJJ 线段树合并+线段树二分+并查集
题目传送门 https://lydsy.com/JudgeOnline/problem.php?id=4399 题解 毒瘤题 \(9\) 种操作还有支持动态图的连通性 仔细读题 $ c<=7$. ...
- BZOJ 4399: 魔法少女LJJ(线段树)
传送门 解题思路 出题人真会玩..操作\(2\)线段树合并,然后每棵线段树维护元素个数和.对于\(6\)这个询问,因为乘积太大,所以要用对数.时间复杂度\(O(nlogn)\) 代码 #include ...
- BZOJ4399 魔法少女LJJ【线段树合并】【并查集】
Description 在森林中见过会动的树,在沙漠中见过会动的仙人掌过后,魔法少女LJJ已经觉得自己见过世界上的所有稀奇古怪的事情了 LJJ感叹道"这里真是个迷人的绿色世界,空气清新.淡雅 ...
- 【bzoj4399】魔法少女LJJ 并查集+权值线段树合并
题目描述 在森林中见过会动的树,在沙漠中见过会动的仙人掌过后,魔法少女LJJ已经觉得自己见过世界上的所有稀奇古怪的事情了LJJ感叹道“这里真是个迷人的绿色世界,空气清新.淡雅,到处散发着醉人的奶浆味: ...
随机推荐
- Boost.Asio基本原理(CSDN也有Markdown了,好开森)
Boost.Asio基本原理 这一章涵盖了使用Boost.Asio时必须知道的一些事情.我们也将深入研究比同步编程更复杂.更有乐趣的异步编程. 网络API 这一部分包含了当使用Boost.Asio编写 ...
- mysql函数之五:group_concat mysql 把结果集中的一列数据用指定分隔符转换成一行
函数使用说明:该函数返回带有来自一个组的连接的非NULL 值的字符串结果.其完整的语法如下 GROUP_CONCAT([DISTINCT] expr [,expr ...] [ORDER BY {un ...
- codeforce 510C Fox And Names(拓扑排序)
Fox And Names time limit per test 2 seconds memory limit per test 256 megabytes input standard input ...
- Nor Flash的CFI与JEDEC接口
Flash 存储器接口还有两个标准:CFI和JEDEC.CFI为公共Flash接口[Common Flash Interface],用来帮助程序从Flash芯片中获取操作方式信息(发送命令,从nor ...
- Python unittest excel数据驱动 写入
之前写过一篇关于获取excel数据进行迭代的方法,今天补充上写入的方法.由于我用的是Python3,不兼容xlutils,所以无法使用copy excel的方式来写入.这里使用xlwt3创建excel ...
- pa15-三省吾身
序号 项 1 凡事提前10分钟 凡事提前10分钟,会让你有充裕的时间应对可能的突发事件,更加从容. 试着把起床闹钟提前10分钟,你就会发现你出门不必急匆匆,早饭也可慢慢享用,一整天的状态也 ...
- samba Nginx
1.samba 2.nfs 3.crond 4.nginx ifconfig yum install net-tools -y ifconfig #查看所有已激活的网卡信息 ifconfig eth0 ...
- DAY2-python数据类型、字符编码、文件处理
阅读目录 一.引子 二.数字 三.字符串 四.列表 五.元祖 六.字典 七.集合 八.数据类型总结 九.运算符 十.字符编码 十一.文件处理 一.引子 1 什么是数据? x=10,10是我们要存储的数 ...
- Android实现智能提示的文本输入框AutoCompleteTextView
今天我们要讲一个十分简单的内容,就是一个安卓控件的使用,用法很简单,但是很常用的一个.这里我用两种不同的写法来处理.当然,无论用哪一种写法,效果都是一样的. 我们先来看效果图. 要实现这种效果十分简单 ...
- android手势(gesture)
需要实现两个接口,OnTouchListener ,OnGestureListener 在接口方法中实现各种事件 详见:http://www.cnblogs.com/JczmDeveloper/p/3 ...