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的更多相关文章

  1. HDU 4085 斯坦纳树

    题目大意: 给定无向图,让前k个点都能到达后k个点(保护地)中的一个,而且前k个点每个需要占据后k个中的一个,相互不冲突 找到实现这个条件达到的选择边的最小总权值 这里很容易看出,最后选到的边不保证整 ...

  2. hdu 3311 斯坦纳树

    思路:虚拟一个0号节点,将每个点建一条到0号节点的边,权值为挖井需要的价值.并要保证0号节点同另外n个寺庙一样被选择即可. 然后就是求斯坦纳树了. #include<map> #inclu ...

  3. 【hdu3311】Dig The Wells(斯坦纳树+dp)

    传送门 题意: 给出\(n\)个重要点,还有其余\(m\)个点,\(p\)条边. 现在要在这\(n+m\)个点中挖几口水井,每个地方的费用为\(w_i\).连接边也有费用. 问使得这\(n\)个地点都 ...

  4. 【bzoj4006】[JLOI2015]管道连接(斯坦纳树+dp)

    题目链接 题意: 给出\(n\)个点,\(m\)条边,同时给出\(p\)个重要的点以及对应特征. 现在要选出一些边,问使得这\(p\)个所有特征相同的点相连,问最小代价. 思路: 斯坦纳树的应用场景一 ...

  5. [WC2008]游览计划(斯坦纳树)

    [Luogu4294] 题解 : 斯坦纳树 \(dp[i][j]\) 表示以\(i\)号节点为根,当前状态为\(j\)(与\(i\)连通的点为\(1\)) 当根\(i\)不改变时状态转移方程是: \( ...

  6. 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]  ...

  7. 【BZOJ2595】游览计划(状压DP,斯坦纳树)

    题意:见题面(我发现自己真是越来越懒了) 有N*M的矩阵,每个格子有一个值a[i,j] 现要求将其中的K个点(称为关键点)用格子连接起来,取(i,j)的费用就是a[i,j] 求K点全部连通的最小花费以 ...

  8. 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 ...

  9. HDU 3311 Dig The Wells(斯坦纳树)

    [题目链接] http://acm.hdu.edu.cn/showproblem.php?pid=3311 [题意] 给定k座庙,n个其他点,m条边,点权代表挖井费用,边权代表连边费用,问使得k座庙里 ...

随机推荐

  1. 简单谈谈java中匿名内部类构造函数?

    先看看下面的代码能不能编译通过: public static void main(String[] args) {List l1 = new ArrayList();List l2 = new Arr ...

  2. 性能优化-屏幕常亮与CPU唤醒

    Android在不使用的时候,屏幕在一段时间以后会变暗,再过一段时间就会熄屏,此时CPU就会休眠,那么在这个时候,Timer.Handler.Thread.Service等都会暂停,有时候我们需要屏幕 ...

  3. 洛谷 题解 P2676 【超级书架】

    题解 P2676 [超级书架] 这题就只是一个从大到小的排序而已,用"sort"函数 再用"while"判断奶牛塔的高度是否比书架高度要高 送上代码: #inc ...

  4. idea调试框架时如何能够进入maven依赖包的源码

    最近用idea使用spring security做登录与权限控制,在调试过程中碰到问题,经常需要断点追踪源码,深入查看其中问题.但是maven中引用的包,进入后都是.class文件,非源码.导致查看时 ...

  5. class.forName 和 classLoader的区别

    Java中的Class.forName()和ClassLoader都可以用来对类进行加载.Class.forName()除了将类的.class文件加载到JVM中 还会对类进行解释,执行类中的stati ...

  6. 记录一次线上yarn RM频繁切换的故障

    周末一大早被报警惊醒,rm频繁切换 急急忙忙排查 看到两处错误日志 错误信息1 ervation <memory:0, vCores:0> 2019-12-21 11:51:57,781 ...

  7. SpringBoot 对IBM MQ进行数据监听接收以及数据发送

    一.需求介绍 后端使用Spring Boot2.0框架,要实现IBM MQ的实时数据JMS监听接收处理,并形成回执通过MQ队列发送. 二.引入依赖jar包 <dependency> < ...

  8. 对Python中print函数参数的认识

    输出函数是最常用的,对print()参数的准确认识尤为重要. sep='':sep参数表示函数中不同value的分隔符,默认为一个空格. end='':end参数表示函数结尾的处理,默认换行. 例如: ...

  9. spark异常篇-OutOfMemory:GC overhead limit exceeded

    执行如下代码时报错 # encoding:utf-8 from pyspark import SparkConf, SparkContext from pyspark.sql import Spark ...

  10. Java打包

    打成jar包 如果你使用的是maven来管理项目,执行以下命令既可以 cd 项目跟目录(和pom.xml同级)mvn clean package## 或者执行下面的命令## 排除测试代码后进行打包mv ...