题意: 给你一个双向连通图,求 获得权值最大 的 哈密顿通路的 权值 和 这个权值对应的数目;
其中权值计算方法是 列如 ABCD 权值是a+b+c+d+ab+bc+cd 如果 A,B,C 和B,C,D 可构成三角形分别加上abc,bcd;
这个题 和poj 3311 很相像: 那个需要记录一个最后到达的地方 这个需要记录俩个罢了
DP[i][a][b]其中 i 二进制 中1表示这个点走过了 最后走的的 的是b>>a
因为对于已经走过了{1,2,3,4,,5,6,..,N} 为了推出下一次走到X 我们要获得权值 只和最后走的俩个点有关?
给出代码::
#include <cstdio>
#include <algorithm>
#include <cmath>
#include <cstdlib>
#include <cstring>
#include <iostream>
using namespace std;
long long dp[<<][][];
long long num[<<][][];
int n;
int val[];
bool Map[][];
void solve()
{
for(int i=;i<(<<n);i++)for(int a=;a<n;a++)for(int b=;b<n;b++)
{
if(!Map[a][b])continue;
if(a==b)continue;
if((i&(<<a))==||(i&(<<b))==)continue;
if(i==((<<a)+(<<b)))
{
dp[i][a][b]=val[a]+val[b]+val[a]*val[b];
num[i][a][b]=;
continue;
}
else
{
for(int c=;c<n;c++)
{
if(c==a||c==b||(!Map[b][c])) continue;
if((i&(<<c))==) continue;
long long tmp;
if(dp[i^(<<a)][b][c]==-)continue;
if(Map[a][c]) tmp=val[a]+val[a]*val[b]+val[a]*val[b]*val[c];
else tmp=val[a]+val[a]*val[b];
if(dp[i][a][b]<dp[i^(<<a)][b][c]+tmp)
{
num[i][a][b]=num[i^(<<a)][b][c];
dp[i][a][b]=dp[i^(<<a)][b][c]+tmp;
}
else if(dp[i][a][b]==dp[i^(<<a)][b][c]+tmp)
num[i][a][b]+=num[i^(<<a)][b][c];
}
}
}
long long ans=-;
long long ans_num=;
for(int b=;b<n;b++) for(int c=;c<n;c++)
{
if(ans<dp[(<<n)-][b][c])
{
ans=dp[(<<n)-][b][c];
ans_num=num[(<<n)-][b][c];
}
else if(ans==dp[(<<n)-][b][c])
ans_num+=num[(<<n)-][b][c]; }
if(ans==-) printf("0 0\n");
else printf("%lld %lld\n",ans,ans_num/);
}
int main()
{
int q;
scanf("%d",&q);
while(q--)
{
memset(Map,false,sizeof(Map));
memset(dp,-,sizeof(dp));
memset(num,,sizeof(num));
int m;
scanf("%d%d",&n,&m);
for(int i=;i<n;i++)scanf("%d",&val[i]);
for(int i=;i<=m;i++)
{
int a,b;
scanf("%d%d",&a,&b);
a--,b--;
Map[a][b]=Map[b][a]=true;
}
if(n==) printf("%d 1\n",val[]);
else
solve();
}
return ;
}
poj 2288 Islands and Bridges的更多相关文章
-
POJ 2288 Islands and Bridges(状压dp)
http://poj.org/problem?id=2288 题意: 有n个岛屿,每个岛屿有一个权值V,一条哈密顿路径C1,C2,...Cn的值为3部分之和: 第1部分,将路径中每个岛屿的权值累加起来 ...
-
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> ...
-
【以前的空间】poj 2288 Islands and Bridges
一个不错的题解 : http://blog.csdn.net/accry/article/details/6607703 这是一道状态压缩.每个点有一个值,我们最后要求一个最值sum.sum由三部分组 ...
-
POJ 2288 Islands and Bridges (状压DP,变形)
题意: 给一个无向图,n个点m条边,每个点有点权,要求找到一条哈密顿路径,使得该路径的f(path)值最大.输出f值,若有多条最大f值的路径,输出路径数量. f值由如下3点累加而来: (1)所有点权之 ...
-
poj 2288 Islands and Bridges (状压dp+Tsp问题)
这道题千辛万苦啊! 这道题要涉及到当前点和前面两个点,那就设dp[state][i][j]为当前状态为state,当前点为i,前一个点为j 这个状态表示和之前做炮兵那题很像,就是涉及到三个点时,就多设 ...
-
POJ 2288 Islands and Bridges(状压DP)题解
题意:n个点,m有向边,w[i]表示i的价值,求价值最大的哈密顿图(只经过所有点一次).价值为:所有点的w之和,加上,每条边的价值 = w[i] * w[j],加上,如果连续的三个点相互连接的价值 = ...
-
poj 2288 Islands and Bridges_状态压缩dp_哈密尔顿回路问题
题目链接 题目描述:哈密尔顿路问题.n个点,每一个点有权值,设哈密尔顿路为 C1C2...Cn,Ci的权值为Vi,一条哈密尔顿路的值分为三部分计算: 1.每一个点的权值之和 2.对于图中的每一条CiC ...
-
poj 2280 Islands and Bridges 哈密尔顿路 状压dp
题目链接 题意 给定一个\(N\)个点的无向图,求一条哈密尔顿路径\(C_1C_2...C_n\),使其\(value\)最大. \(value\)的计算方式如下:\[\begin{aligned}v ...
随机推荐
-
Python3.5之TuShare
这部分是直接搬运过来的,官方网站http://tushare.waditu.com/ TuShare是一个免费.开源的python财经数据接口包.主要实现对股票等金融数据从数据采集.清洗加工 到 数据 ...
-
InterBase数据库迁移到MySQL(数据导出)
这篇我将记叙我的第二个脚本程序,这篇我使用InterBase数据库提供的“isql”命令来导出我所要的数据,但是由于“isql”命令没有直接导出数据的语句,说以我采用的是导入一个配置文件,在这个文件中 ...
-
php处理数组函数大全
PHP:指示支持该函数的最早的 PHP 版本. 函数 描述 PHP array() 创建数组. 3 array_change_key_case() 返回其键均为大写或小写的数组. 4 array_ch ...
-
《C++Primer》复习——with C++11 [3]
1.我们的程序经常使用很多IO库,用来输入输出例如:istream(输入流)类型,提供输入操作. ostream(输出流)类型, 提供输出操作. cin, 一个istream对象,从标准输入读取数据. ...
-
JAXB - Hello World
We'll stick with the tradition and use a sort of "Hello World" XML document to illustrate ...
-
openStack开源云repo db local or on-line 实战部署之Ruiy王者归来
preface/pre,获取OpenStack核心模块组件及其子组件包(当前仅针对centos6*)及其依赖包 eg,picture
-
Android Studio环境下代码混淆+签名打包
Android Studio环境下代码混淆+签名打包 作者 Mr_冯先生 关注 2016.08.21 01:10 字数 1040 阅读 734评论 5喜欢 34 注:本文使用的Android Stud ...
-
Ignite与Spark集成时,ClassNotFoundException问题解决
参考文章:https://apacheignite-fs.readme.io/docs/installation-deployment Spark application deployment mod ...
-
(六十九)使用block进行消息传递
在两个类之间进行消息传递,一般通过代理或者block进行,代理写起来较为麻烦,block较为简单,但是block需要特别注意内存泄漏问题,注意self和block之间要为弱引用,下面介绍使用block ...
-
c#核心基础--类的构造方法
一.构造方法 类的构造方法是类的成员方法的一种,它的作用是对类中的成员进行初始化操作.类的构造方法分为: 1.静态构造方法 2.实例构造方法 3.私有构造方法 1.静态构造方法 类的静态构造方法是类的 ...