[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 ...
随机推荐
- Scapy之ARP询问
引言 校园网中,有同学遭受永恒之蓝攻击,但是被杀毒软件查下,并知道了攻击者的ip也是校园网.所以我想看一下,这个ip是PC,还是路由器. 在ip视角,路由器和pc没什么差别. 实现 首先是构造arp报 ...
- sqlserver结束和监视耗时的sql
在对象资源管理器中右击服务器地址选择“活动和监视器”. 点击最近耗费大量资源的查询
- 团队作业7——第二次项目冲刺(Beta版本12.04——12.07)
1.当天站立式会议照片 本次会议在5号公寓3楼召开,本次会议内容:①:熟悉每个人想做的模块.②:根据项目要求还没做的完成. 2.每个人的工作 经过会议讨论后确定了每个人的分工 组员 任务 陈福鹏 实现 ...
- 关闭Centos5.5的写磁盘I/O功能
一个Linux文件默认有3个时间. atime:对此文件的访问时间. ctime:此文件inode发生变化的时间. mtime:此文件的修改时间. 如果有多个小文件(比如Web服务器的页面上有多个小图 ...
- 网页访问过程(基于CDN)
1. 全局负载均衡(基于DNS) 如果有多台 WEB 服务器同时为一个域名提供服务时,即一条 URL 对应多个 IP 地址,那么该 URL 的权威域名服务器可能会根据该 URL 解析出多个 IP 地址 ...
- scrapy-继承默认的user-agent 中间件
class MyUserAgentMiddleware(UserAgentMiddleware): def __init__(self, user_agent): self.user_agent = ...
- [日常工作] SQLSERVER 数据库出问题..搜索到的有用的网页信息
Finding a table name from a page ID By: Paul Randal Posted on: September 25, 2014 1:42 am (Check o ...
- Java乐观锁、悲观锁
乐观锁 乐观锁(Optimistic Lock), 顾名思义,就是很乐观,每次去拿数据的时候都认为别人不会修改,所以不会上锁,但是在更新的时候会判断一下在此期间别人有没有去更新这个数据,可以使用版本号 ...
- delphi的ActionToolBar控件的使用 [问题点数:50分]
delphi那些默认图标在哪里就是那些Text,Label,Checkbox,显示在palette上面的那些图标. 上面的按钮是在ActionManager中添加的,让后拖到ActionToolBar ...
- java 重写你可以这么理解 因为 方法名和参数类型个数相同 所以这就是重写了 然后 因为是重写 所以 返回值必须相同
java 重写你可以这么理解 因为 方法名和参数类型个数相同 所以这就是重写了 然后 因为是重写 所以 返回值必须相同