2015-09-21

                

                造桥基建工程

  题目大意,就是有n座岛和k座桥,要你找一条哈密顿圈(找完所有的岛,并且每个岛只经过一次),当经过一座岛就加上岛的价值,如果两岛联通,则加上两座岛的价值之积,如果三座岛之间构成三角联通,则再加上三岛之积,问最大价值的哈密顿圈和最大价值和哈密顿圈的个数

  哈密顿圈是是一个NP完全的问题,用DP就可以解决这个问题,现在的问题就是,怎么解决呢?

  首先我们要明确,这一题要用DP做什么,首先这一题的最后肯定要求到最后岛全部都通过的情况,然后还需要保留前两个岛的信息

  那么这个时候我们可以想到用状态压缩去做(因为这里岛只能经过一次,而且经过以后状态都是等效的)101111,就表示第五个岛还没被经过,本来呢这个状态压缩应该是保留状态的,但是这样一来我们就需要保留1<<n个*3个维度,那么这一题我们可以放弃了,因为内存肯定不够,所以我们要想着优化储存空间,由于我们只用担心前两个岛是什么,而不是岛的所在状态(反正表示也是1001111)这样的,所以我们可以把两个岛的维度改为n*n,那么最后dp就是一个[1<<n][n][n]的三维数组

  具体而言,对于不合法的位置,我们老方法,要定义其为-1,同时还要判断图是否联通

  我们先从i=1(0000....0001)开始枚举,表示岛已经经过的位置,如果我们需要上一次还没来到这个岛的位置的状态,我们使用i^(1<<(j-1))这个方法,并且判断第j个岛的前两座岛是否有冲突。

  如果算出来的dp值比dp[i][j][k]要大,则更新,并且num是继承dp[state_old][k][q]的状态,如果想等,则使num[i][j][k]+=num[state_old][k][q](表示还有更多的路与此状态相关)。

  最后基准状态,q=0的时候可以放入岛的价值,那么最后就可以一起处理了,并且只有一个岛的时候,要单独处理,并且数据类型要是long long

  参考:http://blog.csdn.net/lenleaves/article/details/7981788

 #include <stdio.h>
#include <stdlib.h>
#include <string.h> static int islands_value[];
int map[][];
long long dp[ << ][][];//因为要保存两个以上状态,所以必须是三维数组,前面一位是状态数
long long num[ << ][][]; void Search(const int);
static int If_Valid(const int, const int); int main(void)
{
int sum_islands, sum_paths, sum_obj, i, j, k, x, y; scanf("%d", &sum_obj);
for (i = ; i < sum_obj; i++)//输入
{
scanf("%d%d", &sum_islands, &sum_paths);
for (j = ; j <= sum_islands; j++)
scanf("%d", &islands_value[j]);
memset(map, , sizeof(map));
memset(dp, -, sizeof(dp));
memset(num, , sizeof(num));
for (k = ; k < sum_paths; k++)
{
scanf("%d%d", &x, &y);
map[x][y] = map[y][x] = ;//读图1
}
Search(sum_islands);
}
return ;
} static int If_Valid(const int x, const int y)
{
/*If_Value函数:
功能:判断y位置是否被访问过
返回值:被访问过返回1,未被访问放回0,如果y是0,则就是基准情况,一定要给通过
*/
if (y == ) return ;
if (x&( << (y - ))) return ;
else return ;
} void Search(const int sum_islands)
{
int i, j, k, q, state_old;
long long tmp, ans1, ans2; if (sum_islands == )
{
printf("%d 1\n", islands_value[]);
return;
}
for (j = ; j <= sum_islands; j++)
{
dp[ << (j - )][j][] = islands_value[j];
num[ << (j - )][j][] = ;//初始化的时候价值为1
}
for (i = ; i < ( << sum_islands);i++)
{
for (j = ; j <= sum_islands; j++)
{
state_old = i ^ ( << (j - ));//得到当j还没有出现时候的位置
if (If_Valid(i, j))//这个位置就是要考虑的位置
{
for (k = ; k <= sum_islands; k++)
{
if (k != j && map[j][k] && If_Valid(i, k))
{
for (q = ; q <= sum_islands; q++)
{
if (!map[k][q] && q) continue;//q = 0的时候是没有意义的
if (q != j&&q != k
&& If_Valid(state_old, q)//是否是需要的q
&& dp[state_old][k][q] != -)//这个位置一定要有意义
{
tmp = islands_value[j] + islands_value[j] * islands_value[k] + dp[state_old][k][q];
if (map[j][q])
//假设j和q是联通的
tmp += islands_value[j] * islands_value[k] * islands_value[q];
if (tmp == dp[i][j][k])
//如果得到的结果是和ijk这个位置已经重复了,则自增(中间状态)
num[i][j][k] += num[state_old][k][q];
if (tmp > dp[i][j][k])
//如果比ijk这个状态还要大,则更新dp,继承num
{
dp[i][j][k] = tmp;
num[i][j][k] = num[state_old][k][q];
}
}
}
}
}
}
}
}
for (ans1 = -, ans2 = , j = ; j <= sum_islands; j++)
{
for (k = ; k <= sum_islands; k++)
{
if (k != j)
{
if (ans1 == dp[( << sum_islands) - ][j][k])
ans2 += num[( << sum_islands) - ][j][k];
if (ans1 < dp[( << sum_islands) - ][j][k])
{
ans1 = dp[( << sum_islands) - ][j][k];
ans2 = num[( << sum_islands) - ][j][k];
}
}
}
}
if (ans1 == -)
printf("0 0\n");
else
printf("%lld %lld\n", ans1, ans2 / );
}

DP:Islands and Bridges(POJ 2288)的更多相关文章

  1. 【状压dp】Islands and Bridges

    Islands and Bridges Time Limit: 4000MS   Memory Limit: 65536K Total Submissions: 11034   Accepted: 2 ...

  2. [poj2288] Islands and Bridges (状压dp)

    Description Given a map of islands and bridges that connect these islands, a Hamilton path, as we al ...

  3. HDU 1668 Islands and Bridges

    Islands and Bridges Time Limit: 4000ms Memory Limit: 65536KB This problem will be judged on HDU. Ori ...

  4. POJ2288 Islands and Bridges

    Description Given a map of islands and bridges that connect these islands, a Hamilton path, as we al ...

  5. poj 2288 Islands and Bridges ——状压DP

    题目:http://poj.org/problem?id=2288 状压挺明显的: 一开始写了(记忆化)搜索,但一直T: #include<iostream> #include<cs ...

  6. poj 2288 Islands and Bridges——状压dp(哈密尔顿回路)

    题目:http://poj.org/problem?id=2288 不知为什么记忆化搜索就是WA得不得了! #include<iostream> #include<cstdio> ...

  7. POJ 2288 Islands and Bridges(状压dp)

    http://poj.org/problem?id=2288 题意: 有n个岛屿,每个岛屿有一个权值V,一条哈密顿路径C1,C2,...Cn的值为3部分之和: 第1部分,将路径中每个岛屿的权值累加起来 ...

  8. Islands and Bridges(POJ 2288状压dp)

    题意:给你一个图和每个点的价值,边权值为连接两点权值的积,走哈密顿通路,若到达的点和上上个点相连则价值加三点乘积,求哈密顿通路的最大价值,和最大价值哈密顿通路的条数. 分析:开始看这个题很吓人,但想想 ...

  9. POJ 2288 Islands and Bridges (状压DP,变形)

    题意: 给一个无向图,n个点m条边,每个点有点权,要求找到一条哈密顿路径,使得该路径的f(path)值最大.输出f值,若有多条最大f值的路径,输出路径数量. f值由如下3点累加而来: (1)所有点权之 ...

随机推荐

  1. OI历程日常

    之前的一直没来的及记录,表示从今往后连载 10.29 蒟蒻正在紧张的备战NOIP 整改了一下faebdc学长的模拟题,T1直接可以暴力破解,T2二分,O(nlog^2n)开始二分写残了,调了半天唉,现 ...

  2. 洛谷P3386 【模板】二分图匹配

    匈牙利算法模板 /*by SilverN*/ #include<algorithm> #include<iostream> #include<cstring> #i ...

  3. sdibt 1244 烦人的幻灯片

    在这个OJ站还没号,暂时没提交,只是过了样例 真不愧是烦人的幻灯片,烦了我一小时 ---更新:OJ测试完毕,AC 烦人的幻灯片问题 Time Limit: 1 Sec  Memory Limit: 6 ...

  4. workon在zsh中不起作用

    先装了workon,然后装了zsh,发现在zsh里不起作用 翻了一下网上没有解答,就看了看bashrc文件,发现一句 source /usr/local/bin/virtualenvwrapper.s ...

  5. vector与set区别(基础知识)

    首先,vector是序列式容器而set是关联式容器.set包含0个或多个不重复不排序的元素.也就是说set能够保证它里面所有的元素都是不重复的.另外对set容器进行插入时可以指定插入位置或者不指定插入 ...

  6. IDEA 从SVN检出项目相关配置

    1.新建好一个工程,然后通过SVN检出项目 2.检出后一般tomcat的环境是配置好的,点击上方Project Structure按钮,弹出窗体,查看Project项,一般没问题,如果要配置就配置Pr ...

  7. Extreme Learning Machine(ELM)的工程哲学

    Extreme Learning Machine(ELM)的工程哲学 David_Wang2015 发布于2015年5月6日 11:29 工程问题往往需要的是一定精度范围内的结果,而不是“真正的”结果 ...

  8. 禁止掉非法IP登陆服务器

    今天查看线上服务器日志/var/log/secure发现有很多国外的ip尝试登陆服务器,以前一直没太注意这方面,作为系统管理员真是失职啊,虽然服务器已经设置了强密码,但是看到有人想搞你还是很不爽的.一 ...

  9. IOS学习笔记—苹果推送机制APNs

    转自:唐韧_Ryan http://blog.csdn.net/ryantang03/article/details/8482259 推送是解决轮询所造成的流量消耗和 电量消耗的一个比较好的解决方案, ...

  10. 使用 array_multisort 对多维数组排序

    array_multisort() 函数对多个数组或多维数组进行排序. 用法详看:http://www.w3school.com.cn/php/func_array_multisort.asp 例子: ...