vijos 1037 搭建双塔 【DP】

% 这题是小猫给我讲的 %

题目:

Mr. F有N块水晶,每块水晶有一个高度,他想用这N块水晶搭建两座有同样高度的塔,使他们成为一座双塔,Mr. F可以从这N块水晶中任取M(1≤M≤N)块来搭建。但是他不知道能否使两座塔有同样的高度,也不知道如果能搭建成一座双塔,这座双塔的最大高度是多少。所以他来请你帮忙。

给定水晶的数量N(1≤N≤100)和每块水晶的高度Hi(N块水晶高度的总和不超过2000),你的任务是判断Mr. F能否用这些水晶搭建成一座双塔(两座塔有同样的高度),如果能,则输出所能搭建的双塔的最大高度,否则输出“Impossible”。

分析:

我们发现此题十分关注双塔的高度差,而进一步想到:如果已知两塔的高度差,那么再添一块水晶,无非就只有减小高度差或是增大高度差这两种影响。(增大高度差的情况万不可忽略,因为有可能将来会出现一块大水晶突然补齐高度差使达到最优解)。

f [ i ] [ j ] 表示用前 i 个水晶搭出两座塔并且高的那个高度为 j

有4种情况:

1、不放

2、放在低的上面并且没有超过高的

3、放在低的上面并且超过了高的

4、放在了高的上面

下面是参考代码:

/*f[i][j]表示用前i个水晶搭出两座塔并且高的那个高度为j*/
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
int a[400],n,ans=0;
int f[3000][3000];
int main()
{
 cin>>n;
 for(int i=1;i<=n;i++)
  cin>>a[i];
 memset(f,-2,sizeof(f));
 f[0][0]=0;
 for(int i=1;i<=n;i++)
  for(int j=0;j<=2000;j++)
  {
   f[i][j]=max(f[i-1][j],f[i-1][j+a[i]]);//不放和放了却没超过高塔
   if(a[i]<j)
    f[i][j]=max(f[i][j],f[i-1][j-a[i]]+a[i]);//放在高塔上
   else
    f[i][j]=max(f[i][j],f[i-1][a[i]-j]+j);//放在低塔上并且超过了高塔
  }
 for(int i=1;i<=n;i++)
  ans=max(f[i][0],ans);
 if(ans==0)
  cout<<"Impossible";
 else
  cout<<ans;
 return 0;
}

时间: 2024-10-11 05:46:33

vijos 1037 搭建双塔 【DP】的相关文章

shu_1299 vijos 1037(搭建双塔)

http://202.121.199.212/JudgeOnline/problem.php?cid=1078&pid=7 分析: 经典DP题 dp[ i ][ j ]:前i个水晶搭建的高度差为 j 的双塔中较高塔的高度: dp[ i ][ j ] = d[ i-1 ][ j ]  ,                       不放第i个水晶: dp[ i ][ j ] = f[ i-1 ][ h[i]-j ] + j,               第i个水晶放在较矮的塔上,原来较矮的塔变成较

VIJOS P1037搭建双塔[DP]

描述 2001年9月11日,一场突发的灾难将纽约世界贸易中心大厦夷为平地,Mr. F曾亲眼目睹了这次灾难.为了纪念“9?11”事件,Mr. F决定自己用水晶来搭建一座双塔. Mr. F有N块水晶,每块水晶有一个高度,他想用这N块水晶搭建两座有同样高度的塔,使他们成为一座双塔,Mr. F可以从这N块水晶中任取M(1≤M≤N)块来搭建.但是他不知道能否使两座塔有同样的高度,也不知道如果能搭建成一座双塔,这座双塔的最大高度是多少.所以他来请你帮忙. 给定水晶的数量N(1≤N≤100)和每块水晶的高度H

vijos P1037搭建双塔

P1037搭建双塔 Accepted 标签:动态规划 背包 描述 2001年9月11日,一场突发的灾难将纽约世界贸易中心大厦夷为平地,Mr. F曾亲眼目睹了这次灾难.为了纪念“9?11”事件,Mr. F决定自己用水晶来搭建一座双塔. Mr. F有N块水晶,每块水晶有一个高度,他想用这N块水晶搭建两座有同样高度的塔,使他们成为一座双塔,Mr. F可以从这N块水晶中任取M(1≤M≤N)块来搭建.但是他不知道能否使两座塔有同样的高度,也不知道如果能搭建成一座双塔,这座双塔的最大高度是多少.所以他来请你

vijos 1243 生产产品 DP + 单调队列优化

LINK 题意:有1个产品,m个步骤编号为1~m.步骤要在n个机器人的手中生产完成.其中,第i个步骤在第j个机器人手中的生产时间给定为$T[i][j]$,切换机器人消耗cost.步骤必须按顺序,同一个机器人不能连续完成超过l个步骤.求完成所有步骤的最短时间是多少.其中$m<=10^5$,$n<=5$,$l<=5*10^4$ 思路:这题用DP考虑易得一个转移方程$dp[i][j]=\min^{i-1}_{v=i-L}{(dp[v][x] + sum[i][j] - sum[v][j]) +

ZOJ 2059 The Twin Towers(双塔DP)

The Twin Towers Time Limit: 2 Seconds      Memory Limit: 65536 KB Twin towers we see you standing tall, though a building's lost our faith will never fall. Twin towers the world hears your call, though you're gone it only strengthens our resolve. We

搭建双塔(codevs 1037)

描述 2001年9月11日,一场突发的灾难将纽约世界贸易中心大厦夷为平地,Mr. F曾亲眼目睹了这次灾难.为了纪念“9?11”事件,Mr. F决定自己用水晶来搭建一座双塔. Mr. F有N块水晶,每块水晶有一个高度,他想用这N块水晶搭建两座有同样高度的塔,使他们成为一座双塔,Mr. F可以从这N块水晶中任取M(1≤M≤N)块来搭建.但是他不知道能否使两座塔有同样的高度,也不知道如果能搭建成一座双塔,这座双塔的最大高度是多少.所以他来请你帮忙. 给定水晶的数量N(1≤N≤100)和每块水晶的高度H

vijosP1037搭建双塔

链接:https://vijos.org/p/1037 [思路] DP. [代码] 1 #include<iostream> 2 #include<cstring> 3 using namespace std; 4 5 const int maxn = 100+10; 6 7 int a[maxn]; 8 int d[maxn][2010]; 9 int n,sum; 10 11 int main() { 12 ios::sync_with_stdio(false); 13 cin

vijos 1037 背包+标记

描述 2001年9月11日,一场突发的灾难将纽约世界贸易中心大厦夷为平地,Mr. F曾亲眼目睹了这次灾难.为了纪念“9?11”事件,Mr. F决定自己用水晶来搭建一座双塔. Mr. F有N块水晶,每块水晶有一个高度,他想用这N块水晶搭建两座有同样高度的塔,使他们成为一座双塔,Mr. F可以从这N块水晶中任取M(1≤M≤N)块来搭建.但是他不知道能否使两座塔有同样的高度,也不知道如果能搭建成一座双塔,这座双塔的最大高度是多少.所以他来请你帮忙. 给定水晶的数量N(1≤N≤100)和每块水晶的高度H

Laoj P1299 搭建双塔

试题描述 2001年9月11日,一场突发的灾难将纽约世界贸易中心大厦夷为平地,Mr. F曾亲眼目睹了这次灾难.为了纪念"9?11"事件,Mr. F决定自己用水晶来搭建一座双塔. Mr. F有N块水晶,每块水晶有一个高度,他想用这N块水晶搭建两座有同样高度的塔,使他们成为一座双塔,Mr. F可以从这N块水晶中任取M(1≤M≤N)块来搭建.但是他不知道能否使两座塔有同样的高度,也不知道如果能搭建成一座双塔,这座双塔的最大高度是多少.所以他来请你帮忙. 给定水晶的数量N(1≤N≤100)和每