[IOI2011]Race
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
0 1 1
1 2 2
1 3 4
Sample Output
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的更多相关文章
- BZOJ 2599: [IOI2011]Race( 点分治 )
数据范围是N:20w, K100w. 点分治, 我们只需考虑经过当前树根的方案. K最大只有100w, 直接开个数组CNT[x]表示与当前树根距离为x的最少边数, 然后就可以对根的子树依次dfs并更新 ...
- 【BZOJ2599】[IOI2011]Race 树的点分治
[BZOJ2599][IOI2011]Race Description 给一棵树,每条边有权.求一条简单路径,权值和等于K,且边的数量最小.N <= 200000, K <= 100000 ...
- [IOI2011]Race 点分治
[IOI2011]Race LG传送门 点分治板子题. 直接点分治统计,统计的时候开个桶维护下就好了. 注(tiao)意(le)细(hen)节(jiu). #include<cstdio> ...
- bzoj 2599 [IOI2011]Race 点分
[IOI2011]Race Time Limit: 70 Sec Memory Limit: 128 MBSubmit: 4768 Solved: 1393[Submit][Status][Dis ...
- [bzoj2599][IOI2011]Race——点分治
Brief Description 给定一棵带权树,你需要找到一个点对,他们之间的距离为k,且路径中间的边的个数最少. Algorithm Analyse 我们考虑点分治. 对于子树,我们递归处理,所 ...
- 2599: [IOI2011]Race
2599: [IOI2011]Race 链接 分析 被memset卡... 点分治,对于重心,遍历子树,记录一个数组T[i],表示以重心为起点的长度为i的路径中最少的边数是多少.然后先遍历子树,更新答 ...
- 模板—点分治B(合并子树)(洛谷P4149 [IOI2011]Race)
洛谷P4149 [IOI2011]Race 点分治作用(目前只知道这个): 求一棵树上满足条件的节点二元组(u,v)个数,比较典型的是求dis(u,v)(dis表示距离)满足条件的(u,v)个数. 算 ...
- bzoj2599: [IOI2011]Race(点分治)
写了四五道点分治的题目了,算是比较理解点分治是什么东西了吧= = 点分治主要用来解决点对之间的问题的,比如距离为不大于K的点有多少对. 这道题要求距离等于K的点对中连接两点的最小边数. 那么其实道理是 ...
- BZOJ2599 [IOI2011]Race
传送门 点分治,黄学长的选根方法会T掉,换了这个人的选根方法就可以了. 当然,你也可以选择黄学长的奇淫优化 //BZOJ 2599 //by Cydiater //2016.9.23 #include ...
随机推荐
- 【贪心算法】POJ-1328 区间问题
一.题目 Description Assume the coasting is an infinite straight line. Land is in one side of coasting, ...
- python learning OOP1.py
class Student(object): # 构造函数 # 第一个参数永远是 self 表示一个实例本身,但是传参的时候不需要传 # 在Python中,实例的变量名如果以__开头,就变成了一个私有 ...
- windows和RedHat双系统安装说明
该博客记录了安装windows和RedHat双系统的方法.这里的windows系统是win8.1,RedHat是RHEL-server-7.0-x86_64-LinuxProbe.Com.iso,该i ...
- 结对作业(web)
作业源代码地址:https://git.coding.net/mal123/arithmetic.git 网页版测试地址:http://47.93.197.5:8080/mal_war_explode ...
- bootstrap使用总结
bootstrap是一个webcss框架,集合了html/css/jquery为一家,创建响应式的页面.所谓的响应式就是适配不同的上网设备. 使用bootstrap的步骤: 1.下载bootstrap ...
- LeetCode题解:(221) Maximal Square
题目说明 Given a 2D binary matrix filled with 0's and 1's, find the largest square containing only 1's a ...
- 利用ceye中的dns来获取数据
安恒杯的一道命令执行题目 查看,存在robots.txt文件 查看index.txt文件,存在where_is_flag.php文件 使用cat没有任何回显 可以使用ceye平台利用dns记录内容,网 ...
- three.js:Failed to execute 'texImage2D' on 'WebGLRenderingContext解决方案
three.js加载图片时,出现Failed to execute 'texImage2D' on 'WebGLRenderingContext .Tainted canvases may not b ...
- Opera官网打不开 下载Opera最新版本的实际地址
目前Opera官网可以打开,但是点下载时就会出错,国内无法访问Opera的下载地址,无法通过官网直接下载Opera浏览器.下面提供下载的方式. 一.通过官方的ftp站点下载 FTP地址为 http:/ ...
- BZOJ5092 分割序列(贪心)
设si为该序列的异或前缀和,则显然相当于求Σmax{sj+sj^si} (i=1~n,j=0~i).从高位到低位考虑,如果该位si为1,无论sj怎么填都是一样的:如果该位si为0,则sj该位应尽量为1 ...