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. 迁移学习(MixMatch)《MixMatch: A Holistic Approach to Semi-Supervised Learning》

    论文信息 论文标题:MixMatch: A Holistic Approach to Semi-Supervised Learning论文作者:David Berthelot, Nicholas Ca ...

  2. 第k个数【模板题】

    第k个数 给定一个长度为 \(n\) 的整数数列,以及一个整数 \(k\),请用快速选择算法求出数列从小到大排序后的第 \(k\) 个数. 输入格式 第一行包含两个整数 \(n\) 和 \(k\). ...

  3. Flutter异常监控 - 伍 | 关于异常监控框架设计的思考

    前言 最近阅读 Catcher.BugSnag.Rollbar 三个 Flutter 异常监控开源框架,文章链接如下: Flutter 异常监控 - 壹 | 从 Zone 说起 Flutter 异常监 ...

  4. BOM、DOM、两者查找标签的方式和操作标签的方式、操作class和css的方法、事件、jQuery类库

    今日内容 BOM操作 BOM(Browser Object Model)是指浏览器对象模型,它使 JavaScript 有能力与浏览器进行"对话". windows对象 windo ...

  5. Python导入Excel表格数据并以字典dict格式保存

      本文介绍基于Python语言,将一个Excel表格文件中的数据导入到Python中,并将其通过字典格式来存储的方法.   我们以如下所示的一个表格(.xlsx格式)作为简单的示例.其中,表格共有两 ...

  6. for循環,逗號運算符,其他(+= -= *= /=)

  7. 使用linux命令直接在网上下载文件,解压,改名

    举例: 我们想要在服务器某个路径下,下载一个node.js包 操作如下 假如文件地址为https://npm.taobao.org/mirrors/node/v16.9.1/node-v16.9.1- ...

  8. BBS登录与注册功能的总结

    BBS登录与注册功能的总结 一.表设计:表名 外键字段 表名 models.py from django.db import models from django.contrib.auth.model ...

  9. Arm-Linux子系统的互相Notify

    前言: Linux下面不同的子系统一个个的组成了整个系统的运行环节,为了让这些子系统能够互相通讯,有一种叫做:notify chain(通知链)的东西.本篇看下. 概括 所谓通知链,有通知,就有执行的 ...

  10. spring security登录认证流程解析

    转 https://blog.csdn.net/qq_37142346/article/details/80032336 1.说明 用户认证流程 认证结果如何在多个请求之间共享 获取认证用户信息.用户 ...