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. Leetcode题库——1.两数之和

    @author: ZZQ @software: PyCharm @file: addTwoNumbers.py @time: 2018/9/18 10:35 要求:给定两个非空链表来表示两个非负整数. ...

  2. python learning Network Programming.py

    Socket # 用一个 Socke t表示"打开了一个网络连接" # 打开一个 Socket 需要知道目标计算机的IP地址和端口号,再指定协议类型即可. # TCP # 主动发起 ...

  3. week3c:个人博客作业

    程序测试: 一个基本的测试. 在Visual Studio 2013 中使用C++单元测试 操作如下: 这是我学到的过程. 有复杂程序的测试.以后有时间再弄.

  4. 23_IO_第23天(字节流、字符流)_讲义

    今日内容介绍 1.字节流 2.字符流 01输入和输出 * A:输入和输出 * a: 参照物 * 到底是输入还是输出,都是以Java程序为参照 * b: Output * 把内存中的数据存储到持久化设备 ...

  5. angularJS1笔记-(14)-自定义指令(scope)

    index.html: <!DOCTYPE html> <html lang="en"> <head> <meta charset=&qu ...

  6. <mvc:annotation-driven/>的作用

    <mvc:annotation-driven>的作用是: 会自动注册DefaultAnnotationHandlerMapping与AnnotationMethodHandlerAdapt ...

  7. Scrum Meeting Beta - 10

    Scrum Meeting Beta - 10 NewTeam 2017/12/11 地点:新主楼F座二楼 任务反馈 团队成员 完成任务 计划任务 安万贺 完成了作业详情的本地存储Issue #165 ...

  8. js对数组进行浅复制,深复制的方法

    js 数组常用方法,数组的拷贝(不影响原数组),数组相等 置顶2017年07月17日 17:39:26 阅读数:4640 改变原数组的方法: pop();删除尾部的第一个元素并且返回这个元素: var ...

  9. 在Google Chrome中快速解除网页屏蔽鼠标右键、复制等限制

    第一步,将书签栏设置为显示状态!   第二步,添加新书签——>在标签栏点击右键,选择“添加网页”. 第三步,设置新书签的内容. 1.起名.这个凭个人爱好吧 2.网址栏输入: javascript ...

  10. ng-include 上ng-controller 无法获取控件

    A.Html内容如下 <div> <div kendo-grid="testGrid" k-options="testOptions"> ...