题目

首先我们知道,在树上距离一个点最远的点一定是直径的两个端点之一

首先两遍\(\rm dfs\)把直径求出来,定义\(d(u)\)表示点\(u\)距离其最远点的距离,有了直径我们就能求出\(d\)数组了

当然可以树形\(\rm dp\),设\(f_{x,i,j}\)表示在\(x\)子树内部选择一个最大值为\(i\)最小值为\(j\)的最大联通块是多少,显然这样的复杂度奇高无比

考虑把求出的直径来好好利用一下

首先感性的发现一下,设\(x\)为直径中点,那么\(d(x)\)肯定是最小的点权,又发现我们钦定\(x\)为根,那么一个点的点权肯定比其子树内部的点都要小

那么我们就钦定\(x\)为根,之后枚举一个联通块的最高点\(i\),也就是点权的最小值

由于深度越大点权越大,于是我们就把这个子树内部所有点权不超过\(d(i)+L\)的点都选上,这显然构成一个联通块

于是问题转化成了求一个子树内部有多少个点的点权小于某个给定值,不难发现这是主席树裸题

说的好,于是我选择\(\rm dsu\ on\ tree\),复杂度\(O(nq\log n+n\log^2n)\),莫名其妙跑得比并查集老哥们快

代码

#include<bits/stdc++.h>
#define re register
#define LL long long
#define lb(x) ((x)&(-x))
#define max(a,b) ((a)>(b)?(a):(b))
inline int read() {
char c=getchar();int x=0;while(c<'0'||c>'9') c=getchar();
while(c>='0'&&c<='9') x=(x<<3)+(x<<1)+c-48,c=getchar();return x;
}
const int maxn=1e5+5;
struct E{int v,nxt,w;}e[maxn<<1];
int n,m,num,Rt,rt,sz,S;
int head[maxn],sum[maxn],son[maxn],pos[maxn],dep[maxn],d[maxn],Ans[55];
LL pre[maxn],mx[maxn],q[55],c[maxn];
inline void add(int x,int y,int w) {
e[++num].v=y;e[num].nxt=head[x];head[x]=num;e[num].w=w;
}
inline int find(LL x) {
int l=1,r=sz,now=0;
while(l<=r) {
int mid=l+r>>1;
if(c[mid]<=x) now=mid,l=mid+1;else r=mid-1;
}
return now;
}
void dfs(int x,int fa) {
for(re int i=head[x];i;i=e[i].nxt) {
if(e[i].v==fa) continue;
pre[e[i].v]=pre[x]+e[i].w;
dfs(e[i].v,x);
}
}
void dfs1(int x) {
sum[x]=1;
for(re int i=head[x];i;i=e[i].nxt) {
if(dep[e[i].v]) continue;
dep[e[i].v]=dep[x]+1;dfs1(e[i].v);sum[x]+=sum[e[i].v];
if(sum[e[i].v]>sum[son[x]]) son[x]=e[i].v;
}
}
inline void add(int x,int v) {
for(re int i=x;i<=sz;i+=lb(i)) d[i]+=v;
}
inline int ask(int x) {
int now=0;
for(re int i=x;i;i-=lb(i)) now+=d[i];
return now;
}
void calc(int x,int o) {
add(pos[x],o);
for(re int i=head[x];i;i=e[i].nxt)
if(dep[e[i].v]>dep[x]&&S!=e[i].v) calc(e[i].v,o);
}
void dsu(int x,int k) {
for(re int i=head[x];i;i=e[i].nxt)
if(dep[e[i].v]>dep[x]&&son[x]!=e[i].v) dsu(e[i].v,0);
if(son[x]) dsu(son[x],1);
S=son[x],calc(x,1),S=0;
for(re int i=1;i<=m;i++) {
int g=ask(find(mx[x]+q[i]));
Ans[i]=max(Ans[i],g);
}
if(!k) calc(x,-1);
}
int main() {
n=read();
for(re int x,y,w,i=1;i<n;++i)
x=read(),y=read(),w=read(),add(x,y,w),add(y,x,w);
m=read();for(re int i=1;i<=m;i++) scanf("%lld",&q[i]);
dfs(1,0);rt=1;
for(re int i=2;i<=n;i++) if(pre[i]>pre[rt]) rt=i;
pre[rt]=0;dfs(rt,0);Rt=1;
for(re int i=1;i<=n;i++) mx[i]=pre[i];
for(re int i=2;i<=n;i++) if(pre[i]>pre[Rt]) Rt=i;
pre[Rt]=0;dfs(Rt,0);
for(re int i=1;i<=n;i++) mx[i]=max(mx[i],pre[i]);
rt=1;for(re int i=2;i<=n;i++) if(mx[i]<mx[rt]) rt=i;
dep[rt]=1;dfs1(rt);
for(re int i=1;i<=n;i++) c[i]=mx[i];
std::sort(c+1,c+n+1);sz=std::unique(c+1,c+n+1)-c-1;
for(re int i=1;i<=n;i++) pos[i]=find(mx[i]);
dsu(rt,1);for(re int i=1;i<=m;i++) printf("%d\n",Ans[i]);
return 0;
}

【CF516D】Drazil and Morning Exercise的更多相关文章

  1. 「CF516D」 Drazil and Morning Exercise

    「CF516D」 Drazil and Morning Exercise 传送门 这个 \(f_i\) 显然可以通过树形 \(\texttt{DP}\) 直接求. 然后看到这种差值问题感觉就可以二分转 ...

  2. 【Cf #292 D】Drazil and Morning Exercise(树的直径,树上差分)

    有一个经典的问题存在于这个子问题里,就是求出每个点到其他点的最远距离. 这个问题和树的直径有很大的关系,因为事实上距离每个点最远的点一定是直径的两个端点.所以我们可以很容易地进行$3$遍$Dfs$就可 ...

  3. 【UFLDL】Exercise: Convolutional Neural Network

    这个exercise需要完成cnn中的forward pass,cost,error和gradient的计算.需要弄清楚每一层的以上四个步骤的原理,并且要充分利用matlab的矩阵运算.大概把过程总结 ...

  4. 【原】Coursera—Andrew Ng机器学习—编程作业 Programming Exercise 1 线性回归

    作业说明 Exercise 1,Week 2,使用Octave实现线性回归模型.数据集  ex1data1.txt ,ex1data2.txt 单变量线性回归必须实现,实现代价函数计算Computin ...

  5. 【codeforces 515D】Drazil and Tiles

    [题目链接]:http://codeforces.com/contest/515/problem/D [题意] 给你一个n*m的格子; 然后让你用1*2的长方形去填格子的空缺; 如果有填满的方案且方案 ...

  6. 【codeforces 515C】Drazil and Factorial

    [题目链接]:http://codeforces.com/contest/515/problem/C [题意] 定义f(n)=n这个数各个位置上的数的阶乘的乘积; 给你a; 让你另外求一个不含0和1的 ...

  7. 【codeforces 515B】Drazil and His Happy Friends

    [题目链接]:http://codeforces.com/contest/515/problem/B [题意] 第i天选择第i%n个男生,第i%m个女生,让他们一起去吃饭; 只要这一对中有一个人是开心 ...

  8. 【codeforces 515A】Drazil and Date

    [题目链接]:http://codeforces.com/contest/515/problem/A [题意] 每次只能走到相邻的四个格子中的一个; 告诉你最后走到了(a,b)走了多少步->s ...

  9. Python高手之路【三】python基础之函数

    基本数据类型补充: set 是一个无序且不重复的元素集合 class set(object): """ set() -> new empty set object ...

随机推荐

  1. 1、postman介绍与安装

    postman介绍 官方介绍:Developers use Postman to build modern software for the API-first world. 个人理解postman是 ...

  2. sed use case: Filter without editing

    if we want to filter with sed pattern and just print the filtered lines without any further editing ...

  3. ICPC Asia Nanning 2017 L. Twice Equation (规律 高精度运算)

    题目链接:Twice Equation 比赛链接:ICPC Asia Nanning 2017 Description For given \(L\), find the smallest \(n\) ...

  4. 拾遗:Git 常用操作回顾

    温故而知新,可以为师矣. Git 布局 工作区---->暂存区---->本地仓库---->远程仓库 Create Repository git init PATH git add P ...

  5. Java通道

    通道(Channel)是数据源和Java程序之间的开放连接,用于执行I/O操作.Channel接口在java.nio.channels包中.通道(Channel)接口只声明了两个方法:close()和 ...

  6. Serializable 和Parcelable 详解

    序列化:为了保存在内存中的各种对象的状态,并可以把保存的对象的状态读出来 安卓中实现序列化的接口有两个,一个是serializable,一个是parcelable. 一.实现序列化: 1.是可以将对象 ...

  7. web前端Vue+Django rest framework 框架 生鲜电商项目实战✍✍✍

    web前端Vue+Django rest framework 框架 生鲜电商项目实战  整个课程都看完了,这个课程的分享可以往下看,下面有链接,之前做java开发也做了一些年头,也分享下自己看这个视频 ...

  8. 33-Ubuntu-用户权限-04-修改目录权限

    修改目录权限 例:test 1.可执行权限---x 减少目录x权限,无法切换到该目录 chmod -x test 2.读权限 ---r 减少目录r权限,无法读取(ls)目录信息 chmod -r te ...

  9. layui 封装自定义模块

    转自:https://lianghongbo.cn/blog/430585105a35948c layui是国人开发的一款非常简洁的UI框架,使用了模块化加载方式,因此在使用过程中我们难免需要添加自己 ...

  10. CKEditor与CKFinder学习--自定义界面及按钮事件捕获

    原文地址:CKEditor与CKFinder学习--自定义界面及按钮事件捕获  讨厌CSDN的广告,吃香太难看! 效果图 界面操作图 原始界面 调整后的界面(删除了flush,表单元素等) 该界面的皮 ...