题目描述

给一棵树,每条边有权。求一条简单路径,权值和等于 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. 一个tcp连接可以发多少http请求

    -----来自:松若章 -----zhuanlan.zhihu.com/p/61423830 曾经有这么一道经典面试题:从 URL 在浏览器被被输入到页面展现的过程中发生了什么?相信大多数准备过的同学 ...

  2. crontab中反引号和$()无效的解决

    问题描述 1.增加了一条crontab,删除本月中2天以前的日志 10 02  * * * /bin/find /data/logs/php/$(date  +%Y%m)/ -mtime +2 | x ...

  3. spring:ApplicationContext的三个实现类

    * ApplicationContest的三个常用实现类* ClassPathXmlApplicationContext:它可以加载类路径的配置文件,要求配置文件必须在类路径下,如果不在则加载不了* ...

  4. 【珍惜时间】vuepro

    老规矩放上大大的github开源地址:https://github.com/goodheart222/vuepro 我们再来看看项目的效果,初步根据效果做到心中有数 看到效果的话,我们会发现,肯定是有 ...

  5. vue-cli3使用yarn run build打包找不到路径

    vue-cli3使用yarn run build打包项目部署到服务器上面,运行空白 解决办法非常方便,直接创建vue.config.js 在vue.config.js中添加即可 再打包项目即成功

  6. bzoj4144 Petrol

    题意:给你一张n个点m条边的带权无向图.其中由s个点是加油站.询问从x加油站到y加油站,油箱容量<=b,能否走到? n,m,q,s<=20W,b<=2e9. 标程: #include ...

  7. leetcode-154-寻找旋转排序数组中的最小值

    题目描述: 方法一: class Solution: def findMin(self, nums: List[int]) -> int: left, right = 0, len(nums) ...

  8. [JZOJ3177] 【GDOI2013模拟5】安全监控

    题目 描述 (样例都懒得发出来了) 题目大意 给你一个有向图,从111号点出发,绕一圈回来.这一圈中必须经过222号点. 问经过的最少的点数(不重复). 思考历程 一看就觉得是一道神题. 然后仔细观察 ...

  9. vim的个性化配置(方便编程)

    在用户主目录下新建vimrc即可.例如: vim ~/.vimrc 然后复制进去即可.   配置如下: "关闭vim一致性原则 set nocompatible "显示行号 set ...

  10. 高效开发 Dubbo?用 Spring Boot 可得劲!

    不仅简化了 Dubbo 基于 xml 配置的方式,也提高了日常开发效率,甚至提升了工作幸福感. 为了节省亲爱的读者您的时间,请根据以下2点提示来阅读本文,以提高您的阅读收获效率哦. 如果您只有简单的 ...