博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
loj 1406(状态压缩)
阅读量:6160 次
发布时间:2019-06-21

本文共 1526 字,大约阅读时间需要 5 分钟。

题目链接:

思路:首先可以预处理出在每个顶点的状态的合法状态vis[u][state], 然后标记那些合法状态mark[state]。最后就是记忆化搜索了,对于当前状态state,我们有res = min(res, 1 + Solve(state ^ substate)), 其中substate为state的子状态,并且substate = (substate  - 1) & state.那么最终就是要求Solve((1 << n) - 1)了。

1 #include 
2 #include
3 #include
4 #include
5 #include
6 using namespace std; 7 8 int n, m; 9 bool vis[16][1 << 16];10 bool mark[1 << 16];11 int dp[1 << 16];12 vector
g[16];13 14 void dfs(int u, int state)15 {16 vis[u][state] = true;17 mark[state] = true;18 for (int i = 0; i < (int)g[u].size(); i++) {19 int v = g[u][i];20 if (!vis[v][state | (1 << v)]) {21 dfs(v, state | (1 << v));22 }23 }24 }25 26 int Solve(int state)27 {28 if (state == 0) return 0;29 if (dp[state] != -1) return dp[state];30 int res = 16;31 for (int i = state; i > 0; i = (i - 1) & state) {32 if (mark[i]) {33 res = min(res, 1 + Solve(state ^ i));34 }35 }36 return dp[state] = res;37 }38 39 40 int main()41 {42 int _case, t = 1;43 scanf("%d", &_case);44 while (_case--) {45 scanf("%d %d", &n, &m);46 for (int i = 0; i < n; i++) g[i].clear();47 while (m--) {48 int u, v;49 scanf("%d %d", &u, &v);50 u--, v--;51 g[u].push_back(v);52 }53 memset(vis, false, sizeof(vis));54 memset(mark, false, sizeof(mark));55 for (int i = 0; i < n; i++) dfs(i, 1 << i);56 memset(dp, -1, sizeof(dp));57 printf("Case %d: %d\n", t++, Solve((1 << n) -1));58 }59 return 0;60 }
View Code

 

 

转载地址:http://puafa.baihongyu.com/

你可能感兴趣的文章
智力大冲浪
查看>>
虚拟机VMware 9安装苹果MAC OSX 10.8图文教程
查看>>
微信小程序开发-框架
查看>>
redo、undo、binlog的区别
查看>>
RecycleView设置顶部分割线(记录一个坑)
查看>>
汉字转拼音 (转)
查看>>
会计基础_001
查看>>
小程序: 查看正在写的页面
查看>>
Jenkins持续集成环境部署
查看>>
MWeb 1.4 新功能介绍二:静态博客功能增强
查看>>
预处理、const与sizeof相关面试题
查看>>
爬虫豆瓣top250项目-开发文档
查看>>
有趣的数学书籍
查看>>
teamviewer 卸载干净
查看>>
eclipse的maven、Scala环境搭建
查看>>
架构师之路(一)- 什么是软件架构
查看>>
USACO 土地购买
查看>>
【原创】远景能源面试--一面
查看>>
B1010.一元多项式求导(25)
查看>>
10、程序员和编译器之间的关系
查看>>