https://www.bilibili.com/video/BV1Z4411x7Kw?from=search&seid=13855865082722302053

状压介绍:

状态表示:

转移方程:i是当前节点,j是下一步要走的节点

子集枚举:

核心代码:1。由当前枚举未知

首先枚举状态,枚举S中包含的节点:枚举i能去的节点

2.由已知枚举当前

枚举状态S:S ^ (1 << i) 表示去掉i节点

5435. 并行课程 II

yxc代码

 const int INF = 1000;
vector<int> f; int minNumberOfSemesters(int n, vector<vector<int>>& edges, int k) {
f = vector<int>(1 << n, INF);//dp数组
for (auto& e: edges) e[0] --, e[1] -- ;
f[0] = 0;// 还没选任何课
for (int i = 0; i < 1 << n; i ++ ) {//遍历所有状态
vector<bool> st(n);// 求当前可以选的课有哪些
for (auto& e: edges) {
int x = e[0], y = e[1];
if (!(i >> x & 1)) //如果x没有修过
st[y] = true;//y有些先修课没被修过
}
int state = 0;//state的二进制位表示可以修的课程
for (int j = 0; j < n; j ++ )
if (!st[j] && !(i >> j & 1))// j所有先修课修过,并且j还没有被修
state += 1 << j;//则修课程j dfs(n, k, i, state, 0,0);//i的状态转移
} return f[(1 << n) - 1];//返回全部修过时的最短
}
//state中选出k个元素来更新
void dfs(int n, int k, int i, int state, int now, int start) {
//参数 i 是当前状态,state是可以选的状态, now 是当前已选的状态, start是开始选的元素,避免重复计算
if (!k || !state) {//选了k个课,或者没课可选
f[i | now] = min(f[i | now], f[i] + 1);
return;
}
//当前可以选那些课
for (int j = start; j < n; j ++ ) {
if (state >> j & 1) {//如果state中包含这些课
dfs(n, k - 1, i, state - (1 << j), now + (1 << j), j + 1);
//state去掉当前选的课, now加上当前选的课
}
}
}

HUD 5148

http://acm.hdu.edu.cn/showproblem.php?pid=5418

递推:

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int maxn = 17;
const int INF = 0x3f3f3f3f;
int n, m, g[maxn][maxn], dp[1<<maxn][maxn]; void floyd()
{
for(int k = 0; k < n; k++)
for(int i = 0; i < n; i++)
for(int j = 0; j < n; j++)
g[i][j] = min(g[i][j], g[i][k]+g[k][j]);
} int main(void)
{
int t;
cin >> t;
while(t--)
{
scanf("%d%d", &n, &m);
//对图进行初始化
for(int i = 0; i < n; i++)
for(int j = 0; j < n; j++)
g[i][j] = i==j ? 0 : INF;
//插入边
while(m--)
{
int u, v, w;
scanf("%d%d%d", &u, &v, &w);
u--, v--;
g[u][v] = min(g[u][v], w);
g[v][u] = min(g[v][u], w);
}
//floyd算法
floyd();
memset(dp, INF, sizeof(dp));
dp[0][0] = 0;
for(int s = 0; s < 1<<n; s++)//遍历所有状态
for(int v = 0; v < n; v++)//遍历所有节点
if(dp[s][v] != INF)
for(int u = 0; u < n; u++)
dp[s|(1<<u)][u] = min(dp[s|(1<<u)][u], dp[s][v]+g[v][u]);
printf("%d\n", dp[(1<<n)-1][0]);
}
return 0;
}

递归:

#include<iostream>
#include<cstdio>
#include<cstring>
#include<vector>
#include<algorithm>
using namespace std;
const int maxn = 17;
const int INF = 0x3f3f3f3f;
int dp[1<<maxn][maxn], g[maxn][maxn], n, m; void floyd()
{
for(int k = 0; k < n; k++)
for(int i = 0; i < n; i++)
for(int j = 0; j < n; j++)
g[i][j] = min(g[i][j], g[i][k]+g[k][j]);
} int dfs(int S, int v)
{
if(dp[S][v] >= 0) return dp[S][v]; // 状态已经遍历过
if(S == (1<<n)-1 && v == 0) return dp[S][v] = 0;//回到终点
int res = INF;
for(int u = 0; u < n; u++)
{
if(!(S>>u & 1))//当前还没遍历过u节点,则遍历u节点
res = min(res, dfs(S | 1<<u, u)+g[v][u]);
}
return dp[S][v] = res;
} int main(void)
{
int t;
cin >> t;
while(t--)
{
scanf("%d%d", &n, &m);
//将图初始化
for(int i = 0; i < n; i++)
for(int j = 0; j < n; j++)
g[i][j] = i==j ? 0 : INF;
//加入边
while(m--)
{
int u, v, w;
scanf("%d%d%d", &u, &v, &w);
u--, v--;
g[u][v] = min(g[u][v], w);
g[v][u] = min(g[v][u], w);
}
//因为有些节点不是直接相连,进行floyd算法
floyd();
//dp 数组初始化
memset(dp, -1, sizeof(dp));
//dfs遍历
printf("%d\n", dfs(0, 0));
}
return 0;
}

dp-状压dp的更多相关文章

  1. 【BZOJ】1076 [SCOI2008]奖励关 期望DP+状压DP

    [题意]n种宝物,k关游戏,每关游戏给出一种宝物,可捡可不捡.每种宝物有一个价值(有负数).每个宝物有前提宝物列表,必须在前面的关卡取得列表宝物才能捡起这个宝物,求期望收益.k<=100,n&l ...

  2. CCF 201312-4 有趣的数 (数位DP, 状压DP, 组合数学+暴力枚举, 推公式, 矩阵快速幂)

    问题描述 我们把一个数称为有趣的,当且仅当: 1. 它的数字只包含0, 1, 2, 3,且这四个数字都出现过至少一次. 2. 所有的0都出现在所有的1之前,而所有的2都出现在所有的3之前. 3. 最高 ...

  3. hdu 4352 "XHXJ's LIS"(数位DP+状压DP+LIS)

    传送门 参考博文: [1]:http://www.voidcn.com/article/p-ehojgauy-ot.html 题解: 将数字num字符串化: 求[L,R]区间最长上升子序列长度为 K ...

  4. [转]状态压缩dp(状压dp)

    状态压缩动态规划(简称状压dp)是另一类非常典型的动态规划,通常使用在NP问题的小规模求解中,虽然是指数级别的复杂度,但速度比搜索快,其思想非常值得借鉴. 为了更好的理解状压dp,首先介绍位运算相关的 ...

  5. 状态压缩dp 状压dp 详解

    说到状压dp,一般和二进制少不了关系(还常和博弈论结合起来考,这个坑我挖了还没填qwq),二进制是个好东西啊,所以二进制的各种运算是前置知识,不了解的话走下面链接进百度百科 https://baike ...

  6. 洛谷 P3343 - [ZJOI2015]地震后的幻想乡(朴素状压 DP/状压 DP+微积分)

    题面传送门 鸽子 tzc 竟然来补题解了,奇迹奇迹( 神仙题 %%%%%%%%%%%% 解法 1: 首先一件很明显的事情是这个最小值可以通过类似 Kruskal 求最小生成树的方法求得.我们将所有边按 ...

  7. 51nod 1673 树有几多愁(链表维护树形DP+状压DP)

    题意 lyk有一棵树,它想给这棵树重标号. 重标号后,这棵树的所有叶子节点的值为它到根的路径上的编号最小的点的编号. 这棵树的烦恼值为所有叶子节点的值的乘积. lyk想让这棵树的烦恼值最大,你只需输出 ...

  8. BZOJ3836 [Poi2014]Tourism 【树形dp +状压dp】

    题目链接 BZOJ3836 题解 显然这是个\(NP\)完全问题,此题的解决全仗任意两点间不存在节点数超过10的简单路径的性质 这意味着什么呢? \(dfs\)树深度不超过\(10\) \(10\)很 ...

  9. SCUT - 254 - 欧洲爆破 - 概率dp - 状压dp

    https://scut.online/p/254 思路很清晰,写起来很恶心. #include<bits/stdc++.h> using namespace std; #define l ...

  10. bzoj1076: [SCOI2008]奖励关(期望dp+状压dp)

    1076: [SCOI2008]奖励关 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 2989  Solved: 1557[Submit][Statu ...

随机推荐

  1. 【随笔】Ubuntu18.04下virtualbox卡死的解决办法

    //得到该进程ID X pgrep Xorg //杀掉进程 kill X 然后重新登陆帐号即可

  2. ArcGIS工具 - 计算折点数量

    在GIS中,点构成线,线构成面,面构成体,维度增加,模型也加复杂.有时,我们需要统计线面等要素到底由多少个点构成,系统工具没有此功能,为源地理提供了三种解决方案. 方法一 折点转点 使用要素折点转点工 ...

  3. flutter 2.x运行flutter run 报错Cannot run with sound null safety, because the following dependenciesdon'

    flutter 2.x运行flutter run 报错Cannot run with sound null safety, because the following dependenciesdon' ...

  4. 玩转web3第二篇——Web3UI Kit

    介绍 开发web2应用的时候,可以很方便找到很多优秀的UI库,比如antd,material ui,element ui等等,但web3应用对应的UI库却不多. 今天给大家介绍一款优秀的WEB3的UI ...

  5. 【随笔记】Android 命令行联网、更新DNS、同步网络时间

    一.命令行联网 # 启用网卡 busybox ifconfig wlan0 up # 启用服务 wpa_supplicant -iwlan0 -Dnl80211 -c/system/etc/wifi/ ...

  6. STL序列式容器使用注意、概念总结

    引入 最近看了<STL源码剖析>的第 4 章和第 5 章,介绍了 C++ STL 中的序列式容器和关联式容器,本文将总结序列式容器的基础概念,不会详细它们的实现原理(想知道自个儿看书吧,我 ...

  7. 日志添加request-id

    package com.xf.config; import java.util.Date; import javax.servlet.http.HttpServletRequest; import j ...

  8. 线程基础知识11-CAS+自旋锁

    1.CAS是什么(CompareAndSet) CAS(Compare and swap)比较和替换是设计并发算法时用到的一种技术.简单来说,比较和替换是使用一个期望值和一个变量的当前值进行比较,如果 ...

  9. keypoint数据结构

    公有属性 (float)angle:角度,表示关键点的方向,-1为初值. (int)class_id:当要对图片进行分类时,我们可以用class_id对每个特征点进行区分,未设定时为-1,需要靠自己设 ...

  10. 学习Java Day26

    今天看书学习JAR文件,但是看了半天并没有看懂,于是借助了B站学习了JAR文件的创建及其中的一些jar程序选项该怎么使用