2599: [IOI2011]Race

Time Limit: 70 Sec  Memory Limit: 128 MB
http://www.lydsy.com/JudgeOnline/problem.php?id=2599

Description

给一棵树,每条边有权.求一条简单路径,权值和等于K,且边的数量最小.N <= 200000, K <= 1000000

Input

第一行 两个整数 n, k
第二..n行 每行三个整数 表示一条无向边的两端和权值 (注意点的编号从0开始)

Output

一个整数 表示最小边数量 如果不存在这样的路径 输出-1

Sample Input

4 3
0 1 1
1 2 2
1 3 4

Sample Output

2
点分治
是求两点间边权和<=k的数量
这里是求两点间边权和=k的最少边数
我们仍然可以借用上题的方法,多记录一个节点到根节点的经过的边数即可
有2个地方需要修改:
① 上题需要减去同一子树中不合法的点对个数
    本题虽不需要考虑同一子树内的情况,但需要在计算时跳过同一子树内的2个点
    具体做法是 在递归记录点与根节点间权值和、边数时,顺带记录每个点属于 当前根节点的哪颗子树
     计算时,如果属于同一子树,跳过
    代码:
  if(fa==head||!fa) deep[deep[0].edge_sum].id=x;

  else deep[deep[0].edge_sum].id=deep[deep[0].edge_sum-1].id;

当前点为x,head表示当前根节点下的哪颗子树,fa表示x的父节点,id记录当前点属于哪颗子树

上题能不能采用同样的方法,避免计算子树内部的情况呢?

不能。因为排序仅按边权大小排,累计答案的方式是加r-l,即一堆满足条件的点判断一次,一起累加。

判断的两个点在同一子树内,其他的点可能不在同一子树内

②统计答案的时候,仍然可以同上题一样采用两边指针向中间逼近的方式

但要特殊处理指针指向位置周围边权相等的情况

 

#include<cstdio>
#include<algorithm>
#define N 200001
using namespace std;
int n,k,tot,sum,root,son[N],f[N],d[N],ans=N;
int front[N],to[N*],next[N*],w[N*];
bool v[N];
struct node
{
int dis,edge_sum,id;
}deep[N];
void getroot(int x,int fa)
{
son[x]=;f[x]=;
for(int i=front[x];i;i=next[i])
{
if(to[i]==fa||v[to[i]]) continue;
getroot(to[i],x);
son[x]+=son[to[i]];
f[x]=max(f[x],son[to[i]]);
}
f[x]=max(f[x],sum-son[x]);
if(f[x]<f[root]) root=x;
}
void getdeep(int head,int x,int fa,int edge_sum)
{
deep[++deep[].edge_sum]=(node){d[x],edge_sum};
if(fa==head||!fa) deep[deep[].edge_sum].id=x;
else deep[deep[].edge_sum].id=deep[deep[].edge_sum-].id;
for(int i=front[x];i;i=next[i])
{
if(v[to[i]]||to[i]==fa) continue;
d[to[i]]=d[x]+w[i];
getdeep(head,to[i],x,edge_sum+);
}
}
bool cmp(node l,node r)
{
/*if(l.dis!=r.dis) return l.dis<r.dis;
return l.edge_sum>r.edge_sum;*/
return l.dis<r.dis;
}
void cal(int x,int now)
{
d[x]=now;deep[].edge_sum=;
getdeep(x,x,,);
int l=,r=deep[].edge_sum,t=;
sort(deep+,deep+r+,cmp);
while(l<r)
{
/*if(deep[l].dis+deep[r].dis==k&&deep[l].id!=deep[r].id)
{
ans=min(ans,deep[l].edge_sum+deep[r].edge_sum);
//printf("%d %d\n",deep[l].dis,deep[r].dis);
l++; }*/ // 错误的
if(deep[l].dis+deep[r].dis==k)
{
int p1=l,p2=r;
while(deep[p1].dis+deep[r].dis==k) p1++;p1--;
while(deep[p2].dis+deep[l].dis==k) p2--;p2++;
for(int i=l;i<=p1;i++)
for(int j=p2;j<=r;j++)
if(deep[i].id!=deep[j].id)
ans=min(ans,deep[i].edge_sum+deep[j].edge_sum);
l=p1+;r=p2-;
}
else if(deep[l].dis+deep[r].dis<k) l++;
else r--;
}
}
void work(int x)
{
cal(x,);
v[x]=true;
for(int i=front[x];i;i=next[i])
{
if(v[to[i]]) continue;
sum=son[to[i]];
root=;
getroot(to[i],root);
work(root);
}
}
void add(int u,int v,int val)
{
to[++tot]=v;next[tot]=front[u];front[u]=tot;w[tot]=val;
to[++tot]=u;next[tot]=front[v];front[v]=tot;w[tot]=val;
}
int read()
{
int x=,f=;char c=getchar();
while(c<''||c>'') {if(c=='-') f=-;c=getchar();}
while(c>=''&&c<='') {x=x*+c-'';c=getchar();}
return x*f;
}
int main()
{
n=read();k=read();
int x,y,z;
for(int i=;i<n;i++)
{
x=read();y=read();z=read();
x++;y++;
add(x,y,z);
}
f[]=N;
sum=n;
getroot(,);
work(root);
printf("%d",ans==N ? -:ans);
}

[IOI2011]Race的更多相关文章

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

    数据范围是N:20w, K100w. 点分治, 我们只需考虑经过当前树根的方案. K最大只有100w, 直接开个数组CNT[x]表示与当前树根距离为x的最少边数, 然后就可以对根的子树依次dfs并更新 ...

  2. 【BZOJ2599】[IOI2011]Race 树的点分治

    [BZOJ2599][IOI2011]Race Description 给一棵树,每条边有权.求一条简单路径,权值和等于K,且边的数量最小.N <= 200000, K <= 100000 ...

  3. [IOI2011]Race 点分治

    [IOI2011]Race LG传送门 点分治板子题. 直接点分治统计,统计的时候开个桶维护下就好了. 注(tiao)意(le)细(hen)节(jiu). #include<cstdio> ...

  4. bzoj 2599 [IOI2011]Race 点分

    [IOI2011]Race Time Limit: 70 Sec  Memory Limit: 128 MBSubmit: 4768  Solved: 1393[Submit][Status][Dis ...

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

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

  6. 2599: [IOI2011]Race

    2599: [IOI2011]Race 链接 分析 被memset卡... 点分治,对于重心,遍历子树,记录一个数组T[i],表示以重心为起点的长度为i的路径中最少的边数是多少.然后先遍历子树,更新答 ...

  7. 模板—点分治B(合并子树)(洛谷P4149 [IOI2011]Race)

    洛谷P4149 [IOI2011]Race 点分治作用(目前只知道这个): 求一棵树上满足条件的节点二元组(u,v)个数,比较典型的是求dis(u,v)(dis表示距离)满足条件的(u,v)个数. 算 ...

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

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

  9. BZOJ2599 [IOI2011]Race

    传送门 点分治,黄学长的选根方法会T掉,换了这个人的选根方法就可以了. 当然,你也可以选择黄学长的奇淫优化 //BZOJ 2599 //by Cydiater //2016.9.23 #include ...

随机推荐

  1. ADT图及图的实现及图的应用

    图: 图中涉及的定义: 有向图: 顶点之间的相关连接具有方向性: 无向图: 顶点之间相关连接没有方向性: 完全图: 若G是无向图,则顶点数n和边数e满足:0<=e<=n(n-1)/2,当e ...

  2. iOS- 如何使用Alcatraz来高效的管理Xcode-Plugin(Xcode插件)

    1.前言 相信各位iOS攻城师用的Xocde的快捷插件也不少,今天向大家分享一款能高效快捷的管理Xcode-Plugin的软件<Alcatraz>,自己亲自体验后,爱不释手.   (这里用 ...

  3. windows下的C++ socket服务器(3)

    int make_server_socket(int port) { WSADATA inet_WsaData; WSAStartup(MAKEWORD(), &inet_WsaData); ...

  4. Linux网卡配置文件路径是什么?要使服务器上外网,必须满足的条件有哪些?需要配置什么?

    Linux网卡配置文件路径是什么?要使服务器上外网,必须满足的条件有哪些?需要配置什么?    答:    网卡配置文件路径:/etc/sysconfig/network-scripts/ifcfg- ...

  5. Robot Framework 教程 (5) - 连接Oracel数据库

    Robot Framework 提供了多种Library.其中Database Library可用来连接操作数据库. 1.安装Database Library 打开Robot Framework官网, ...

  6. Cannot open the disk 'D:\win7-ie8\Windows 7 x64.vmdk' or one of the snapshot

    使用机子过程中断电,开机后使用虚拟机提示[Cannot open the disk 'D:\win7-ie8\Windows 7 x64.vmdk' or one of the snapshot],找 ...

  7. VBA笔记(一)——基础配置

    开启VBA编程环境——VBE 方法一:按<Alt+F11>组合建 方法二:查看代码 宏设置 当然启用宏的设置方式不同,宏的启动方式也不一样. 首先打开“office 按钮”,选择“exce ...

  8. JavaWeb基础【1】—— Tomcat

    此笔记是学习黑马程序员JavaWeb系列视频的课堂笔记. 感谢黑马程序员. 一.Tomcat概述 Tomcat服务器由Apache提供,开源免费.由于Sun和其他公司参与到了Tomcat的开发中,所以 ...

  9. java传统的文件拷贝 相当于两个大缸需要通过一个勺子(字节数组)一点一点运过去

  10. MySQL复制 -- 复制出错怎么办?

    假如我们生产环境复制出错?该怎么办呢? 下面提供几种办法: 1. 手工处理,补回不一致数据(可以利用主库来补数据.也可以利用binlog来补数据) 2.用开源工具来解决一致性问题 3.自己造轮子,解决 ...