2019.01.20 bzoj3784: 树上的路径(二分答案+点分治)
传送门
点分治好题。
题意简述:给一棵带边权的树,问所有路径中前mmm大的。m≤300000m\le300000m≤300000
思路:
网上有题解写了可以通过什么点分治序转化成超级钢琴那道题的做法蒟蒻吓得瑟瑟发抖。
然后由于我比较菜想了一个二分答案的方法。
我们二分第mmm大的值,每次用点分治检验合法性。
二分完了之后我们再跑一次点分统计答案。
然后第一个二分的时候直接做是logn3log^3_nlogn3的。
考虑降下来一个logloglog。
我们先dfsdfsdfs一次树把每个点作为重心的时候的所有distdistdist预处理下来就可以省掉一个logloglog啦! 空间复杂度O(nlogn)233
代码:
#include<bits/stdc++.h>
#define ri register int
#define fi first
#define se second
using namespace std;
inline int read(){
int ans=0;
char ch=getchar();
while(!isdigit(ch))ch=getchar();
while(isdigit(ch))ans=(ans<<3)+(ans<<1)+(ch^48),ch=getchar();
return ans;
}
typedef long long ll;
typedef pair<int,int> pii;
const int N=5e4+5,M=3e5+5;
int n,m,siz[N],msiz[N],stk[M],top=0,rt,all,Rt;
bool vis[N];
vector<pii>e[N];
vector<int>dis[N],g[N],inv[N];
ll ans;
priority_queue<int>ANS;
void getroot(int p,int fa){
siz[p]=1,msiz[p]=0;
for(ri i=0,v;i<e[p].size();++i){
if(vis[v=e[p][i].fi]||v==fa)continue;
getroot(v,p),siz[p]+=siz[v],msiz[p]=max(msiz[p],siz[v]);
}
msiz[p]=max(msiz[p],all-siz[p]);
if(msiz[p]<msiz[rt])rt=p;
}
void solve(int p,int fa,int dist,vector<int>&anc){
anc.push_back(dist);
for(ri i=0,v;i<e[p].size();++i){
if(vis[v=e[p][i].fi]||v==fa)continue;
solve(v,p,dist+e[p][i].se,anc);
}
}
void dfs(int p){
vis[p]=1,solve(p,0,0,dis[p]);
for(ri i=0,v;i<e[p].size();++i){
if(vis[v=e[p][i].fi])continue;
rt=0,all=siz[v],getroot(v,p),solve(v,p,e[p][i].se,inv[rt]),g[p].push_back(rt),dfs(rt);
}
}
inline ll calc(vector<int>&v,int lim){
ll ret=0;
for(ri i=0;i<v.size();++i)ret+=v.end()-lower_bound(v.begin(),v.end(),lim-v[i]);
return ret;
}
void Dfs(int p,int lim){
ans+=calc(dis[p],lim);
for(ri i=0;i<g[p].size();++i)ans-=calc(inv[g[p][i]],lim);
if(ans>=m)return;
for(ri i=0;i<g[p].size();++i)Dfs(g[p][i],lim);
}
inline bool check(int mid){return ans=0,Dfs(Rt,mid),ans>=m*2;}
void DFS(int p,int fa,int dist){
stk[++top]=dist;
for(ri i=0,v;i<e[p].size();++i)if((v=e[p][i].fi)!=fa&&!vis[v])DFS(v,p,dist+e[p][i].se);
}
inline void Solve(int p,int lim){
stk[top=1]=0;
for(ri i=0,len,v;i<e[p].size();++i){
if(vis[v=e[p][i].fi])continue;
len=top,DFS(v,p,e[p][i].se);
for(ri j=len+1,pos;j<=top;j++){
pos=lower_bound(stk+1,stk+len+1,lim-stk[j])-stk;
for(ri k=pos;k<=len;++k)ANS.push(stk[j]+stk[k]);
}
sort(stk+len+1,stk+top+1),inplace_merge(stk+1,stk+len+1,stk+top+1);
}
}
inline void ask(int p,int lim){vis[p]=1,Solve(p,lim);for(ri i=0;i<g[p].size();++i)ask(g[p][i],lim);}
int main(){
n=read(),m=read();
int L=0,R=0,K=0;
for(ri i=1,u,v,w;i<n;++i)u=read(),v=read(),w=read(),e[u].push_back(pii(v,w)),e[v].push_back(pii(u,w)),R+=w;
msiz[rt=0]=all=n,getroot(1,0),Rt=rt,dfs(Rt);
for(ri i=1;i<=n;++i)sort(dis[i].begin(),dis[i].end()),sort(inv[i].begin(),inv[i].end());
while(L<=R){
int mid=L+R>>1;
if(check(mid))K=mid,L=mid+1;
else R=mid-1;
}
memset(vis,0,sizeof(vis)),ask(Rt,K);
for(ri i=1;i<=m;++i){
if(ANS.size())cout<<ANS.top()<<'\n',ANS.pop();
else cout<<K<<'\n';
}
return 0;
}
2019.01.20 bzoj3784: 树上的路径(二分答案+点分治)的更多相关文章
- BZOJ 4326 NOIP2015 运输计划(树上差分+LCA+二分答案)
4326: NOIP2015 运输计划 Time Limit: 30 Sec Memory Limit: 128 MB Submit: 1388 Solved: 860 [Submit][Stat ...
- 2019.03.04 bzoj5308: [Zjoi2018]胖(二分答案+st表)
传送门 想题5分钟调题两小时系列 其实还是我tcl 读完题之后自然会知道一个关键点能够更新的点是一段连续的区间,于是我们对于每个点能到的左右区间二分答案,用ststst表维护一下查询即可. 代码: # ...
- 洛谷P2680 运输计划 [LCA,树上差分,二分答案]
题目传送门 运输计划 Description 公元 2044 年,人类进入了宇宙纪元.L 国有 n 个星球,还有 n?1 条双向航道,每条航道建立在两个星球之间, 这 n?1 条航道连通了 L 国的所 ...
- 2019.01.20 NOIP模拟 迅雷(kruskal/二分+并查集)
传送门 题意简述:给一张带权无向图,有a,ba,ba,b两类特殊点和普通点,问使得至少有一个aaa和一个bbb连通所需要的所有边边权最小值的最大值是多少. 思路: 一眼发现可以二分,考虑怎么check ...
- BZOJ3784 : 树上的路径
树的点分治,在分治的时候将所有点到根的距离依次放入一个数组q中. 对于一棵子树里的点,合法的路径一定是q[L]..q[R]的某个数加上自己到重心的距离. 定义五元组(v,l,m,r,w),表示当前路径 ...
- 2019.01.20 bzoj2238: Mst(kruskal+树链剖分)
传送门 树链剖分菜题. 题意简述:给一个无向图,边有边权,每次询问删一条边(对后面的询问无影响)之后的最小生成树. 思路: 先跑一次kruskalkruskalkruskal并把跑出来的最小生成树给链 ...
- 2019.01.20 bzoj2388: 旅行规划(分块+凸包)
传送门 分块好题. 题意:维护区间加,维护区间前缀和的最大值(前缀和指从1开始的). 思路: 考虑分块维护答案. 我们把每个点看成(i,sumi)(i,sum_i)(i,sumi)答案一定会在凸包上 ...
- 2019.01.20 bzoj3999: [TJOI2015]旅游(树链剖分)
传送门 树链剖分菜题. 题意不清差评. 题意简述(保证清晰):给一棵带权的树,每次从aaa走到bbb,在走过的路径上任意找两个点,求后访问的点与先访问的点点权差的最大值. 思路: 考虑暴力:维护路径的 ...
- 2019.01.09 bzoj3697: 采药人的路径(点分治)
传送门 点分治好题. 题意:给出一棵树,边分两种,求满足由两条两种边数相等的路径拼成的路径数. 思路: 考虑将边的种类转化成边权−1-1−1和111,这样就只用考虑由两条权值为000的路径拼成的路径数 ...
随机推荐
- Codeforces Beta Round #44 (Div. 2)
Codeforces Beta Round #44 (Div. 2) http://codeforces.com/contest/47 A #include<bits/stdc++.h> ...
- ThreaLocal
ThreadLocal概念:线程局部变量,是一种多线程间并发访问变量的解决方案.与其synchronized等加锁的方式不同,ThreadLocal完全不提供锁,而使用以空间换时间的手段,为每个线程提 ...
- Exception starting filter encodingFilter
1. maven运行tomcat6出现错误Exception starting filter encodingFilter: 严重: Exception starting filter encodin ...
- git中 vi/vim的命令
一.vi & vim 有两种工作模式: 1.命令模式:接受.执行 vi操作命令的模式,打开文件后的默认模式: 2.编辑模式:对打开的文件内容进行 增.删.改 操作的模式: 在编辑模式下按下ES ...
- git查看历史操作
在提交了若干更新,又或者克隆了某个项目之后,偶尔想回顾下过往提交历史.可以使用git log命令来实现. 最简单的查看提交历史命令如下: $ git log $ git log --oneline $ ...
- make ;makefile; cmake; qmake的区分
1. make 是用来执行Makefile的.2. Makefile是类unix环境下(比如Linux)的类似于批处理的"脚本"文件.其基本语法是: 目标+依赖+命令,只有在目标文 ...
- Couchbase学习和使用
Couchbase介绍 couchbase的关键有两点:延后写入和松散存储.延后写入,顾名思义,couchbase在对数据进行增删时会先体现在内存中,而不会立刻体现在硬盘上,从内存的修改到硬盘的修改这 ...
- php 多进程 父进程的阻塞与非阻塞
php中进程的阻塞,主要是父进程等待子进程退出. 1.php代码如下: <?php //定义进程数量 define('FORK_NUMS', 5); //用于保存进程pid $pids = ar ...
- 函数 day9
一,什么是函数? 函数的定义与调用 s = 'fkdsagadfdsagfdsagg' count = 0 for i in s: count += 1 print(count) l1 = [1,2, ...
- oracle 分页 where 三层
查询[start,start+limit],包含start,包含start+limit,如start=21,limit=10结果就是21到30,包含21和30SELECT * FROM (SELECT ...