Race bzoj-2599

    题目大意:询问一颗树上最短的、长度为k的链,边有边权,n个节点。

    注释:$1\le n \le 2\cdot 10^5$,$1\le k \le 10^6$。

      想法:树上点分治的另一种表现方式。首先,由于题目中要求的是最小值,我们发现这东西可加不可减。不可减意味着什么?意味着我们递归计算当前树时无法将它的单个子树的情况减掉。所以之前的单步容斥的算法就收到了打压qwq。我们思考另一种方法。首先,类似于dfs的,我一定是对于当前root一颗子树一颗子树地递归,只有当前子树的信息已经完全处理好的情况下我才会去处理下一颗子树。这就相当于我在处理当前子树的时候之前的子树已经是完善的了。所以,我可以开一个桶,记录长度为i的链的最短长度,然后对于当前子树我可以直接调用桶中信息,即可。然后,还原桶的操作是简单的,就是说如果当前桶所代表的边权小于等于k,那么久有可能是被更新过的,将其还原即可。

    最后,附上丑陋的代码... ...

#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#define N 200010
using namespace std;
const int inf=1<<30;
int to[2*N],nxt[2*N],head[N],val[N*2],tot;
int size[N],vis[N],v[N*5],f[N],dis[N],deep[N];
//dis数组表示当前节点到根节点之间的边权和
//deep数组表示当前节点到根节点路径深度,即链长度
//vis数组表示当前节点是否已经被删除(当过根节点)
//v数组是桶
int ms,root;
int n,k;
int ans=inf;
inline void add(int x,int y,int z)//加边
{
to[++tot]=y;
val[tot]=z;
nxt[tot]=head[x];
head[x]=tot;
}
void getroot(int pos,int fa)//找重心、处理size
{
size[pos]=1;
f[pos]=0;
for(int i=head[pos];i;i=nxt[i])
{
if(to[i]==fa||vis[to[i]]) continue;
getroot(to[i],pos);
size[pos]+=size[to[i]];
f[pos]=max(size[to[i]],f[pos]);
}
f[pos]=max(f[pos],ms-size[pos]);
if(f[root]>f[pos]) root=pos;
// puts("getroot");
}
void clear(int pos,int fa)//还原桶
{
if(dis[pos]<=k) v[dis[pos]]=inf;
for(int i=head[pos];i;i=nxt[i])
{
if(to[i]==fa||vis[to[i]]) continue;
clear(to[i],pos);
}
// puts("clear");
}
void insert(int pos,int fa)//修改桶
{
if(dis[pos]<=k)
{
v[dis[pos]]=min(v[dis[pos]],deep[pos]);
}
for(int i=head[pos];i;i=nxt[i])
{
if(vis[to[i]]||to[i]==fa) continue;
insert(to[i],pos);
// puts("insert");
}
}
void calc(int pos,int fa)//计算答案
{
if(dis[pos]<=k)
{
ans=min(ans,deep[pos]+v[k-dis[pos]]);
}
for(int i=head[pos];i;i=nxt[i])
{
if(vis[to[i]]||to[i]==fa) continue;
deep[to[i]]=deep[pos]+1;
dis[to[i]]=dis[pos]+val[i];
calc(to[i],pos);
}
// puts("calc");
}
void solve(int pos)//点分治过程
{
vis[pos]=true;
// deep[pos]=0;
v[0]=0;
for(int i=head[pos];i;i=nxt[i])
{
if(vis[to[i]]) continue;
deep[to[i]]=1;
dis[to[i]]=val[i];
calc(to[i],0);
insert(to[i],0);
}
for(int i=head[pos];i;i=nxt[i])
{
if(vis[to[i]]) continue;
clear(to[i],0);
}
for(int i=head[pos];i;i=nxt[i])
{
if(vis[to[i]]) continue;
ms=size[to[i]];
root=0;
getroot(to[i],0);
solve(root);
}
// puts("solve");
}
int main()
{
// int n,k;
scanf("%d%d",&n,&k);
for(int x,y,z,i=1;i<n;i++)
{
scanf("%d%d%d",&x,&y,&z);
add(x+1,y+1,z);//点编号是从0开始的
add(y+1,x+1,z);
}
// v[0]=0;
for(int i=1;i<=k;i++)
{
v[i]=inf;
}
root=0;
f[0]=n;
// dfs(1);
ms=n;
getroot(1,0);
// printf("%d\n",root);
solve(root);
if(ans==inf) printf("-1\n");
else printf("%d\n",ans);
return 0;
}

    小结:一定要注意dis数组和deep数组分别的含义。然后主函数里所有函数的fa都是0,因为是递归处理,但是其实将fa改成pos也没有问题,因为pos节点已经当过重心、被删除了。

[bzoj2599][IOI2011]Race_树上点分治的更多相关文章

  1. bzoj2599: [IOI2011]Race(点分治)

    写了四五道点分治的题目了,算是比较理解点分治是什么东西了吧= = 点分治主要用来解决点对之间的问题的,比如距离为不大于K的点有多少对. 这道题要求距离等于K的点对中连接两点的最小边数. 那么其实道理是 ...

  2. [luogu4149][bzoj2599][IOI2011]Race【点分治】

    题目描述 给一棵树,每条边有权.求一条简单路径,权值和等于 K,且边的数量最小. 题解 比较明显需要用到点分治,我们定义\(d\)数组表示当前节点到根节点\(rt\)之间有多少个节点,也可以表示有多少 ...

  3. 2019.01.09 bzoj2599: [IOI2011]Race(点分治)

    传送门 题意:给一棵树,每条边有权.求一条路径,权值和等于K,且边的数量最小. 思路: 考虑点分治如何合并. 我们利用树形dpdpdp求树的直径的方法,边dfsdfsdfs子树边统计答案即可. 代码: ...

  4. BZOJ2599 [IOI2011]Race 【点分治】

    题目 给一棵树,每条边有权.求一条简单路径,权值和等于K,且边的数量最小.N <= 200000, K <= 1000000 输入格式 第一行 两个整数 n, k 第二..n行 每行三个整 ...

  5. BZOJ_2599_[IOI2011]Race_点分治

    BZOJ_2599_[IOI2011]Race_点分治 Description 给一棵树,每条边有权.求一条简单路径,权值和等于K,且边的数量最小.N <= 200000, K <= 10 ...

  6. [bzoj2599][IOI2011]Race——点分治

    Brief Description 给定一棵带权树,你需要找到一个点对,他们之间的距离为k,且路径中间的边的个数最少. Algorithm Analyse 我们考虑点分治. 对于子树,我们递归处理,所 ...

  7. 【BZOJ2599】Race(点分治)

    [BZOJ2599]Race(点分治) 题面 BZOJ权限题,洛谷 题解 好久没写过点分治了... 在ppl的帮助下终于想起来了 orz ppl 首先回忆一下怎么求有没有正好是\(K\)的路径 维护一 ...

  8. codeforces 161D Distance in Tree 树上点分治

    链接:https://codeforces.com/contest/161/problem/D 题意:给一个树,求距离恰好为$k$的点对是多少 题解:对于一个树,距离为$k$的点对要么经过根节点,要么 ...

  9. POJ 1741 Tree 树上点分治

    题目链接:http://poj.org/problem?id=1741 题意: 给定一棵包含$n$个点的带边权树,求距离小于等于K的点对数量 题解: 显然,枚举所有点的子树可以获得答案,但是朴素发$O ...

随机推荐

  1. webpack的初步使用(01)

    webpack:1.安装:在项目文件下先npm init初始化,一路回车2.进入到建立的项目下:cd projectname3.安装webpack:npm install webpack --save ...

  2. [Swift通天遁地]七、数据与安全-(12)使用Instruments Leaks工具检测内存泄露

    ★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★➤微信公众号:山青咏芝(shanqingyongzhi)➤博客园地址:山青咏芝(https://www.cnblogs. ...

  3. C语言关键字之sizeof

    C语言关键字 sizeof 是一个操作符,返回对象或类型所占内存字节数,类型为size_t(定义在<stddef.h>),有2种用法: sizeof unary-expression si ...

  4. phpstorm如何配置xdebug?(hpStudy+PhpStorm+XDebug配置)

    xdebug是什么? 初次接触,反复试了几次终于把这个xdebug给搞清楚了,类似于前端的控制台这样的东西,可以根据断点展示我们想要看的数据. 如何配置xdebug? 配置前说明: 1.phpStud ...

  5. Alpha Edition [ Group 1 ]

    Deltafish Alpha Edition 一.博客归档(记录人:娄雨禛) 小组会议 DeltaFish 校园物资共享平台 第一次小组会议 DeltaFish 校园物资共享平台 第二次小组会议 D ...

  6. 基于openstack平台的几种Cloud DB解决方案

    方案一.openstack 官方 trove解决方案 此方案进行过镜像的打包,由于网络问题,还未能成功实现 方案二.salt 或者ansible+ docker 由于 docker部署数据库,在数据库 ...

  7. 搜索条件两个时间,通过php数组排序,保证select语句between时间 前小后大

    //搜索条件两个时间,通过数组排序,保证select语句between时间 前小后大 $sort_array=[$_POST['clockDate1'],$_POST['clockDate2']]; ...

  8. 3星|《IBM商业价值报告:区块链》:一些重要行业对区块链的态度和已经发生的区块链的应用

    区块链项目开发指南 (区块链技术丛书) 介绍IBM的专家们调研许多重要行业与组织后总结的各行业对区块链的态度和实际的应用.看起来有点意思,不过有两个缺点: 1:这些实际已经发生的应用基本没看到相关的新 ...

  9. 【sqli-labs】 less53 GET -Blind based -Order By Clause -String -Stacked injection(GET型基于盲注的字符型Order By从句堆叠注入)

    http://192.168.136.128/sqli-labs-master/Less-53/?sort=1';insert into users(id,username,password) val ...

  10. java_File对象

    package File; import java.io.File; import java.io.IOException; public class file { public static voi ...