5457: 城市

Time Limit: 20 Sec  Memory Limit: 512 MB
Submit: 18  Solved: 12
[Submit][Status][Discuss]

Description

有n座城市,m个民族。这些城市之间由n-1条道路连接形成了以城市1为根的有根树。每个城市都是某一民族的聚居
地,Master知道第i个城市的民族是A_i,人数是B_i。为了维护稳定,Master需要知道某个区域内人数最多的民族
。他向你提出n个询问,其中第i个询问是:求以i为根的子树内,人数最多的民族有是哪个,这个民族有多少人。
如果子树内人数最多的民族有多个,输出其中编号最小的民族。

Input

共有2*n行。
第一行有两个整数n, m。
接下来n-1行,每行有两个整数u, v,表示一条连接u和v的道路。
接下来n行,第i行有两个整数A_i, B_i。
n<=400000,m<=n,1<=A_i<=m,0<=B_i<=1000。

Output

共有n行。
第i行两个整数x, y,分别表示以i为根的子树中人数最多的民族和它的人数。

Sample Input

8 6
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

2 13
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: 城市(线段树合并)(尚待优化)的更多相关文章

  1. BZOJ #5457: 城市 [线段树合并]

    线段树合并的板子题,每次从下到上合并就完事了 // by Isaunoya #include <bits/stdc++.h> using namespace std; #define re ...

  2. BZOJ 4530 LCT/线段树合并

    //By SiriusRen #include <cstdio> #include <cstring> #include <algorithm> using nam ...

  3. 线段树合并 || BZOJ 5457: 城市

    题面:https://www.lydsy.com/JudgeOnline/problem.php?id=5457 题解: 线段树合并,对于每个节点维护sum(以该节点为根的子树中最大的种类和)和kin ...

  4. [BZOJ 2212] [Poi2011] Tree Rotations 【线段树合并】

    题目链接:BZOJ - 2212 题目分析 子树 x 内的逆序对个数为 :x 左子树内的逆序对个数 + x 右子树内的逆序对个数 + 跨越 x 左子树与右子树的逆序对. 左右子树内部的逆序对与是否交换 ...

  5. BZOJ.4399.魔法少女LJJ(线段树合并)

    BZOJ 注意\(c\leq7\)→_→ 然后就是裸的权值线段树+线段树合并了. 对于取\(\max/\min\)操作可以直接区间修改清空超出范围的值,然后更新到对应位置上就行了(比如对\(v\)取\ ...

  6. BZOJ.5461.[PKUWC2018]Minimax(DP 线段树合并)

    BZOJ LOJ 令\(f[i][j]\)表示以\(i\)为根的子树,权值\(j\)作为根节点的概率. 设\(i\)的两棵子树分别为\(x,y\),记\(p_a\)表示\(f[x][a]\),\(p_ ...

  7. BZOJ.3307.雨天的尾巴(dsu on tree/线段树合并)

    BZOJ 洛谷 \(dsu\ on\ tree\).(线段树合并的做法也挺显然不写了) 如果没写过\(dsu\)可以看这里. 对修改操作做一下差分放到对应点上,就成了求每个点子树内出现次数最多的颜色, ...

  8. BZOJ.3653.谈笑风生(长链剖分/线段树合并/树状数组)

    BZOJ 洛谷 \(Description\) 给定一棵树,每次询问给定\(p,k\),求满足\(p,a\)都是\(b\)的祖先,且\(p,a\)距离不超过\(k\)的三元组\(p,a,b\)个数. ...

  9. BZOJ.5417.[NOI2018]你的名字(后缀自动机 线段树合并)

    LOJ 洛谷 BZOJ 考虑\(l=1,r=|S|\)的情况: 对\(S\)串建SAM,\(T\)在上面匹配,可以得到每个位置\(i\)的后缀的最长匹配长度\(mx[i]\). 因为要去重,对\(T\ ...

随机推荐

  1. JVM 内存调优 与 实际案例

    堆内存设置 原理 JVM堆内存分为2块:Permanent Space 和 Heap Space. Permanent 即 持久代(Permanent Generation),主要存放的是Java类定 ...

  2. C# 人民币转成大写

    /// <summary> /// 转换人民币大小金额 /// </summary> /// <param name="num">金额</ ...

  3. ELK 6.x 部署

    Elasticsearch版本:6.3.2 Kibana版本:6.3.2 1.es安装 按照官方提示操作即可. 通过yum安装或者下载tar包解压. 安装完成之后,需要修改一些配置 ①修改文件 /et ...

  4. git拉取GitLab工程报错Repository not found

    # git clone http://xxx/jiqing/frog.git 正克隆到 'frog'... fatal: repository 'http://xxx/jiqing/frog.git/ ...

  5. 1004: [HNOI2008]Cards burnside定理

    https://www.lydsy.com/JudgeOnline/problem.php?id=1004 输入数据保证任意多次洗牌都可用这 m种洗牌法中的一种代替,且对每种洗牌法,都存在一种洗牌法使 ...

  6. 搞懂分布式技术3:初探分布式协调服务zookeeper

    搞懂分布式技术3:初探分布式协调服务zookeeper 1.Zookeepr是什么 Zookeeper是一个典型的分布式数据一致性的解决方案,分布式应用程序可以基于它实现诸如数据发布/订阅,负载均衡, ...

  7. 外层页面与iframe相互调用的方法

    iframe调用外层页面的方法: parent.func(); 外层页面调用里面的iframe中的内容方法:  $("iframe").contents().find(" ...

  8. Ubuntu 安装GNU Scientific library(GSL)

    注: 此系列为自己之前所搭建网站内容. 由于论文数据处理的需要,需要使用libeemd这个包,需要安装gsl科学库,windows下没有办法,只能转战ubuntu进行科学计算. GSL(GNU Sci ...

  9. python爬取商品信息

    老严要爬某网购网站的商品信息,正好我最近在学python,就一起写了一个简单的爬虫程序. 需求:某网的商品信息,包括商品名,市场价和售价 工具:python2.7.8,urllib2,re #codi ...

  10. eureka-3-常用注解

    @EnableDiscoveryClient @EnableEurekaClient 上面两个注解都是用在应用的启动类上面,声明这是一个Eureka Client ,现在说明两个注解的区别. spri ...