BZOJ4009:[HNOI2015]接水果
浅谈树状数组与线段树:https://www.cnblogs.com/AKMer/p/9946944.html
题目传送门:https://www.lydsy.com/JudgeOnline/problem.php?id=4009
把每个水果\((u,v)\)转化成平面上的一个点\((dfn[u],dfn[v])\)。
我们考虑一个盘子,它可以接的水果的点集应该是矩形状的。
假设盘子的两个点是\(u,v\):
1、当\(u,v\)的\(lca\)不是\(u,v\)之一时:
那么凡是在矩形\((dfn[u],dfn[v],dfn[u]+siz[u]-1,dfn[v]+siz[v]-1)\)内的点都可以被这个盘子接住。
2、当\(u,v\)的\(lca\)是\(u\)或\(v\)时:
假设\(lca\)是\(u\),\(node\)是\(u\)的儿子并且是\(v\)的祖先。那么这个时候存在两个矩阵:
\((1,dfn[v],dfn[node]-1,dfn[v]+siz[v]-1)\)与\((dfn[v],dfn[node]+siz[node],dfn[v]+siz[v]-1,n)\)
我们对于每个矩阵做扫描线,外层套一个值域线段树,然后把水果和盘子放在一起\(sort\),动态修改然后在线段树上二分查找就行了。
时间复杂度:\(O(nlog^2n)\)
空间复杂度:\(O(nlog^2n)\)
代码如下:
#include <cstdio>
#include <iostream>
#include <algorithm>
using namespace std;
typedef pair<int,int> pii;
const int maxn=4e4+4;
int n,P,Q,tot,tim,cnt;
int now[maxn],pre[maxn*2],son[maxn*2];
int dep[maxn],dfn[maxn],siz[maxn],f[maxn][17];
int a[maxn],b[maxn],tmp[maxn],c[maxn],ans[maxn];
int read() {
int x=0,f=1;char ch=getchar();
for(;ch<'0'||ch>'9';ch=getchar())if(ch=='-')f=-1;
for(;ch>='0'&&ch<='9';ch=getchar())x=x*10+ch-'0';
return x*f;
}
struct Project {
int x,y1,y2,k,opt;
Project() {}
Project(int _x,int _y1,int _y2,int _k,int _opt) {
x=_x,y1=_y1,y2=_y2,k=_k,opt=_opt;
}
bool operator<(const Project &a)const {
if(x==a.x)return (opt!=0)>(a.opt!=0);//当x相同时先修改
return x<a.x;
}
}p[maxn*3];
struct pos_segment_tree {
int tot;
int sum[maxn*175],ls[maxn*175],rs[maxn*175];
void change(int &p,int l,int r,int L,int R,int opt) {
if(!p)p=++tot;
if(L<=l&&r<=R) {sum[p]+=opt;return;}
int mid=(l+r)>>1;
if(R<=mid)change(ls[p],l,mid,L,R,opt);
else if(L>mid)change(rs[p],mid+1,r,L,R,opt);
else change(ls[p],l,mid,L,mid,opt),change(rs[p],mid+1,r,mid+1,R,opt);
}
int query(int p,int l,int r,int pos) {
if(l==r)return sum[p];
int mid=(l+r)>>1,res=sum[p];
if(pos<=mid)res+=query(ls[p],l,mid,pos);
else res+=query(rs[p],mid+1,r,pos);
return res;
}
}T_inside;
struct val_segment_tree {
int rt[maxn<<2];
void change(int p,int l,int r,int pos,int L,int R,int v) {
T_inside.change(rt[p],1,n,L,R,v);
if(l==r)return;
int mid=(l+r)>>1;
if(pos<=mid)change(p<<1,l,mid,pos,L,R,v);
else change(p<<1|1,mid+1,r,pos,L,R,v);
}
int query(int p,int l,int r,int pos,int rk) {
if(l==r)return tmp[l];
int mid=(l+r)>>1,tmp=T_inside.query(rt[p<<1],1,n,pos);
if(tmp>=rk)return query(p<<1,l,mid,pos,rk);
else return query(p<<1|1,mid+1,r,pos,rk-tmp);
}
}T_outside;
void add(int a,int b) {
pre[++tot]=now[a];
now[a]=tot,son[tot]=b;
}
void dfs(int fa,int u) {
dep[u]=dep[fa]+1,siz[u]=1;
f[u][0]=fa,dfn[u]=++tim;
for(int i=1;i<=16;i++)
f[u][i]=f[f[u][i-1]][i-1];
for(int p=now[u],v=son[p];p;p=pre[p],v=son[p])
if(v!=fa)dfs(u,v),siz[u]+=siz[v];
}
pii lca(int u,int v) {
if(dep[u]<dep[v])swap(u,v);
for(int i=16;~i;i--)
if(dep[f[u][i]]>dep[v])
u=f[u][i];
if(f[u][0]==v)return make_pair(v,u);
if(dep[u]>dep[v])u=f[u][0];
for(int i=16;~i;i--)
if(f[u][i]!=f[v][i])
u=f[u][i],v=f[v][i];
return make_pair(f[u][0],0);
}
int main() {
n=read(),P=read(),Q=read();
for(int i=1;i<n;i++) {
int a=read(),b=read();
add(a,b),add(b,a);
}dfs(0,1);tot=Q;
for(int i=1;i<=P;i++)
a[i]=read(),b[i]=read(),tmp[i]=c[i]=read();
sort(tmp+1,tmp+P+1);
cnt=unique(tmp+1,tmp+P+1)-tmp-1;
for(int i=1;i<=P;i++)
c[i]=lower_bound(tmp+1,tmp+cnt+1,c[i])-tmp;
for(int i=1;i<=Q;i++) {
int a=read(),b=read();
a=dfn[a],b=dfn[b];
p[i].x=min(a,b);p[i].y1=max(a,b);
p[i].k=read(),p[i].y2=i;
}
for(int i=1;i<=P;i++) {
if(dep[a[i]]<dep[b[i]])swap(a[i],b[i]);
pii tmp=lca(a[i],b[i]);
int u=tmp.first,v=tmp.second;
int l=dfn[a[i]],r=dfn[a[i]]+siz[a[i]]-1;
if(u==b[i]) {
int L=1,R=dfn[v]-1;
if(L<=R) {
p[++tot]=Project(L,l,r,c[i],1);
p[++tot]=Project(R+1,l,r,c[i],-1);//R+1是因为要差分
}
L=dfn[v]+siz[v],R=n;
if(L<=R) {
p[++tot]=Project(l,L,R,c[i],1);
p[++tot]=Project(r+1,L,R,c[i],-1);
}
}
else {
int L=dfn[b[i]],R=dfn[b[i]]+siz[b[i]]-1;
if(r>R)swap(l,L),swap(r,R);
p[++tot]=Project(l,L,R,c[i],1);
p[++tot]=Project(r+1,L,R,c[i],-1);
}
}
sort(p+1,p+tot+1);
for(int i=1;i<=tot;i++) {
if(p[i].opt==0)ans[p[i].y2]=T_outside.query(1,1,cnt,p[i].y1,p[i].k);
else T_outside.change(1,1,cnt,p[i].k,p[i].y1,p[i].y2,p[i].opt);
}
for(int i=1;i<=Q;i++)
printf("%d\n",ans[i]);
return 0;
}
BZOJ4009:[HNOI2015]接水果的更多相关文章
- [BZOJ4009][HNOI2015]接水果(整体二分)
[HNOI2015]接水果 时间限制:60s 空间限制:512MB 题目描述 风见幽香非常喜欢玩一个叫做 osu!的游戏,其中她最喜欢玩的模式就是接水果. 由于她已经DT FC 了The b ...
- BZOJ4009: [HNOI2015]接水果
4009: [HNOI2015]接水果 Description 风见幽香非常喜欢玩一个叫做 osu!的游戏,其中她最喜欢玩的模式就是接水果. 由于她已经DT FC 了The big black, 她 ...
- 2018.10.02 bzoj4009: [HNOI2015]接水果(整体二分)
传送门 整体二分好题. 考虑水果被盘子接住的条件. 不妨设水果表示的路径为(x1,y1)(x_1,y_1)(x1,y1),盘子表示的为(x2,y2)(x_2,y_2)(x2,y2) 不妨设df ...
- [bzoj4009] [HNOI2015]接水果 整体二分+扫描线+dfs序+树状数组
Description 风见幽香非常喜欢玩一个叫做 osu!的游戏,其中她最喜欢玩的模式就是接水果. 由于她已经DT FC 了The big black, 她觉得这个游戏太简单了,于是发明了一个更 加 ...
- bzoj4009: [HNOI2015]接水果(整体二分)
题目描述 风见幽香非常喜欢玩一个叫做 osu!的游戏,其中她最喜欢玩的模式就是接水果.由于她已经DT FC 了The big black, 她觉得这个游戏太简单了,于是发明了一个更加难的版本. 首先有 ...
- bzoj4009 [HNOI2015]接水果 整体二分+扫描线+树状数组+dfs序
题目传送门 https://lydsy.com/JudgeOnline/problem.php?id=4009 题解 考虑怎样的情况就会有一个链覆盖另一个链. 设被覆盖的链为 \(a - b\),覆盖 ...
- 【BZOJ4009】[HNOI2015]接水果 DFS序+整体二分+扫描线+树状数组
[BZOJ4009][HNOI2015]接水果 Description 风见幽香非常喜欢玩一个叫做 osu!的游戏,其中她最喜欢玩的模式就是接水果.由于她已经DT FC 了The big black, ...
- 【BZOJ4009】接水果(整体二分,扫描线)
[BZOJ4009]接水果(整体二分,扫描线) 题面 为什么这都是权限题???,洛谷真良心 题解 看到这道题,感觉就是主席树/整体二分之类的东西 (因为要求第\(k\)大) 但是,读完题目之后,我们发 ...
- BZOJ 4009: [HNOI2015]接水果
4009: [HNOI2015]接水果 Time Limit: 60 Sec Memory Limit: 512 MBSubmit: 636 Solved: 300[Submit][Status] ...
- 洛谷 P3242 [HNOI2015]接水果 解题报告
P3242 [HNOI2015]接水果 题目描述 风见幽香非常喜欢玩一个叫做 \(osu!\) 的游戏,其中她最喜欢玩的模式就是接水果.由于她已经\(DT\) \(FC\) 了\(\tt{The\ b ...
随机推荐
- 九度OJ 1205:N阶楼梯上楼问题 (斐波那契数列)
时间限制:1 秒 内存限制:128 兆 特殊判题:否 提交:3739 解决:1470 题目描述: N阶楼梯上楼问题:一次可以走两阶或一阶,问有多少种上楼方式.(要求采用非递归) 输入: 输入包括一个整 ...
- 九度OJ 1189:还是约瑟夫环 (约瑟夫环)
时间限制:1 秒 内存限制:32 兆 特殊判题:否 提交:820 解决:522 题目描述: 生成一个长度为21的数组,依次存入1到21: 建立一个长度为21的单向链表,将上述数组中的数字依次存入链表每 ...
- Centos 6 安装 python2.7 和 pip
一.安装 python2.7 [root@crazy-acong ~]# cd /data/tools/ [root@crazy-acong tools]# yum groupinstall &quo ...
- js判断undefined类型,undefined,null, 的区别详细解析
js判断undefined类型 今天使用showModalDialog打开页面,返回值时.当打开的页面点击关闭按钮或直接点浏览器上的关闭则返回值是undefined所以自作聪明判断 var reVal ...
- SVN代码merge
如何merge代码?建议用命令搞merge,客户端图形界面不是很给力.SVN 1.5以上版本,可以使用SVN的自动合并:将主干合并到分支:进入分支目录,执行命令: svn merge http://s ...
- mysql索引学习----2----创建索引、修改索引、删除索引的命令语句
查看表中已经存在 index:show index from table_name; 创建和删除索引索引的创建可以在CREATE TABLE语句中进行,也可以单独用CREATE INDEX或ALTER ...
- ubuntu防火墙
开启防火墙 sudo ufw enable 关闭防火墙 sudo ufw disable 查看防火墙状态 sudo utw status
- HDR(High Dynamic Range) - 高动态范围
1. Dynamic Range 动态范围是指一个场景的最亮和最暗部分之间的相对比值 2. Tone-mapping 现实真正存在的亮度差,即最亮的物体亮度和最暗的物体亮度之比为 , 而人类的眼睛 ...
- nodejs之入门
一.安装: 去官网下载稳定版本,安装即可. 安装后执行node -v,显示node版本,然后输入node回车,然后输出console.log(111);,正常输出即可. 二.入门介绍: 1.nodej ...
- 0521 HTML基础
一.web标准 web准备介绍: w3c:万维网联盟组织,用来制定web标准的机构(组织) web标准:制作网页遵循的规范 web准备规范的分类:结构标准.表现标准.行为标准. 结构:html.表示: ...