LUOGU P4149 [IOI2011]Race
题目描述
给一棵树,每条边有权。求一条简单路径,权值和等于 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的更多相关文章
- 模板—点分治B(合并子树)(洛谷P4149 [IOI2011]Race)
洛谷P4149 [IOI2011]Race 点分治作用(目前只知道这个): 求一棵树上满足条件的节点二元组(u,v)个数,比较典型的是求dis(u,v)(dis表示距离)满足条件的(u,v)个数. 算 ...
- P4149 [IOI2011]Race
对于这道题,明显是点分治,权值等于k,可以用桶统计树上路径(但注意要清空); 对于每颗子树,先与之前的子树拼k,再更新桶,维护t["len"]最小边数; #include < ...
- 洛谷P4149 [IOI2011]Race(点分治)
题目描述 给一棵树,每条边有权.求一条简单路径,权值和等于 KK ,且边的数量最小. 输入输出格式 输入格式: 第一行:两个整数 n,kn,k . 第二至 nn 行:每行三个整数,表示一条无向边的 ...
- [洛谷P4149][IOI2011]Race
题目大意:给一棵树,每条边有边权.求一条简单路径,权值和等于$K$,且边的数量最小. 题解:点分治,考虑到这是最小值,不满足可减性,于是点分中的更新答案的地方计算重复的部分要做更改,就用一个数组记录前 ...
- [LUOGU] 4149 [IOI2011]Race
点分治裸题 #include<iostream> #include<cstring> #include<cstdio> using namespace std; i ...
- P4149 [IOI2011]Race 点分治
思路: 点分治 提交:5次 题解: 刚开始用排序+双指针写的,但是调了一晚上,总是有两个点过不了,第二天发现原因是排序时的\(cmp\)函数写错了:如果对于路径长度相同的,我们从小往大按边数排序,当双 ...
- 洛谷$P4149\ [IOI2011]\ Race$ 点分治
正解:点分治 解题报告: 传送门$QwQ$ 昂先不考虑关于那个长度的限制考虑怎么做? 就开个桶,记录所有边的取值,每次加入边的时候查下是否可行就成$QwQ$ 然后现在考虑加入这个长度的限制?就考虑把这 ...
- 洛谷 P4149 [IOI2011]Race-树分治(点分治,不容斥版)+读入挂-树上求一条路径,权值和等于 K,且边的数量最小
P4149 [IOI2011]Race 题目描述 给一棵树,每条边有权.求一条简单路径,权值和等于 KK,且边的数量最小. 输入格式 第一行包含两个整数 n, Kn,K. 接下来 n - 1n−1 行 ...
- BZOJ 2599: [IOI2011]Race( 点分治 )
数据范围是N:20w, K100w. 点分治, 我们只需考虑经过当前树根的方案. K最大只有100w, 直接开个数组CNT[x]表示与当前树根距离为x的最少边数, 然后就可以对根的子树依次dfs并更新 ...
随机推荐
- 操作RDS数据库
- vue/cli 3.0脚手架搭建
在vue 2.9.6中,搭建vue-cli脚手架的流程是这样的: 首先 全局安装vue-cli,在cmd中输入命令: npm install --global vue-cli 安装成功: 安装完成 ...
- loj6005 [网络流24题]最长递增子序列
题意:给你一个序列,求不严格上升lcs长度/最多有几个没有重复元素的lcs/如果x1和xn可以多次出现,求最多有几个lcs?n<=500. 标程: #include<cstdio> ...
- 监听事件动态改变dom状态
html代码: <table class="table table-striped"> <thead> <tr> <th>分类ID& ...
- leetccode-130-被围绕的区域
题目描述: 方法一:dfs class Solution: def solve(self, board: List[List[str]]) -> None: """ ...
- [violet6] 故乡的梦
题目 描述 不知每日疲于在城市的水泥森林里奔波的你会不会有时也曾向往过乡村的生活.你会不会幻想过,在夏日一个静谧的午后,你沉睡于乡间路边的树荫里,一片叶子落在了你的肩上, 而你正做着一个悠长的梦,一个 ...
- 如何在屏幕上查看命令的输出以及在Linux中写入文件
在Linux中输出命令可以做很多事情(http://www.nanke0834.com) 您可以将命令的输出分配给变量,将其发送到另一个命令/程序以通过管道进行处理或将其重定向到文件以进行进一步分析. ...
- 【Uva 12128】Perfect Service
[Link]: [Description] 给你n个机器组成的一棵树,然后,让你在某些机器上安装服务器. 要求,每个机器如果没有安装服务器,都要恰好和一个安装了服务器的机器连接. 问你,最少要安装多少 ...
- Error configuring application listener of class [org.springframework.web.util.Log4jConfigListener]
1.启动项目发现如下错误: 严重: Error configuring application listener of class [org.springframework.web.util.Log4 ...
- VUE环境下获取当前时间并格式化--按秒数更新
<el-col :span="8"><div class="grid-content title-time"> {{date}}< ...