poj 2288 Islands and Bridges (状压dp+Tsp问题)
这道题千辛万苦啊!
这道题要涉及到当前点和前面两个点,那就设dp[state][i][j]为当前状态为state,当前点为i,前一个点为j
这个状态表示和之前做炮兵那题很像,就是涉及到三个点时,就多设一维表示前一个点(炮兵那题把点换成行)
这道题有很多细节需要注意
(1)计算路径长度。这道题一开始怎么不重复又方便的计算长度难住了我。
后来看到题解直接在初始化的时候算上路径,非常牛逼
然后前两个点的路径就包含进去了。
首先加上前一个点和当前点的价值
然后看有没有构成三角形,有就再加上
(2)更新答案。这里更新答案要dp完了之后再弄,在dp时更新会出错
多打几行代码不会死的,重要是要ac,麻烦一点就麻烦一点
(3)初始化问题。这里谈谈填表法和刷表法初始化的不同
如果是刷表法,那么就不知道当前状态合不合理,那就每次都需要判断一下
一般来说一开始全部初始化为-1表示全部不合理,然后就把一开始合理的部分(比如起点)赋初值(一般为0)。
如果是填表法的话,一般来说不需要判断合不合理
但是这道题不一样,并不知道前两个点的状态是否合法,所以需要判断。
(4)这道题有个比较坑的地方,就是n=1时要特判
(5)然后自己头脑一定要清楚哪一个变量是第几个点!!
我一般是写i是当前点,j是前一个点,k是前前个点
(6)下标范围是0到n-1,那么就写1 << n
(7)方案数。这道题方案数最后要除以2,因为可以反着走,但题目里算同一种
然后dp弄方案一般可以开一个数组,意义是和dp数组一模一样的,只不过存的是方案数
然后符合就加上
#include<cstdio>
#include<cstring>
#include<algorithm>
#define REP(i, a, b) for(int i = (a); i < (b); i++)
#define _for(i, a, b) for(int i = (a); i <= (b); i++)
using namespace std;
typedef long long ll;
const int MAXN = 15;
int dp[(1 << 13) + 10][MAXN][MAXN], w[MAXN];
int g[MAXN][MAXN], n, m;
ll ways[(1 << 13) + 10][MAXN][MAXN];
int main()
{
int T;
scanf("%d", &T);
while(T--)
{
memset(g, 0, sizeof(g));
memset(dp, -1, sizeof(dp)); //初始化要注意
memset(ways, 0, sizeof(ways));
scanf("%d%d", &n, &m);
REP(i, 0, n) scanf("%d", &w[i]);
while(m--)
{
int u, v;
scanf("%d%d", &u, &v); u--; v--;
g[u][v] = g[v][u] = 1;
}
if(n == 1) { printf("%d 1\n", w[0]); continue; } //特判
REP(i, 0, n) //初始化
REP(j, 0, n)
if(g[i][j])
{
dp[(1<<i)|(1<<j)][i][j] = w[i] + w[j] + w[i] * w[j];
ways[(1<<i)|(1<<j)][i][j] = 1;
}
REP(S, 0, 1 << n)
REP(i, 0, n) if(S & (1 << i))
REP(j, 0, n) if((S & (1 << j)) && g[i][j])
REP(k, 0, n) if((S & (1 << k)) && g[j][k])
{
if(i == j || j == k || i == k || dp[S^(1<<i)][j][k] == -1) continue;
ll t = dp[S^(1<<i)][j][k] + w[i] + w[j] * w[i]; // 注意这里哪一个是最后一点
if(g[i][k]) t += w[i] * w[j] * w[k];
if(dp[S][i][j] < t)
{
dp[S][i][j] = t;
ways[S][i][j] = ways[S^(1<<i)][j][k];
}
else if(dp[S][i][j] == t) ways[S][i][j] += ways[S^(1<<i)][j][k]; //这里是else if 写if会错
}
ll ans = 0, num = 0; //分开来做
int S = (1 << n) - 1;
REP(i, 0, n)
REP(j, 0, n)
if(g[i][j])
{
if(ans < dp[S][i][j])
{
ans = dp[S][i][j];
num = ways[S][i][j];
}
else if(ans == dp[S][i][j])
num += ways[S][i][j];
}
printf("%lld %lld\n", ans, num / 2);
}
return 0;
}
poj 2288 Islands and Bridges (状压dp+Tsp问题)的更多相关文章
- poj 2288 Islands and Bridges ——状压DP
题目:http://poj.org/problem?id=2288 状压挺明显的: 一开始写了(记忆化)搜索,但一直T: #include<iostream> #include<cs ...
- poj 2288 Islands and Bridges——状压dp(哈密尔顿回路)
题目:http://poj.org/problem?id=2288 不知为什么记忆化搜索就是WA得不得了! #include<iostream> #include<cstdio> ...
- [poj2288] Islands and Bridges (状压dp)
Description Given a map of islands and bridges that connect these islands, a Hamilton path, as we al ...
- POJ 1185 炮兵阵地(状压DP)
炮兵阵地 Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 26426 Accepted: 10185 Descriptio ...
- POJ 2411 Mondriaan's Dream -- 状压DP
题目:Mondriaan's Dream 链接:http://poj.org/problem?id=2411 题意:用 1*2 的瓷砖去填 n*m 的地板,问有多少种填法. 思路: 很久很久以前便做过 ...
- POJ 2411 Mondriaan's Dream ——状压DP 插头DP
[题目分析] 用1*2的牌铺满n*m的格子. 刚开始用到动规想写一个n*m*2^m,写了半天才知道会有重复的情况. So Sad. 然后想到数据范围这么小,爆搜好了.于是把每一种状态对应的转移都搜了出 ...
- 【POJ 2923】Relocation(状压DP+DP)
题意是给你n个物品,每次两辆车运,容量分别是c1,c2,求最少运送次数.好像不是很好想,我看了网上的题解才做出来.先用状压DP计算i状态下,第一辆可以运送的重量,用该状态的重量总和-第一辆可以运送的, ...
- POJ 1185 炮兵阵地 (状压DP)
题目链接 题意 : 中文题不详述. 思路 :状压DP,1表示该位置放炮弹,0表示不放.dp[i][j][k],代表第 i 行的状态为k时第i-1行的状态为 j 时放置的最大炮弹数.只是注意判断的时候不 ...
- 动态规划晋级——POJ 3254 Corn Fields【状压DP】
转载请注明出处:http://blog.csdn.net/a1dark 分析:刚开始学状压DP比较困难.多看看就发现其实也没有想象中那么难.这道题由于列数较小.所以将行压缩成二进制来看.首先处理第一行 ...
- POJ 1185 炮兵阵地 【状压DP】
<题目链接> 题目大意: 司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队.一个N*M的地图由N行M列组成,地图的每一格可能是山地(用"H" 表示),也可能是平 ...
随机推荐
- find命令扩展
1.1 方法一 |xargs 通过|xargs将前面命令的执行结果传给后面. [root@znix ~]# find /clsn/ -type f -name "*.sh" |x ...
- Windows使用docker打开新窗口error解决办法
环境 win7 Error: error during connect: Get http://%2F%2F.%2Fpipe%2Fdocker_engine/v1.26/containers/json ...
- 提高生产力:发送邮件API和Web服务(包含源码)
在Web开发中,发邮件是一种非常常见的功能或任务. 发送邮件的6种方式 一文提到了6种方法,文章发表后,有网友指出了还有另外一种方法,Ant中也可以发送邮件. 打开Foxmail之类的邮件客户端或者在 ...
- 3、用js文件写mongo命令+update命令
一.用js写一个模拟用户登录日志表的信息 var userName = "chengxiang"; //声明一个登录名 var timeStamp = Date.parse(new ...
- 洛谷 U249 匹配
U249 匹配 题目描述 输入整数s和两个整数集合A和B,从这A和B中各取一个数,如果它们的和等于s,称为“匹配”.编程统计匹配的总次数 输入输出格式 输入格式: 第一行为三个整数s(0<s≤1 ...
- POJ 2018
又一水,设dp[i]为以i结尾的有最大平均值的起始位置. #include <iostream> #include <cstdio> #include <cstring& ...
- HDU 2838 (树状数组求逆序数)
题意: 给你N个排列不规则的数(1~N),任务是把它从小到大排好,每次仅仅能交换相邻两个数,交换一次的代价为两数之和.求最小代价 思路:对于当前数X.我们如果知道前面比它大的数有多少,如果为K,那么有 ...
- angularjs1-路由
<!DOCTYPE html> <html> <head> <meta http-equiv="Content-Type" content ...
- JavaScript:让你彻底弄清offset
ylbtech-JavaScript:让你彻底弄清offset 1.返回顶部 1. 很多初学者对于JavaScript中的offset.scroll.client一直弄不明白,虽然网上到处都可以看一张 ...
- DBS-Oracle:表的连接查询
ylbtech-DBS-Oracle:表的连接查询 链接查询是指基于两个或两个以上表或试图的查询.在实际应用中,查询单个表可能无法满足应用程序的实际需求(例如显示雇员的部门名称以及雇员名),在这种情况 ...