Description

  





  

  

  

Solution

  

  看到这种路径统计问题,一般就想到要用点分治去做。

  

  对于每个重心\(u\),统计经过\(u\)的合法的路径之中的最大值。

  

  第一类路径是从\(u\)出发的,直接逐个子树深搜统计就可以了。第二类路径是由两棵不同子树中的两条第一类路径拼接而成的。

  

  如果仅仅是统计长度在\([l,r]\)之间的路径有多少条,经典的统计+容斥做法就可以解决。然而现在的问题比较复杂,一来不好容斥,二来两两路径配对需要有判定条件:两条路径的接口边颜色是否相同。

  

  我们可以采用一种不需要容斥的做法:逐一枚举子树,并逐一考虑由子树内的每个点出发去其他子树的路径。枚举到当前这棵子树内的某个节点\(x\)时,我们用\(sum[d]\)记录在之前的子树中,相对于\(u\)深度为\(d\)的点的某些信息。那么对于一个点\(x\),它可以接上的路径的信息应该有\(sum[i],\;i\in[l-dep_x,r-dep_x]\)。

  

  所以\(sum[d]\)记录的是什么呢?按照题目意思,我们应该记录所有从深度为\(d\)的点出发到\(u\)路径中的路径最大值。可是注意这个最大值不一定相对所有询问点来说都是最大值,有可能此最大值对应的路径\(\alpha\)与当前询问点路径接口边相同,权值重复,需要减去一次,说不定就不是最大的了。所以,我们额外要记录一个接口颜色异于最大值路径\(\alpha\)的另一条权值最大路径\(\beta\)。

  

  这样一来,对于当前枚举点对应的路径\(\gamma\),想要和先前子树中一条长度为\(d\)的路径结合时,有\(sum[d]=(\alpha,\beta)\)(分别为两条路径的权值)。先看\(\alpha\)的接口边颜色是否和\(\gamma\)相同,如果是,取\(\max\{\alpha-c,\beta\}\)作为最大路径的权值和\(\gamma\)拼接(\(c\)表示\(\alpha\)和\(\gamma\)的接口边颜色的权值);否则,取\(\alpha\)拼接。

  

  有意思的是,因为我们是一个一个子树进行处理,所以当前枚举的所有点的接口边颜色都是一样的,也就意味着不同点尝试配对同一个\(sum\)时的选择都是完全一样的。因此每一个\(sum\)的贡献是定值。我们相当于有一个数列,每次求\([l-dep_x,r-dep_x]\)中的最大值。如果我们把点按照深度来排序,那么这个范围就是一个滑动窗口,可以使用单调队列进行\(\mathcal O(n)\)解决。

  

  

  

Code

  

#include <cstdio>
#include <queue>
#include <algorithm>
using namespace std;
const int N=200005,INF=2e9+5;
int n,m,l,r,cv[N],all;
int best,bestval,size[N];
int udep[N],cc[N];
int lis[N],lcnt,info[N][3];
deque<int> q;
int qv[N];
bool cut[N],vis[N];
int ans;
struct Data{
int v1,c1,v2,c2;
Data(){v1=v2=-INF; c1=c2=-1;}
inline void insert(int v,int c){
if(c!=c1){
if(v>v1)
v2=v1,c2=c1,v1=v,c1=c;
else if(v>v2)
v2=v,c2=c;
}
else if(v>v1) v1=v;
}
int get(int c){
return c==c1?max(v1-cv[c],v2):v1;
}
}s[N];
int h[N],tot;
struct Edge{int v,c,next;}e[N*2];
inline int min(int x,int y){return x<y?x:y;}
inline int max(int x,int y){return x>y?x:y;}
inline void addEdge(int u,int v,int c){
e[++tot]=(Edge){v,c,h[u]}; h[u]=tot;
e[++tot]=(Edge){u,c,h[v]}; h[v]=tot;
}
void getRt(int u,int fa,int sz){
int maxsub=-1;
size[u]=1;
for(int i=h[u],v;i;i=e[i].next)
if((v=e[i].v)!=fa&&!cut[v]){
getRt(v,u,sz);
size[u]+=size[v];
maxsub=max(maxsub,size[v]);
}
maxsub=max(maxsub,sz-size[u]);
if(maxsub<bestval)
bestval=maxsub,best=u;
}
bool cmp(const int &x,const int &y){return udep[x]<udep[y];}
void dfs(int u,int fa,int dep,int &nowx){
nowx=max(nowx,dep);
for(int i=h[u],v;i;i=e[i].next)
if((v=e[i].v)!=fa&&!cut[v])
dfs(v,u,dep+1,nowx);
}
void insert(int u,int fa,int dep,int topc,int fac,int val){
vis[u]=false;
if(l<=dep&&dep<=r) ans=max(ans,val);
s[dep].insert(val,topc);
for(int i=h[u],v;i;i=e[i].next)
if((v=e[i].v)!=fa&&!cut[v])
insert(v,u,dep+1,topc,e[i].c,fac==e[i].c?val:val+cv[e[i].c]);
}
void collect(int x,int topc){
int head=1,tail=1;
lis[1]=x;
info[x][0]=1; info[x][1]=cv[topc]; info[x][2]=topc;
vis[x]=true;
while(head<=tail){
int u=lis[head++];
for(int i=h[u],v;i;i=e[i].next)
if(!cut[v=e[i].v]&&!vis[v]){
lis[++tail]=v;
vis[v]=true;
info[v][0]=info[u][0]+1;
info[v][1]=info[u][1]+(e[i].c==info[u][2]?0:cv[e[i].c]);
info[v][2]=e[i].c;
}
}
lcnt=tail;
}
void solve(int x,int sz){
best=-1; bestval=INF;
getRt(x,0,sz);
int u=best;
cut[u]=true;
static int son[N],cnt;
cnt=0;
int curr=-1;
for(int i=h[u],v;i;i=e[i].next)
if(!cut[v=e[i].v]){
son[++cnt]=v;
cc[v]=e[i].c;
dfs(v,0,1,udep[v]);
curr=max(curr,udep[son[cnt]]);
}
curr=min(r,curr);
for(int i=1;i<=curr;i++) s[i]=Data();
vis[u]=true;
for(int i=1,v;i<=cnt;i++){
v=son[i];
collect(v,cc[v]);
while(!q.empty()) q.pop_back();
int nowl,nowr,qr=0;
bool first=true;
for(int j=lcnt;j>=1;j--)
if(info[lis[j]][0]<r){
int x=lis[j];
nowl=max(1,l-info[x][0]); nowr=r-info[x][0];
if(first){
first=false;
qr=nowl-1;
while(qr<nowr&&qr<curr&&s[qr+1].v1>-INF){
int newval=s[++qr].get(cc[v]);
if(newval>-INF){
while(!q.empty()&&qv[q.back()]<=newval) q.pop_back();
q.push_back(qr);
qv[qr]=newval;
}
}
}
while(!q.empty()&&q.front()<nowl) q.pop_front();
while(qr<nowr&&qr<curr&&s[qr+1].v1>-INF){
int newval=s[++qr].get(cc[v]);
if(newval>-INF){
while(!q.empty()&&qv[q.back()]<=newval) q.pop_back();
q.push_back(qr);
qv[qr]=newval;
}
}
if(!q.empty())
ans=max(ans,info[x][1]+qv[q.front()]);
}
insert(v,0,1,cc[v],cc[v],cv[cc[v]]);
}
for(int i=h[u],v;i;i=e[i].next)
if(!cut[v=e[i].v])
solve(v,size[v]>size[u]?(sz-size[u]):size[v]);
}
int main(){
scanf("%d%d%d%d",&n,&m,&l,&r);
for(int i=1;i<=m;i++) scanf("%d",cv+i);
for(int i=1,u,v,c;i<n;i++){
scanf("%d%d%d",&u,&v,&c);
addEdge(u,v,c);
}
ans=-INF;
solve(1,n);
printf("%d\n",ans);
return 0;
}

【XSY2307】树的难题的更多相关文章

  1. [BJOI2017]树的难题 点分治 线段树

    题面 [BJOI2017]树的难题 题解 考虑点分治. 对于每个点,将所有边按照颜色排序. 那么只需要考虑如何合并2条链. 有2种情况. 合并路径的接口处2条路径颜色不同 合并路径的接口处2条路径颜色 ...

  2. [BJOI2017]树的难题 点分治,线段树合并

    [BJOI2017]树的难题 LG传送门 点分治+线段树合并. 我不会写单调队列,所以就写了好写的线段树. 考虑对于每一个分治中心,把出边按颜色排序,这样就能把颜色相同的子树放在一起处理.用一棵动态开 ...

  3. BZOJ3257 : 树的难题

    设$f[x][i][j]$表示以$x$为根的子树,与$x$连通部分有$i$个黑点,$j$个白点,不联通部分都是均衡的最小代价.若$i>1$,则视作$1$:若$j>2$,则视作$2$. 然后 ...

  4. BZOJ4860 BJOI2017 树的难题 点分治、线段树合并

    传送门 只会线段树……关于单调队列的解法可以去看“重建计划”一题. 看到路径长度$\in [L,R]$考虑点分治.可以知道,在当前分治中心向其他点的路径中,始边(也就是分治中心到对应子树的根的那一条边 ...

  5. [bzoj4860] [BeiJing2017]树的难题

    Description 给你一棵 n 个点的无根树.树上的每条边具有颜色. 一共有 m 种颜色,编号为 1 到 m.第 i 种颜色的权值为 ci.对于一条树上的简单路径,路径上经过的所有边按顺序组成一 ...

  6. bzoj 4860 [BeiJing2017]树的难题

    题面 https://www.lydsy.com/JudgeOnline/problem.php?id=4860 题解 点分治 设当前重心为v 假设已经把所有边按照出发点第一关键字, 颜色第二关键字排 ...

  7. 并不对劲的loj2179:p3714:[BJOI2017]树的难题

    题目大意 有一棵树,\(n\)(\(n\leq2*10^5\))个点,每条边\(i\)有颜色\(w_i\),共有\(m\)(\(m\leq n\))种颜色,第\(i\)种颜色的权值是\(c_i\)(\ ...

  8. [JZOJ3347] 【NOI2013模拟】树的难题

    题目 题目大意 给你一棵树,每个节点有三种黑.白.灰三种颜色. 你要割掉一些边(每条边被割需要付出一定的代价),使得森林的每棵树满足: 没有黑点或至多一个白点. 思考历程 这题一看就知道是一个树形DP ...

  9. G 树的难题

    时间限制 : 10000 MS   空间限制 : 165536 KB 评测说明 : 1s,128m 问题描述 给出一个无根树.树有N个点,边有权值.每个点都有颜色,是黑色.白色.灰色这三种颜色之一,称 ...

随机推荐

  1. 【厚积薄发】Crunch压缩图片的AssetBundle打包

    这是第133篇UWA技术知识分享的推送.今天我们继续为大家精选了若干和开发.优化相关的问题,建议阅读时间10分钟,认真读完必有收获. UWA 问答社区:answer.uwa4d.com UWA QQ群 ...

  2. Spring中的数据库事物管理

    Spring中的数据库事物管理 只要给方法加一个@Transactional注解就可以了 例如:

  3. React Native移动开发实战-3-实现页面间的数据传递

    React Native使用props来实现页面间数据传递和通信.在React Native中,有两种方式可以存储和传递数据:props(属性)以及state(状态),其中: props通常是在父组件 ...

  4. 第十三次ScrumMeeting博客

    第十三次ScrumMeeting博客 本次会议于12月3日(六)21时30分整在3公寓725房间召开,持续20分钟. 与会人员:刘畅.辛德泰.张安澜.赵奕. 1. 每个人的工作(有Issue的内容和链 ...

  5. 使用Python批量修改数据库执行Sql文件

    由于上篇文章中批量修改了文件,有的时候数据库也需要批量修改一下,之前的做法是使用宝塔的phpMyAdmin导出一个已经修改好了的sql文件,然后依次去其他数据库里导入,效率不说极低,也算低了,且都是些 ...

  6. 封装js插件学习指南

    封装js插件学习指南 1.原生JavaScript插件编写指南 => 传送门 2.如何定义一个高逼格的原生JS插件 =>传送门 3.手把手教你用原生JavaScript造轮子 =>  ...

  7. 微信小程序——节奏练耳 宣传页

    节奏练耳是什么? 节奏练耳小程序是一款听音练习节奏的交互式小程序.节奏练耳第一大节是辨认六种音符的练习,剩余九大节的练习题中播放的音频是将时值长短不一的音符组合在一起,配合相应的节奏图片,以提高辨认节 ...

  8. No.1010_第七次团队会议

    渺茫的前景 今天大家都很失望,一来昨天的问题还是继续存在着,仍然没有完成.二来,我们看了一下其余几个组的界面,对自己有些难过. 我们组确实存在人手少的问题,这几天我还因为挑战杯的事情缺席了两天,感觉内 ...

  9. 实验3 --俄罗斯方块 with 20135335郝爽

    一.   实验内容 (一)敏捷开发与XP 内容:1.敏捷开发(Agile Development)是一种以人为核心.迭代.循序渐进的开发方法. 2.极限编程(eXtreme Programming,X ...

  10. 软工1816 · Beta冲刺(7/7)

    团队信息 队名:爸爸饿了 组长博客:here 作业博客:here 组员情况 组员1(组长):王彬 过去两天完成了哪些任务 协助完成安卓端的整合 完成安卓端的美化 协助制作宣传视频 接下来的计划 &am ...