题目描述

给一棵树,每条边有权。求一条简单路径,权值和等于 KKK ,且边的数量最小。
输入输出格式
输入格式:

第一行:两个整数 n,kn,kn,k 。

第二至 nnn 行:每行三个整数,表示一条无向边的两端和权值 (注意点的编号从 000 开始)。

输出格式:

一个整数,表示最小边数量。

如果不存在这样的路径,输出 −1-1−1 。

输入输出样例
输入样例#1: 复制

4 3
0 1 1
1 2 2
1 3 4

输出样例#1: 复制

2

说明

n≤200000,K≤1000000n\le 200000,K\le 1000000n≤200000,K≤1000000 。

解题思路

点分治。用一个叫f的桶来维护,时间复杂度O(nlogn)

代码

#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<algorithm>
#include<queue> using namespace std;
const int MAXN = ;
const int inf = 0x3f3f3f3f; inline int rd(){
int x=,f=;char ch=getchar();
while(!isdigit(ch)) {if(ch=='-') f=-;ch=getchar();}
while(isdigit(ch)) {x=(x<<)+(x<<)+ch-'';ch=getchar();}
return x*f;
} int n,k,ans=0x3f3f3f3f;
int head[MAXN],cnt,sum,mx,rt,siz[MAXN];
int to[MAXN<<],nxt[MAXN<<],val[MAXN<<];
int f[],stk[MAXN],top;
bool vis[MAXN];
queue<int> Q,q; inline void add(int bg,int ed,int w){
to[++cnt]=ed,val[cnt]=w,nxt[cnt]=head[bg],head[bg]=cnt;
} inline void dfs(int x,int fa){
siz[x]=;
for(register int i=head[x];i;i=nxt[i]){
int u=to[i];if(vis[u] || u==fa) continue;
dfs(u,x);siz[x]+=siz[u];
}
} inline void getrt(int x,int fa){
int k=;
for(register int i=head[x];i;i=nxt[i]){
int u=to[i];if(vis[u] || u==fa) continue;
getrt(u,x);k=max(k,siz[u]);
}
k=max(k,sum-siz[x]);
if(k<mx) {mx=k;rt=x;}
} inline void getdis(int x,int fa,int dis,int len){
if(dis==k && len<ans) ans=len;
if(dis<k) if(f[k-dis]) ans=min(ans,f[k-dis]+len);
Q.push(dis);q.push(len);if(dis<k) stk[++top]=dis;
for(register int i=head[x];i;i=nxt[i]){
int u=to[i]; if(vis[u] || u==fa) continue;
getdis(u,x,dis+val[i],len+);
}
} inline void getans(int x){
vis[x]=;
for(register int i=head[x];i;i=nxt[i]){
int u=to[i];if(vis[u]) continue;
getdis(u,x,val[i],);
while(!Q.empty()){
int now=Q.front();Q.pop();
int len=q.front();q.pop();
if(now>k) continue;
if(!f[now]) f[now]=len;
else f[now]=min(f[now],len);
}
}
for(register int i=;i<=top;i++) f[stk[i]]=;top=;
for(register int i=head[x];i;i=nxt[i]){
int u=to[i];if(vis[u]) continue;
dfs(u,x);sum=mx=siz[u];rt=;getrt(u,x);
// cout<<u<<" "<<rt<<endl;
getans(rt);
}
} int main(){
n=rd();k=rd();
for(register int i=;i<n;i++){
int x=rd()+,y=rd()+,z=rd();
add(x,y,z);add(y,x,z);
}sum=n;mx=sum;dfs(,);getrt(,);
// cout<<1<<" "<<rt<<" ";
getans(rt);
if(ans==inf) ans=-;
cout<<ans<<endl;
return ;
}
 

LUOGU P4149 [IOI2011]Race的更多相关文章

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

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

  2. P4149 [IOI2011]Race

    对于这道题,明显是点分治,权值等于k,可以用桶统计树上路径(但注意要清空); 对于每颗子树,先与之前的子树拼k,再更新桶,维护t["len"]最小边数; #include < ...

  3. 洛谷P4149 [IOI2011]Race(点分治)

    题目描述 给一棵树,每条边有权.求一条简单路径,权值和等于 KK ,且边的数量最小. 输入输出格式 输入格式:   第一行:两个整数 n,kn,k . 第二至 nn 行:每行三个整数,表示一条无向边的 ...

  4. [洛谷P4149][IOI2011]Race

    题目大意:给一棵树,每条边有边权.求一条简单路径,权值和等于$K$,且边的数量最小. 题解:点分治,考虑到这是最小值,不满足可减性,于是点分中的更新答案的地方计算重复的部分要做更改,就用一个数组记录前 ...

  5. [LUOGU] 4149 [IOI2011]Race

    点分治裸题 #include<iostream> #include<cstring> #include<cstdio> using namespace std; i ...

  6. P4149 [IOI2011]Race 点分治

    思路: 点分治 提交:5次 题解: 刚开始用排序+双指针写的,但是调了一晚上,总是有两个点过不了,第二天发现原因是排序时的\(cmp\)函数写错了:如果对于路径长度相同的,我们从小往大按边数排序,当双 ...

  7. 洛谷$P4149\ [IOI2011]\ Race$ 点分治

    正解:点分治 解题报告: 传送门$QwQ$ 昂先不考虑关于那个长度的限制考虑怎么做? 就开个桶,记录所有边的取值,每次加入边的时候查下是否可行就成$QwQ$ 然后现在考虑加入这个长度的限制?就考虑把这 ...

  8. 洛谷 P4149 [IOI2011]Race-树分治(点分治,不容斥版)+读入挂-树上求一条路径,权值和等于 K,且边的数量最小

    P4149 [IOI2011]Race 题目描述 给一棵树,每条边有权.求一条简单路径,权值和等于 KK,且边的数量最小. 输入格式 第一行包含两个整数 n, Kn,K. 接下来 n - 1n−1 行 ...

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

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

随机推荐

  1. Linux 下 Nand Flash 调用关系

    Nand Flash 设备添加时数据结构包含关系 struct mtd_partition        partition_info[] --> struct s3c2410_nand_set ...

  2. 使用vue-cli3快速适配H5项目

    跟我老大学到了一招使用vue-cli3快速适配H5项目的方法. 我之前也有进行一个版本的适配,直接使用cnpm install -g vue-cli,然后安装各种插件进行适配,见我之前的博客. 后来老 ...

  3. mybatis结果封装到hashmap中没有null的数据

    1.在ssm框架中 在mybatis-config.xml配置文件中加下面这句代码即可解决 <setting name="callSettersOnNulls" value= ...

  4. React中的表单应用

    React中的表单应用 用户在表单填入的内容,属于用户跟组件的互动,所以不能用this.props读取. var Input = React.createClass({ //初始化组件数据 getIn ...

  5. redis深入学习(一)-----CAP、redis数据类型

    NoSQL数据库的四大分类 KV键值: memcache+redis 文档型数据库(bson格式比较多): MongoDB MongoDB 是一个基于分布式文件存储的数据库.由 C++ 语言编写.旨在 ...

  6. Mac OS X 下有关adb相关问题

    一.什么是adb? ADB的全称是Android Debug Bridge,用来调试Android程序的,白话点就是debug工具! 位置:一般下载Android的SDK时候在platform-too ...

  7. R语言:表格的线图转化

    R语言:表格的线图转化 最先选取的是北京各区普通住宅成交十年(2016年及2006年)涨幅对比.这张图比较plain,主要拿来练习: 1.数据表格的基本整理及计算 2. 数据的初步分析 3.线图的基本 ...

  8. Docker在线文档收集

    极客学院 kubernetes中文社区 易百教程

  9. Spring MVC(四)--控制器接受pojo参数

    以pojo的方式传递参数适用于参数较多的情况,或者是传递对象的这种情况,比如要创建一个用户,用户有十多个属性,此时就可以通过用户的pojo对象来传参数,需要注意的是前端各字段的名称和pojo对应的属性 ...

  10. MyBatis与JPA的区别

    参考博客: https://www.cnblogs.com/llywy/p/10103136.html https://www.jianshu.com/p/32ce87c163d6 MyBatis分为 ...