【BZOJ3784】树上的路径

Description

给定一个N个结点的树,结点用正整数1..N编号。每条边有一个正整数权值。用d(a,b)表示从结点a到结点b路边上经过边的权值。其中要求a<b.将这n*(n-1)/2个距离从大到小排序,输出前M个距离值。

Input

第一行两个正整数N,M
下面N-1行,每行三个正整数a,b,c(a,b<=N,C<=10000)。表示结点a到结点b有一条权值为c的边。

Output

共M行,如题所述.

Sample Input

5 10
1 2 1
1 3 2
2 4 3
2 5 4

Sample Output

7
7
6
5
4
4
3
3
2
1

HINT

N<=50000,M<=Min(300000,n*(n-1) /2 )

题解:“总有那么一种序列,可以满足你某道题的所需的一切性质~”

没错,听说过DFS序吗,听说过BFS序吗,如果你都听说过,那么你听说过点分治序吗?

没错,点分治序,顾名思义,就是点分治时扫过的序列(包括每次找到的重心和从重心出发依次DFS过的所有子树),它的长度是nlogn的。那么这样一个序列有什么性质呢?

如果我们已经确定了一个分治中心和它的子树中的一条链,我们想找到这个分治中心的另一条链(亦或是这个分治中心本身)和它来组成一条路径,那么这些路径的端点在什么位置呢?没错,他们在点分治序上正好是一段连续的序列!并且根据点分治的原理,通过这样我们可以找到树上所有的路径。

那么问题就变成了:给你一个序列,你每次可以从中选取一个二元组(a,b),其中对于每一个b,可以与它搭配的a都在一段给定的区间里。每个二元组的值是a的权值+b的权值,求前k大的二元组。这不就是BZOJ2006超级钢琴吗?直接ST表就好了。

#include <cstdio>
#include <cstring>
#include <iostream>
#include <utility>
#include <queue>
#define mp(A,B,C,D) make_pair(make_pair(A,B),make_pair(C,D))
using namespace std;
const int maxn=50010;
typedef pair<int,int> pii;
int n,m,cnt,maxx,tot,root,nm;
int sm[800010][20],to[maxn<<1],next[maxn<<1],val[maxn<<1],head[maxn],siz[maxn];
int lp[800010],rp[800010],v[800010],vis[maxn],Log[800010];
priority_queue<pair<pii,pii> > pq;
int rd()
{
int ret=0,f=1; char gc=getchar();
while(gc<'0'||gc>'9') {if(gc=='-')f=-f; gc=getchar();}
while(gc>='0'&&gc<='9') ret=ret*10+gc-'0',gc=getchar();
return ret*f;
}
int ms(int a,int b)
{
return v[a]>v[b]?a:b;
}
void add(int a,int b,int c)
{
to[cnt]=b,val[cnt]=c,next[cnt]=head[a],head[a]=cnt++;
}
void getr(int x,int fa)
{
siz[x]=1;
int i,mx=0;
for(i=head[x];i!=-1;i=next[i])
{
if(vis[to[i]]||to[i]==fa) continue;
getr(to[i],x),siz[x]+=siz[to[i]],mx=max(mx,siz[to[i]]);
}
if(maxx>max(mx,tot-siz[x])) root=x,maxx=max(mx,tot-siz[x]);
}
void getd(int x,int fa,int dep)
{
v[++nm]=dep,lp[nm]=lp[nm-1],rp[nm]=rp[nm]?rp[nm]:rp[nm-1];
for(int i=head[x];i!=-1;i=next[i])
{
if(vis[to[i]]||to[i]==fa) continue;
getd(to[i],x,dep+val[i]);
}
}
void dfs(int x)
{
vis[x]=1;
int i;
v[++nm]=0,lp[nm]=nm,rp[nm]=nm-1;
for(i=head[x];i!=-1;i=next[i])
{
if(vis[to[i]]) continue;
rp[nm+1]=nm,getd(to[i],x,val[i]);
}
for(i=head[x];i!=-1;i=next[i])
{
if(vis[to[i]]) continue;
tot=siz[to[i]],maxx=1<<30,getr(to[i],x),dfs(root);
}
}
int query(int a,int b)
{
if(a>b) return 0;
int k=Log[b-a+1];
return ms(sm[a][k],sm[b-(1<<k)+1][k]);
}
int main()
{
n=rd(),m=rd();
int i,j,a,b,c,d,x,y;
memset(head,-1,sizeof(head));
for(i=1;i<n;i++) a=rd(),b=rd(),c=rd(),add(a,b,c),add(b,a,c);
tot=n,maxx=1<<30,getr(1,0),dfs(root);
for(i=1;i<=nm;i++) sm[i][0]=i;
for(i=2;i<=nm;i++) Log[i]=Log[i>>1]+1;
for(j=1;(1<<j)<nm;j++)
for(i=1;i+(1<<j)-1<=nm;i++)
sm[i][j]=ms(sm[i][j-1],sm[i+(1<<j-1)][j-1]);
for(i=1;i<=nm;i++)
{
if(lp[i]>rp[i]) continue;
pq.push(mp(v[i]+v[query(lp[i],rp[i])],i,lp[i],rp[i]));
}
for(i=1;i<=m;i++)
{
pii t1=pq.top().first,t2=pq.top().second;
pq.pop();
printf("%d\n",t1.first),x=t1.second,a=t2.first,b=t2.second,y=query(a,b);
c=query(a,y-1),d=query(y+1,b);
if(c) pq.push(mp(v[x]+v[c],x,a,y-1));
if(d) pq.push(mp(v[x]+v[d],x,y+1,b));
}
return 0;
}

【BZOJ3784】树上的路径 点分治序+ST表的更多相关文章

  1. 【BZOJ-3784】树上的路径 点分治 + ST + 堆

    3784: 树上的路径 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 462  Solved: 153[Submit][Status][Discuss ...

  2. 2019.01.20 bzoj3784: 树上的路径(二分答案+点分治)

    传送门 点分治好题. 题意简述:给一棵带边权的树,问所有路径中前mmm大的.m≤300000m\le300000m≤300000 思路: 网上有题解写了可以通过什么点分治序转化成超级钢琴那道题的做法蒟 ...

  3. BZOJ.3784.树上的路径(点分治 贪心 堆)

    BZOJ \(Description\) 给定一棵\(n\)个点的带权树,求树上\(\frac{n\times(n-1)}{2}\)条路径中,长度最大的\(m\)条路径的长度. \(n\leq5000 ...

  4. BZOJ3784树上的路径

    题目描述 给定一个N个结点的树,结点用正整数1..N编号.每条边有一个正整数权值.用d(a,b)表示从结点a到结点b路边上经过边的权值.其中要求a<b.将这n*(n-1)/2个距离从大到小排序, ...

  5. 【BZOJ3611】[Heoi2014]大工程 欧拉序+ST表+单调栈

    [BZOJ3611][Heoi2014]大工程 Description 国家有一个大工程,要给一个非常大的交通网络里建一些新的通道.  我们这个国家位置非常特殊,可以看成是一个单位边权的树,城市位于顶 ...

  6. BZOJ3784 : 树上的路径

    树的点分治,在分治的时候将所有点到根的距离依次放入一个数组q中. 对于一棵子树里的点,合法的路径一定是q[L]..q[R]的某个数加上自己到重心的距离. 定义五元组(v,l,m,r,w),表示当前路径 ...

  7. BZOJ 3784: 树上的路径 点分治+二分+set

    很容易想出二分这个思路,但是要想办法去掉一个 $log$. 没错,空间换时间. 双指针的部分错了好几次~ Code: #include <set> #include <queue&g ...

  8. 51Nod1766 树上的最远点对 ST表 LCA 线段树

    原文链接https://www.cnblogs.com/zhouzhendong/p/51Nod1766.html 题目传送门 - 51Nod1766 题意 n个点被n-1条边连接成了一颗树,给出a~ ...

  9. [51nod 1766]树上的最远点对 (树的直径+ST表求lca+线段树)

    [51nod 1766]树上的最远点对 (树的直径+ST表求lca+线段树) 题面 给出一棵N个点的树,Q次询问一点编号在区间[l1,r1]内,另一点编号在区间[l2,r2]内的所有点对距离最大值.\ ...

随机推荐

  1. 输入法不能使用ctrl+shift进行切换的问题

    第一种情况就是,你的输入法只有一种(而且这种输入法并不是“中文(简体) 微软拼音输入法”). 如果是只有一种输入法的话,是无法进行切换的,如果你是想要把输入法切换到无输入法状态,那么你可以通过设置任务 ...

  2. CKEditor+SWFUpload实现功能较为强大的编辑器(三)---后台接收图片流程

    在前台配置完CKEditor和SWFUpload之后就可以满足基本的需求了 在这里,我配置的接收异步上传的图片的页面为upload.ashx 在这个ashx中对上传的图片处理的流程如下: contex ...

  3. DevExpress控件之LayoutControl

    一.项目运行中不显示右键菜单 layoutControl1.AllowCustomization = false 二.控件超出容器后不显示滚动条 layoutControl1.AtuoScroll = ...

  4. defer,panic,recover

    Go语言不支持传统的 try…catch…finally 这种异常,因为Go语言的设计者们认为,将异常与控制结构混在一起会很容易使得代码变得混乱.因为开发者很容易滥用异常,甚至一个小小的错误都抛出一个 ...

  5. python的偏函数(partial)

    偏函数就是固定原函数的某个参数,比如newadd就是固定了add方法的第一个参数,让a=3,这样对newadd方法只要传入参数B就可以实现add方法了,刚看偏函数的写法可能会不适应,因为partial ...

  6. 运用JS导出ecxel表格、实现文件重命名

    <!DOCTYPE html> <html> <head lang="en"> <meta charset="UTF-8&quo ...

  7. TCO'10 Wildcard Round 1000pt

    题目大意: 给定一个N*M的棋盘,棋子可以攻击其左右距离不超过K的棋子.问有多少种放法使得棋盘上的棋子不能互相攻击. N,M,K都在1到1000000000的范围内,结果对100003取模. 官方题解 ...

  8. Linux——vi命令的使用

    vi的基本操作 a) 进入vi 在系统提示符号输入vi及文件名称后,就进入vi全屏幕编辑画面: $ vi myfile 不过有一点要特别注意,就是您进入vi之后,是处于「命令行模式(command m ...

  9. 数据库设计--数据流图(DFD)

    1.数据流图的定义 数据流图(DFD)是结构化分析方法中使用的工具,它以图形的方式描绘数据在系统中流动和处理的过程, 因为它仅仅反映系统必须完毕的逻辑功能.所以它是一种功能模型. 在结构化开发方法中. ...

  10. JavaScript 获取文件名,后缀名

    function getBaseName(str) { var segs = str.split('.'); if(segs.length > 1) segs.pop(); return seg ...