HDU 4085 斯坦纳树+DP
https://cn.vjudge.net/problem/HDU-4085
给你n,m,k ,分别表示有n个点,m条边,每条边有一个权值,表示修复这条边需要的代价
从前k个点中任取一个使其和后k个点中的某一个点,通过边连接,并且必须是一一对应,问最小的代价是多少。
先用斯坦纳树模板求出f[i][1<<k] 然后用dp[i]表示所有点为根的情况下连通状态为i的最小花费
这样我们就可以从1dp到1<<k得到答案 注意dp之前要先判总状态是否合法 再判子集是否合法 最后再进行dp更新
#include<bits/stdc++.h>
#define N 6003
#define inf 1000000000
using namespace std;
int n, m, k, tot;
int point[N], next1[N], v[N], len[N];
int f[][( << )], mi[], can[N], dp[( << )];
queue<int> p;
void add(int x, int y, int z) {
tot++;
next1[tot] = point[x];
point[x] = tot;
v[tot] = y;
len[tot] = z;
tot++;
next1[tot] = point[y];
point[y] = tot;
v[tot] = x;
len[tot] = z;
}
void spfa(int sta) {
while (!p.empty()) {
int now = p.front();
p.pop();
for (int i = point[now]; i; i = next1[i]) {
if (f[v[i]][sta] > f[now][sta] + len[i]) {
f[v[i]][sta] = f[now][sta] + len[i];
if (!can[v[i]]) {
can[v[i]] = ;
p.push(v[i]);
}
}
}
can[now] = ;
}
}
bool check(int sta) { //判断当前的状态是否满足一一对应关系
int ans = ;
for (int i = ; i < k; i++) {
if (sta & ( << i))
ans++;
if (sta & ( << (i + k)))
ans--;
}
return (ans == );
}
int main() {
int t;
scanf("%d", &t);
mi[] = ;
for (int i = ; i <= ; i++)
mi[i] = mi[i - ] * ;
for (int T = ; T <= t; T++) {
scanf("%d%d%d", &n, &m, &k);
tot = ;
memset(point, , sizeof(point));
memset(next1, , sizeof(next1));
for (int i = ; i <= m; i++) {
int x, y, z;
scanf("%d%d%d", &x, &y, &z);
add(x, y, z);
}
for (int i = ; i <= n; i++)
for (int j = ; j < mi[]; j++)
f[i][j] = inf;
for (int i = ; i <= k; i++)
f[i][mi[i - ]] = ;
int t = k;
for (int i = n - k + ; i <= n; i++)
f[i][mi[t]] = , t++;
for (int sta = ; sta < mi[t]; sta++) {
for (int i = ; i <= n; i++) {
for (int s = sta & (sta - ); s; s = sta & (s - )) {
int t = f[i][sta - s] + f[i][s];
f[i][sta] = min(f[i][sta], t);
}
if (f[i][sta] != inf)
p.push(i), can[i] = ;
}
spfa(sta);
}
for (int sta = ; sta < mi[t]; sta++) {
dp[sta] = inf;
for (int i = ; i <= n; i++)
dp[sta] = min(dp[sta], f[i][sta]);
}
for (int sta = ; sta < mi[t]; sta++)
if (check(sta))
for (int s = sta & (sta - ); s; s = sta & (s - ))
if (check(s))
dp[sta] = min(dp[sta], dp[s] + dp[sta - s]);
if (dp[mi[t] - ] == inf)
printf("No solution\n");
else
printf("%d\n", dp[mi[t] - ]);
}
}
HDU 4085 斯坦纳树+DP的更多相关文章
- HDU 4085 斯坦纳树
题目大意: 给定无向图,让前k个点都能到达后k个点(保护地)中的一个,而且前k个点每个需要占据后k个中的一个,相互不冲突 找到实现这个条件达到的选择边的最小总权值 这里很容易看出,最后选到的边不保证整 ...
- hdu 3311 斯坦纳树
思路:虚拟一个0号节点,将每个点建一条到0号节点的边,权值为挖井需要的价值.并要保证0号节点同另外n个寺庙一样被选择即可. 然后就是求斯坦纳树了. #include<map> #inclu ...
- 【hdu3311】Dig The Wells(斯坦纳树+dp)
传送门 题意: 给出\(n\)个重要点,还有其余\(m\)个点,\(p\)条边. 现在要在这\(n+m\)个点中挖几口水井,每个地方的费用为\(w_i\).连接边也有费用. 问使得这\(n\)个地点都 ...
- 【bzoj4006】[JLOI2015]管道连接(斯坦纳树+dp)
题目链接 题意: 给出\(n\)个点,\(m\)条边,同时给出\(p\)个重要的点以及对应特征. 现在要选出一些边,问使得这\(p\)个所有特征相同的点相连,问最小代价. 思路: 斯坦纳树的应用场景一 ...
- [WC2008]游览计划(斯坦纳树)
[Luogu4294] 题解 : 斯坦纳树 \(dp[i][j]\) 表示以\(i\)号节点为根,当前状态为\(j\)(与\(i\)连通的点为\(1\)) 当根\(i\)不改变时状态转移方程是: \( ...
- HDU 4085 Peach Blossom Spring 斯坦纳树 状态压缩DP+SPFA
状态压缩dp+spfa解斯坦纳树 枚举子树的形态 dp[i][j] = min(dp[i][j], dp[i][k]+dp[i][l]) 当中k和l是对j的一个划分 依照边进行松弛 dp[i][j] ...
- 【BZOJ2595】游览计划(状压DP,斯坦纳树)
题意:见题面(我发现自己真是越来越懒了) 有N*M的矩阵,每个格子有一个值a[i,j] 现要求将其中的K个点(称为关键点)用格子连接起来,取(i,j)的费用就是a[i,j] 求K点全部连通的最小花费以 ...
- hdu4085 Peach Blossom Spring 斯坦纳树,状态dp
(1)集合中元素表示(1<<i), i从0开始 (2)注意dp[i][ss] = min(dp[i][ss], dp[i][rr | s[i]] + dp[i][(ss ^ rr) | s ...
- HDU 3311 Dig The Wells(斯坦纳树)
[题目链接] http://acm.hdu.edu.cn/showproblem.php?pid=3311 [题意] 给定k座庙,n个其他点,m条边,点权代表挖井费用,边权代表连边费用,问使得k座庙里 ...
随机推荐
- [Kevin英语情报局]那些年我们说过的中式英语
一. blonde hair 金色头发 grey hair 白头发 baijiu 白酒 white wine 白葡萄酒 I don't think it's right 我认为不正确 I'm chin ...
- C++.控制台_界面颜色
1.c++中system(_color )怎样用?-CSDN论坛.html(https://bbs.csdn.net/topics/390758320) 2. 2.1. 颜色属性由两个十六进制数字指定 ...
- C#实现多线程的方式:Task——任务
简介 .NET 4包含新名称空间System.Threading.Tasks,它 包含的类抽象出了线程功能. 在后台使用ThreadPool. 任务表示应完成的某个单元的工作. 这个单元的工作可以在单 ...
- JAVA计算字符串UTF-8字节数
String str = "C++/C#/JAVA软件开发"; try { System.out.println(str.getBytes("UTF-8").l ...
- Linux下配置APUE的编译 报错之后如何处理
APUE即Unix环境高级编程,本书中几乎所有的程序都包含一个apue.h的头文件,那如何配置这个apue.h呢? 官方地址:http://www.apuebook.com/apue3e.html 1 ...
- 洛谷 题解 P2802 【回家】
思路:DFS+剪枝 本题可以用一个字符二维数组来存整个地图,然后在往四个方向进行搜索.注意:当走到家门前要先判断血量!(本人就被坑了) 代码: #include<bits/stdc++.h> ...
- [转帖]curl网站开发指南
curl网站开发指南 http://www.ruanyifeng.com/blog/2011/09/curl.html linux 里面有非常多很好的工具 比如这个 curl 之前 以为 wget 就 ...
- SQL语句中的HAVING关键字
sql中的having语句是在使用group by的时候使用的. 通常where语句是在group by之前做数据筛选的,而having语句是对group by之后的结果进行筛选的. 例如: 从商品销 ...
- JavaScript之二分法
二分法: 二分查找,又称为折半查找. 注意:二分法查找的数组必须是有序的. /* 获取元素88在数组中第一次出现的索引位置 如果数组元素中存在88,则直接返回88在数组中的索引位置即可. 如果 ...
- 剑指offer24:二叉树中和为输入整数值的所有路径。(注意: 在返回值的list中,数组长度大的数组靠前)
1 题目描述 输入一颗二叉树的根节点和一个整数,打印出二叉树中结点值的和为输入整数的所有路径.路径定义为从树的根结点开始往下一直到叶结点所经过的结点形成一条路径.(注意: 在返回值的list中,数组长 ...