当前位置: 首页 > news >正文

P6626 [省选联考 2020 B 卷] 消息传递

~~~~~      原题链接 ~~~~~      总题单链接

题解

~~~~~      先将询问离线,这样就可以在点分治的时候一次性处理。

~~~~~      设当前遍历到的重心是 p p p ,我们要查询所有 经过 p p p 的路径。那么对于 p p p 的子树内的一个点 x x x,它需要加上的贡献就是距离 p p p k − d e p [ x ] k-dep[x] kdep[x] 的点的数量, k k k 就是题面中的 k k k,注意实际操作的时候要先把 x x x 的子树去掉。而对于 p p p 本身,它要加上的贡献自然就是距离 p p p k k k 的点的数量。

代码

#include<bits/stdc++.h>
#include<bits/stdc++.h>
#define ll long long
#define fir first
#define sec second
using namespace std;inline ll read(){char c=getchar();ll res=0;while(c<'0'||c>'9')c=getchar();while(c>='0'&&c<='9'){res=(res<<1)+(res<<3)+(c^48);c=getchar();}return res;
}
inline void write(ll k){if(k>=10)write(k/10);putchar((k%10)|0x30);
}pair<ll,ll>ask[100005];
vector<ll>eg[100005],wit[100005],qwe;
ll dis[100005],dit,dge[100005],ans[100005];
ll n,m,siz[100005],sum,mxs,root,dep[100005];
ll vis[100005],stk[100005],top,init[100005];void getroot(ll fa,ll p){siz[p]=1;ll s=0;for(ll v:eg[p]){if(v==fa||vis[v])continue;getroot(p,v);siz[p]+=siz[v];s=max(s,siz[v]);}s=max(s,sum-s);if(s<mxs)mxs=s,root=p;
}
void getdep(ll fa,ll p){dep[p]=dep[fa]+1;for(ll v:eg[p]){if(v==fa||vis[v])continue;getdep(p,v);}
}
void getdis(ll fa,ll p,ll now,ll opt){if(opt){for(ll it:wit[p])qwe.push_back(it);}dis[++dit]=now;for(ll v:eg[p]){if(v==fa||vis[v])continue;getdis(p,v,now+1,opt);}
}
void getans(ll p){vis[p]=1;top=0;dge[0]=1;getdep(0,p);for(ll v:eg[p]){if(vis[v])continue;dit=0;getdis(p,v,1,0);for(ll i=1;i<=dit;i++){dge[dis[i]]++;if(!init[dis[i]]){init[dis[i]]=1;stk[++top]=dis[i];}}}for(ll it:wit[p])ans[it]+=dge[ask[it].sec];for(ll v:eg[p]){if(vis[v])continue;qwe.clear();dit=0;getdis(p,v,1,1);for(ll i=1;i<=dit;i++)dge[dis[i]]--;for(ll it:qwe)if(ask[it].sec-dep[ask[it].fir]+1>=0)ans[it]+=dge[ask[it].sec-dep[ask[it].fir]+1];for(ll i=1;i<=dit;i++)dge[dis[i]]++;}for(ll i=1;i<=top;i++)dge[stk[i]]=init[stk[i]]=0;
}void divide(ll p){getans(p);for(ll v:eg[p]){if(vis[v])continue;root=v;sum=mxs=siz[v];getroot(p,v);divide(root);}
}void work(){cin>>n>>m;memset(ans,0,sizeof(ans));memset(siz,0,sizeof(siz));memset(vis,0,sizeof(vis));for(ll i=1;i<=n;i++)eg[i].clear(),wit[i].clear();for(ll i=1;i<n;i++){ll x=read(),y=read();eg[x].push_back(y);eg[y].push_back(x);}for(ll i=1;i<=m;i++){ask[i].fir=read(),ask[i].sec=read();wit[ask[i].fir].push_back(i);}divide(1);for(ll i=1;i<=m;i++)write(ans[i]),putchar('\n');
}signed main(){ll T;cin>>T;while(T--)work();return 0;
}

~~~~~      引用请附名 —— —— —— OMG_NOIP


http://www.mrgr.cn/news/12148.html

相关文章:

  • mac Let‘s Encrypt 免费SSL证书申请
  • Java集成百度地图API入门指南
  • 苹果秋季发布会前瞻:iPhone 16领衔新品盛宴
  • 什么是数据库 DevOps?
  • 分布式设计原理——CAP原则
  • 数据导出为Excel接口报错:java.io.IOException: UT010029: Stream is closed
  • 【第54课】XSS跨站Cookie盗取表单劫持网络钓鱼溯源分析项目平台框架
  • Java中金蝶凭证xml转wswsvoucher对象
  • 【区块链 + 智慧文旅】虎年春节数字藏品 | FISCO BCOS应用案例
  • nlp时序模型股价预测的基本思路(持续更新)
  • Python网络爬虫模拟登录与验证解析
  • 【3.3】贪心算法-解分发糖果
  • Apache Doris 使用 CBO 和 RBO 结合的优化策略
  • 此站点的连接不安全,解决方法
  • Sentinel-1 Level 1数据处理的详细算法定义(七)
  • 基于Vue3和Node.js的完整增删改查项目实现教程:从后端封装到前端调用
  • WHAT - 通过 react-use 源码学习 React
  • 配电房挂轨机器人巡检系统的主要优点包括
  • 足球数据分析-基于机器学习的足球比赛角球数预测模型构建
  • 前端:html+css:伪类画箭头(实心)