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. appium如何连接多台设备

    我们在做app自动化的时候,若要考虑兼容性问题,需要跑几台设备,要是一台一台的跑比较耗 时,因此需要考虑使用多线程来同时操作多台设备. 1.我们拿两台设备来模拟操作下,使用:adb devices查看 ...

  2. Axure 环境进度条

    步骤一:拖拉摆放好相关控件 1.4个半圆环,一个白色上半圆环 (上白),一个白色下半圆环 (下白),一个灰色上半圆环 (上灰),一个灰色下半圆环 (下灰),排放层次为: 下灰<下白<上灰& ...

  3. STM32F103和AIR32F103的FreeRTOS中断优先级

    关于 Arm Cortex M 系列内核的中断优先级 https://community.arm.com/arm-community-blogs/b/embedded-blog/posts/cutti ...

  4. C#中检查null的语法糖,非常实用

    c#处理null的几个语法糖,非常实用.(尤其是文末Dictionary那个案例,记得收藏) ??如果左边是的null,那么返回右边的操作数,否则就返回左边的操作数,这个在给变量赋予默认值非常好用. ...

  5. 微软外服札记④——Spark中的那些坑...

    Spark中的那些坑 Spark中的那些坑 前言 读取配置文件 时区陷阱 怪异的DayOfWeek substring陷阱 IP地址解析 枚举的数值 posexplode函数 为什么我的程序运行那么慢 ...

  6. 使用ProPerties集合存储数据-Properties集合中的方法store

    使用ProPerties集合存储数据 java.util.Properties`继承于Hashtable,来表示一个持久的属性集.它使用键值结构存储数据每个键及其对应值都是一个字符串.该类也被许多Ja ...

  7. Linux实时同步

    实时同步 一.概述 背景: 以前,通过rsync+定时任务实现对文件的定时备份/同步 现在,对于NFS来说,需要进行实时同步 选择: 分布式存储 使用实时同步服务+NFS 选择公有云对象存储OSS,七 ...

  8. 0基础搭建基于OpenAI的ChatGPT钉钉聊天机器人

    前言:以下文章来源于我去年写的个人公众号.最近chatgpt又开始流行,顺便把原文内容发到博客园上遛一遛. 注意事项和指引: 注册openai账号,需要有梯子进行访问,最好是欧美国家的IP,亚洲国家容 ...

  9. C#DataTable(转List /JSON/字典 互转)

    1 #region 数据表DataTable 转键值对集合 List 2 /// <summary> 3 /// 数据表DataTable 转键值对集合 List 4 /// 把DataT ...

  10. Linux提权之:利用capabilities提权

    Linux提权之:利用capabilities提权 目录 Linux提权之:利用capabilities提权 1 背景 2 Capabilities机制 3 线程与文件的capabilities 3. ...