传送门

Sitting in Line

Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 262144/262144 K (Java/Others)
Total Submission(s): 293    Accepted Submission(s): 143

Problem Description
度度熊是他同时代中最伟大的数学家,一切数字都要听命于他。现在,又到了度度熊和他的数字仆人们玩排排坐游戏的时候了。游戏的规则十分简单,参与游戏的N个整数将会做成一排,他们将通过不断交换自己的位置,最终达到所有相邻两数乘积的和最大的目的,参与游戏的数字有整数也有负数。度度熊为了在他的数字仆人面前展现他的权威,他规定某些数字只能在坐固定的位置上,没有被度度熊限制的数字则可以自由地交换位置。
 
Input
第一行一个整数T,表示T组数据。
每组测试数据将以如下格式从标准输入读入:

N

a1p1

a2p2

:

aNPN

第一行,整数 N(1≤N≤16),代表参与游戏的整数的个数。

从第二行到第 (N+1) 行,每行两个整数,ai(−10000≤ai≤10000)、pi(pi=−1 或 0≤pi<N),以空格分割。ai代表参与游戏的数字的值,pi代表度度熊为该数字指定的位置,如果pi=−1,代表该数字的位置不被限制。度度熊保证不会为两个数字指定相同的位置。

 
Output
第一行输出:"Case #i:"。i代表第i组测试数据。

第二行输出数字重新排列后最大的所有相邻两数乘积的和,即max{a1⋅a2+a2⋅a3+......+aN−1⋅aN}。

 
Sample Input
2
6
-1 0
2 1
-3 2
4 3
-5 4
6 5
5
40 -1
50 -1
30 -1
20 -1
10 -1
 
Sample Output
Case #1:
-70
Case #2:
4600
 
Source
 
Recommend
wange2014   |   We have carefully selected several similar problems for you:  5695 5694 5693 5692 5690 
 
 
 
题意:
数组中有些元素位置被固定了,有些可以换位置,求
max{a1⋅a2+a2⋅a3+......+aN−1⋅aN}
 
题解:
真不容易,田神给我说了思路,还编了好久2333

状压dp,比赛时没时间思考了。。。

其实,观察所要求的max{a1⋅a2+a2⋅a3+......+aN−1⋅aN}

可以发现,你发完前 i - 1 位 元素后,准备放第 i 位 元素时,当前的 max 只 与 a[i-1] 有关了 (a[i - 1] * a[i]),所以可以用状压dp

定义 dp[o][i] 为放完第i位(以a[i] 为结尾),状态 o 时的max

遍历的顺序见代码

17232841 2016-05-21 22:22:30 Accepted 5691 1294MS 11724K 2946B G++ czy
 #include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <algorithm>
#include <vector> using namespace std; #define N 19
#define ll long long ll ma;
ll a[N];
int p[N];
int have[N];
int n;
ll dp[ ( << ) ][N];
int tot;
ll inf;
vector<int> G[N]; int cal(int x)
{
int ret = ;
while(x){
ret += (x & );
x /= ;
}
return ret;
} void ini()
{
int o;
for(o = ;o < n;o++){
G[o].clear();
}
for(o = ;o < tot;o++){
G[ cal(o) ].push_back(o);
}
} int main()
{ int T;
//freopen("in.txt","r",stdin);
//freopen("out.txt","w",stdout);
scanf("%d",&T);
for(int ccnt = ;ccnt <= T;ccnt++){
scanf("%d",&n);
inf = (1LL << );
ma = -inf;
tot = ( << n);
ini();
int i;
memset(have,-,sizeof(have));
for(i = ;i < n;i++){
scanf("%I64d%d",&a[i],&p[i]);
if(p[i] != -){
have[ p[i] ] = i;
}
}
int o,nt;
for(o = ;o < tot;o++){
for(i = ;i < n;i++){
dp[o][i] = -inf;
}
}
for(i = ;i < n;i++){
if(p[i] == || p[i] == -){ //放在第一个
dp[ ( << i) ][i] = ;
}
} /*
for(i = 0;i < n;i++){
for(int j = 0;j < G[i].size();j++){
printf(" i =%d j = %d g= %d\n",i,j,G[i][j]);
}
}*/
for(i = ;i < n;i++){ //放第i位
int sz = G[i].size();
for(int j = ;j < sz;j++){ //遍历含有i个1的所有数
o = G[i][j];
for(int k = ;k < n;k++) //把第k个数放在第i位
{ if( o & ( << k) ) continue; //k已经放了
if( p[k] != - && p[k] != i ) continue; //k被固定了
if( have[i] != - && have[i] != k ) continue; //位置i被固定了
nt = o | ( << k);
//printf(" i = %d o = %d k = %d nt = %d\n",i,o,k,nt);
for(int pr = ;pr < n;pr++){
if( (o & ( << pr ) ) == ) continue; //pr不在o里
if( dp[o][pr] == -inf ) continue;
//printf(" i = %d o = %d k = %d nt = %d pr = %d\n",i,o,k,nt,pr);
dp[nt][k] = max(dp[nt][k],dp[o][pr] + a[pr] * a[k]);
}
}
}
}
/*
for(o = 0;o < tot;o++){
for(i = 0;i < n;i++){
printf(" o = %d i = %d dp = %I64d\n",o,i,dp[o][i]);
}
}*/
for(i = ;i < n;i++){
ma = max(ma,dp[tot - ][i]);
}
printf("Case #%d:\n",ccnt);
printf("%I64d\n",ma);
} return ;
}

hdu 5691 Sitting in Line的更多相关文章

  1. HDU 5691 ——Sitting in Line——————【状压动规】

    Sitting in Line Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 262144/262144 K (Java/Other ...

  2. hdu 5691 Sitting in line 状压动归

    在本题中,n<=16n<=16n<=16, 不难想到可以将所选数字的编号进行状态压缩. 定义状态 dp[S][j]dp[S][j]dp[S][j],其中 SSS 代表当前所选出的所有 ...

  3. hdu 5691 Sitting in Line 状压dp

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=5691 题解: 和tsp用的状压差不多,就是固定了一些访问顺序. dp[i][j]表示前cnt个点中布 ...

  4. 2016"百度之星" - 初赛(Astar Round2A)1002 / HDU 5691 状态压缩DP

    Sitting in Line Problem Description   度度熊是他同时代中最伟大的数学家,一切数字都要听命于他.现在,又到了度度熊和他的数字仆人们玩排排坐游戏的时候了.游戏的规则十 ...

  5. HDU5691 Sitting in Line【状压DP】

    HDU5691 Sitting in Line 题意: 给出\(n\)个数字,有些数字的位置固定了,现在要求把所有没固定的数字放在一个位置,使得任意相邻两个位置的数字的相乘的和最大 题解: \(n\) ...

  6. hdu 5691(状压DP) Sitting in Line

    题目http://acm.hdu.edu.cn/showproblem.php?pid=5691 状态DP,dp[i][j],i 表示的是一种状态,这个状态指的是当前这个数取或不取,j表示的是以第j个 ...

  7. HDU 3623 Best Cow Line, Gold(模拟,注意思路,简单)

    题目 POJ 3617 和 这道题题目一样,只是范围稍稍再小一点. //模拟试试 #include<stdio.h> #include<string.h> #include&l ...

  8. hdu5691 Sitting in Line(状压dp)

    1 #include<iostream> #include<cstdio> #include<cstring> #include<algorithm> ...

  9. 2016百度之星 初赛2A ABEF

    只做了1001 1002 1005 1006.剩下2题可能以后补? http://acm.hdu.edu.cn/search.php?field=problem&key=2016%22%B0% ...

随机推荐

  1. 更新Svn客户端后,右键菜单中没有TortoiseSVN

    环境: OS:                 Windows XP sp3 升级后SVNServer:    VisualSVN Server 2.7.3 升级后SVNClient:    小乌龟: ...

  2. vue watch 监听

    1.普通的watch data() { return { frontPoints: 0 } }, watch: { frontPoints(newValue, oldValue) { console. ...

  3. c#写出乘法口诀

    显然是显得无聊五分钟写的乘法口诀 static void Main(string[] args)        {            int dq;            int[] array ...

  4. 总结 Swift 中随机数的使用

    在我们开发的过程中,时不时地需要产生一些随机数.这里我们总结一下Swift中常用的一些随机数生成函数.这里我们将在Playground中来做些示例演示. 整型随机数 如果我们想要一个整型的随机数,则可 ...

  5. webAssmebly实现js数组排序 使用custom elements和Shadow DOM自定义组件

    直接上码了……………… .wat源码 (module (type $t0 (func (param i32 i32))) (type $t1 (func (result i32))) (type $t ...

  6. MySQL Group Replication(组复制MGR)

    MGR基本要求: 1.InnoDB存储引擎 2.主键,每个表必须具有已定义的主键或等效的主键,其中等效项是非null唯一键 3.IPv4网络 4.网络性能 5.开启二进制日志并开启GTID模式 6.m ...

  7. redis集群理解

    Redis在3.0中也引入了集群的概念,用于解决一些大数据量和高可用的问题,但是,为了达到高性能的目的,集群不是强一致性的,使用的是异步复制,在数据到主节点后,主节点返回成功,数据被异步地复制给从节点 ...

  8. (22)zabbix触发器依赖关系详解

    概述 zabbix触发器可以设置依赖性,例如我配置了两个触发器,一个触发器定义www.ttlsa.com这个HOST是否在运行中,另一个是www.ttlsa.com的网络是否通畅. 假如网络出现故障, ...

  9. python 基础知识汇总—— if else while continue

    1.if 语句 什么是if语句?if语句用来干什么的? if语句说通俗点,就是判断,如果判断条件为真,那么就执行语句,就像我们生活中例子,如果你饿了,判断为真,就要吃饭,于是你就会执行吃饭这个动作,如 ...

  10. LeetCode(1)Two Sum

    题目: Given an array of integers, find two numbers such that they add up to a specific target number. ...