BZOJ:5457: 城市(线段树合并)(尚待优化)
5457: 城市
Time Limit: 20 Sec Memory Limit: 512 MB
Submit: 18 Solved: 12
[Submit][Status][Discuss]
Description
Input
Output
Sample Input
1 2
1 3
2 4
4 5
3 6
5 7
1 8
2 8
2 5
1 1
3 1
6 7
5 6
1 10
4 6
Sample Output
1 10
5 6
1 10
1 10
5 6
1 10
4 6
思路:每个节点保存一个线段树,表示子树的民族及其数量,维护区间最大值。
然后DFS,线段树合并即可。 7500ms。再8个AC里,我的三个AC代码分别排最后三名。 亟待优化啊!
#include<bits/stdc++.h>
#define rep(i,a,b) for(int i=a;i<=b;i++)
using namespace std;
const int inf=0x3f3f3f3f;
const int maxn=;
void read(int &x){
x=; char c=getchar();
while(c>''||c<'') c=getchar();
while(c>=''&&c<='') x=x*+c-'',c=getchar();
}
struct in{
int l,r,pos,mx;
in(){l=r=mx=pos=;}
}s[maxn*];
int Laxt[maxn],Next[maxn],To[maxn],cnt,M;
int A[maxn],B[maxn],rt[maxn],tot,ans[maxn][];
void add(int u,int v){
Next[++cnt]=Laxt[u]; Laxt[u]=cnt; To[cnt]=v;
}
void pushup(int Now){
if(s[Now].l&&!s[Now].r){
s[Now].mx=s[s[Now].l].mx; s[Now].pos=s[s[Now].l].pos;
}
else if(s[Now].l&&s[Now].r&&s[s[Now].l].mx>=s[s[Now].r].mx){
s[Now].mx=s[s[Now].l].mx; s[Now].pos=s[s[Now].l].pos;
}
else s[Now].mx=s[s[Now].r].mx,s[Now].pos=s[s[Now].r].pos;
}
void insert(int &Now,int L,int R,int pos,int num)
{
if(!Now) Now=++tot;
if(L==R){ s[Now].mx=num; s[Now].pos=L; return ;}
int Mid=(L+R)>>;
if(pos<=Mid) insert(s[Now].l,L,Mid,pos,num);
else insert(s[Now].r,Mid+,R,pos,num);
pushup(Now);
}
int merge(int x,int y,int L,int R){
if(!x||!y) return x|y;
if(L==R) { s[x].mx+=s[y].mx; return x;}
int Mid=(L+R)>>;
s[x].l=merge(s[x].l,s[y].l,L,Mid);
s[x].r=merge(s[x].r,s[y].r,Mid+,R);
pushup(x); return x;
}
void dfs(int u,int f){
for(int i=Laxt[u];i;i=Next[i]){
int v=To[i]; if(v==f) continue;
dfs(v,u);
rt[u]=merge(rt[u],rt[v],,M);
}
ans[u][]=s[rt[u]].mx; ans[u][]=s[rt[u]].pos;
}
int main()
{
int N,u,v;
scanf("%d%d",&N,&M);
rep(i,,N-){
read(u); read(v);
add(u,v); add(v,u);
}
rep(i,,N) read(A[i]),read(B[i]);
rep(i,,N) insert(rt[i],,M,A[i],B[i]);
dfs(,);
rep(i,,N) printf("%d %d\n",ans[i][],ans[i][]);
return ;
}
然后想着每一层,我先合并子树大小较小的。 即按sz排序后再合并,和上次一样,并没有优化。 7988ms
#include<bits/stdc++.h>
#define rep(i,a,b) for(int i=a;i<=b;i++)
using namespace std;
const int inf=0x3f3f3f3f;
const int maxn=;
void read(int &x){
x=; char c=getchar();
while(c>''||c<'') c=getchar();
while(c>=''&&c<='') x=x*+c-'',c=getchar();
}
struct in{
int l,r,pos,mx;
in(){l=r=mx=pos=;}
}s[maxn*];
int Laxt[maxn],Next[maxn],To[maxn],cnt,M,sz[maxn];
int A[maxn],B[maxn],rt[maxn],tot,ans[maxn][];
void add(int u,int v){
Next[++cnt]=Laxt[u]; Laxt[u]=cnt; To[cnt]=v;
}
void pushup(int Now){
if(s[Now].l&&!s[Now].r){
s[Now].mx=s[s[Now].l].mx; s[Now].pos=s[s[Now].l].pos;
}
else if(s[Now].l&&s[Now].r&&s[s[Now].l].mx>=s[s[Now].r].mx){
s[Now].mx=s[s[Now].l].mx; s[Now].pos=s[s[Now].l].pos;
}
else s[Now].mx=s[s[Now].r].mx,s[Now].pos=s[s[Now].r].pos;
}
void insert(int &Now,int L,int R,int pos,int num)
{
if(!Now) Now=++tot;
if(L==R){ s[Now].mx=num; s[Now].pos=L; return ;}
int Mid=(L+R)>>;
if(pos<=Mid) insert(s[Now].l,L,Mid,pos,num);
else insert(s[Now].r,Mid+,R,pos,num);
pushup(Now);
}
int merge(int x,int y,int L,int R){
if(!x||!y) return x|y;
if(L==R) { s[x].mx+=s[y].mx; return x;}
int Mid=(L+R)>>;
s[x].l=merge(s[x].l,s[y].l,L,Mid);
s[x].r=merge(s[x].r,s[y].r,Mid+,R);
pushup(x); return x;
}
void dfs1(int u,int f){
sz[u]=;
for(int i=Laxt[u];i;i=Next[i]){
if(To[i]!=f) dfs1(To[i],u);
sz[u]+=sz[To[i]];
}
}
bool cmp(int w,int v){ return sz[w]<sz[v]; }
void dfs2(int u,int f){
cnt=; vector<int>G;
for(int i=Laxt[u];i;i=Next[i]){
if(To[i]!=f) G.push_back(To[i]);
}
sort(G.begin(),G.end(),cmp);
for(int i=;i<G.size();i++){
int v=G[i]; dfs2(v,u);
rt[u]=merge(rt[u],rt[v],,M);
}
ans[u][]=s[rt[u]].mx; ans[u][]=s[rt[u]].pos;
}
int main()
{
int N,u,v;
scanf("%d%d",&N,&M);
rep(i,,N-){
read(u); read(v);
add(u,v); add(v,u);
}
rep(i,,N) read(A[i]),read(B[i]);
rep(i,,N) insert(rt[i],,M,A[i],B[i]);
dfs1(,);
dfs2(,);
rep(i,,N) printf("%d %d\n",ans[i][],ans[i][]);
return ;
}
BZOJ:5457: 城市(线段树合并)(尚待优化)的更多相关文章
- BZOJ #5457: 城市 [线段树合并]
线段树合并的板子题,每次从下到上合并就完事了 // by Isaunoya #include <bits/stdc++.h> using namespace std; #define re ...
- BZOJ 4530 LCT/线段树合并
//By SiriusRen #include <cstdio> #include <cstring> #include <algorithm> using nam ...
- 线段树合并 || BZOJ 5457: 城市
题面:https://www.lydsy.com/JudgeOnline/problem.php?id=5457 题解: 线段树合并,对于每个节点维护sum(以该节点为根的子树中最大的种类和)和kin ...
- [BZOJ 2212] [Poi2011] Tree Rotations 【线段树合并】
题目链接:BZOJ - 2212 题目分析 子树 x 内的逆序对个数为 :x 左子树内的逆序对个数 + x 右子树内的逆序对个数 + 跨越 x 左子树与右子树的逆序对. 左右子树内部的逆序对与是否交换 ...
- BZOJ.4399.魔法少女LJJ(线段树合并)
BZOJ 注意\(c\leq7\)→_→ 然后就是裸的权值线段树+线段树合并了. 对于取\(\max/\min\)操作可以直接区间修改清空超出范围的值,然后更新到对应位置上就行了(比如对\(v\)取\ ...
- BZOJ.5461.[PKUWC2018]Minimax(DP 线段树合并)
BZOJ LOJ 令\(f[i][j]\)表示以\(i\)为根的子树,权值\(j\)作为根节点的概率. 设\(i\)的两棵子树分别为\(x,y\),记\(p_a\)表示\(f[x][a]\),\(p_ ...
- BZOJ.3307.雨天的尾巴(dsu on tree/线段树合并)
BZOJ 洛谷 \(dsu\ on\ tree\).(线段树合并的做法也挺显然不写了) 如果没写过\(dsu\)可以看这里. 对修改操作做一下差分放到对应点上,就成了求每个点子树内出现次数最多的颜色, ...
- BZOJ.3653.谈笑风生(长链剖分/线段树合并/树状数组)
BZOJ 洛谷 \(Description\) 给定一棵树,每次询问给定\(p,k\),求满足\(p,a\)都是\(b\)的祖先,且\(p,a\)距离不超过\(k\)的三元组\(p,a,b\)个数. ...
- BZOJ.5417.[NOI2018]你的名字(后缀自动机 线段树合并)
LOJ 洛谷 BZOJ 考虑\(l=1,r=|S|\)的情况: 对\(S\)串建SAM,\(T\)在上面匹配,可以得到每个位置\(i\)的后缀的最长匹配长度\(mx[i]\). 因为要去重,对\(T\ ...
随机推荐
- GDPR或使全球域名whois信息被隐藏
什么是GDPR? GDPR 全称为 General Data Protection Regulation,是欧盟 2016 年 4 月通过的一项通用数据保护条例(或称“一般数据保护法案”),是 199 ...
- Faster-rcnn 配置方法
Faster-rcnn 在Linux下的配置方法 感谢@邓学长 建立过程: (下载库的时候要按照库readme 进行操作) opencv 的包下载安装,安装教程 用git命令将这个库下载到本地 fas ...
- vector的坑——C++primer练习6.33总结
说来惭愧,一道简单的对vector递归的题目写了一个多小时,最后还是请教了大神才改出来. 首先贴上原代码: void return_vector(vector<int>::iterator ...
- 缓存技术内部交流_03_Cache Aside
参考资料: http://www.ehcache.org/documentation/3.2/caching-patterns.html http://www.ehcache.org/document ...
- angular2中的路由转场动效
1.为什么有的人路由转动效离场动效不生效? 自己研究发现是加动效的位置放错了 如下: <---! animate-state.component.html --> <div sty ...
- 关于hugepages 3.txt
关于hugepages 3.txt --//有一段时间我一直强调安装oracle一定要配置hugepage,因为现在的服务器内存越来越大,如果还使用4K的页面表,如果内存表占用内存巨大, --//特别 ...
- ContOS网络连接及简单的ssh Xshell连接!
这边简单的记录一下下ContOS网络连接及简单的ssh Xshell连接! 首先你得安装一个Contos Linux系统对吧! 1.找到设置--->网络-->有线连接-->IPv4 ...
- Tracing on Linux
The Linux tracing APIs are a relatively new addition to the kernel and one of the most powerful new ...
- (2) iOS开发之UI处理-UILabel篇
我们经常要根据内容去动态计算控件的高度,比如一个UILabel控件,常常要显示多行内容,并且计算出总高度,如果每个UILabel要多行显示,都要写这么一段代码是非常痛苦的,看代码如下: 我想大 ...
- 控制台小游戏-贪吃蛇,c++和c#版
说是c++版,其实只是用到了c++的cout和cin而已.这是我做的第二个控制台游戏,基本上每一行代码都加上了注释. 游戏嘛,我觉得重要的是了解他的思想,所以后期学了面向对象之后这个游戏的代码我也没有 ...