【IOI1998】Polygon 区间DP
题意翻译
题目可能有些许修改,但大意一致
多边形是一个玩家在一个有n个顶点的多边形上的游戏,如图所示,其中n=4。每个顶点用整数标记,每个边用符号+(加)或符号*(乘积)标记。

第一步,删除其中一条边。随后每一步:
选择一条边连接的两个顶点V1和V2,用边上的运算符计算V1和V2得到的结果来替换这两个顶点。
游戏结束时,只有一个顶点,没有多余的边。
如图所示,玩家先移除编号为3的边。之后,玩家选择计算编号为1的边,然后计算编号为4的边,最后,计算编号为2的边。结果是0。

(翻译者友情提示:这里每条边的运算符旁边的数字为边的编号,不拿来计算)
编写一个程序,给定一个多边形,计算最高可能的分数。
输入格式
输入描述一个有n个顶点的多边形,它包含两行。第一行是数字n,为总边数。
第二行描述这个多边形,一共有2n个读入,每两个读入中第一个是字符,第二个是数字。
第一个字符为第一条边的计算符号(t代表相加,x代表相乘),第二个代表顶点上的数字。首尾相连。
3 < = n < = 50
对于任何一系列的操作,顶点数字都在[-32768,32767]的范围内。
输出格式
第一行,输出最高的分数。在第二行,它必须写出所有可能的被清除后的边仍能得到最高得分的列表,必须严格递增。
输入输出样例
4
t -7 t 4 x 2 x 5
33
1 2
------------------------------------------------------------------- 这道题就用来练一下区间dp了
首先这是一个环状多边形,将其拆成链状处理
a[i+n]=a[i]; op[i+n]=op[i];//断链-> n*2
令 dp[l][r]表示以 l 为首项,以 r 为末项的链经过删边可以得到的最大值
然后,列出状态转移方程。这分为两部分:+和* 1.对于+,即op[i+1]=='t';
dp[l][r]=max(dp[l][k]+dp[k+1][r],dp[l][r]);
2.对于*,情况稍微复杂:因为负负得正,所以我们还需要在转移时记录下
最小值[绝对值最大],用f[i][j]记录
dp[l][r]=max( dp[l][r], max(dp[l][k]*dp[k+1][r],f[l][k]*f[k+1][r]) );
f[l][r]=min( f[l][r], min(dp[l][k]*dp[k+1][r],f[l][k]*f[k+1][r]) );
还要对边界进行处理,
令a[i]为第i个的初始值,那么:
dp[i][i]=f[i][i]=a[i];
同时,我们要先预处理长度为2的区间的答案
if(op[i+1]=='t')
dp[i][i+1]=f[i][i+1]=a[i]+a[i+1];
if(op[i+1]=='x')
dp[i][i+1]=f[i][i+1]=a[i]*a[i+1];
最后!!!输入的时候注意令人智熄的scanf!!!!!
(查错两小时
因为本题中字符和数字在一行中输入,所以
直接写scanf("%s%d",&op[i],&a[i]);会导致读到的是空格
所以要写成scanf("%d\n",&n); 和scanf("%s %d ",&op[i],&a[i]);
但是对于我来说,经常记不到的话,以后有同时输入字符和数字的时候,应该果断选择cin
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define N 120
#define INF 0x7f7f7f7f
#define ll long long
using namespace std;
ll f[N][N],dp[N][N];
int n,a[N];
char op[N];
ll ans=;
int main()
{
scanf("%d",&n);
for(int i=;i<=n;i++)
{
// scanf("%c %d ",&op[i],&a[i]);
cin>>op[i]>>a[i];
a[i+n]=a[i]; op[i+n]=op[i];//断链-> n*2
}
for(int i=;i<=n*;i++)
for(int j=i;j<=n*;j++)
f[i][j]=INF,dp[i][j]=-INF;
for(int i=;i<=n*;i++)//预处len=2
{
if(op[i+]=='t')
dp[i][i+]=f[i][i+]=a[i]+a[i+];
if(op[i+]=='x')
dp[i][i+]=f[i][i+]=a[i]*a[i+];
dp[i][i]=f[i][i]=a[i];
}
for(int len=;len<=n;len++)
for(int l=;l<=n*-len+;l++)
{
int r=l+len-;
for(int k=l;k<r;k++)
{
if(op[k+]=='t')
{
dp[l][r]=max(dp[l][k]+dp[k+][r],dp[l][r]);
f[l][r]=min(f[l][k]+f[k+][r],f[l][r]);
}
else//乘法可能有负负得正 故记录最大和最小值
{
dp[l][r]=max( dp[l][r], max(dp[l][k]*dp[k+][r],f[l][k]*f[k+][r]) );
f[l][r]=min( f[l][r], min(dp[l][k]*dp[k+][r],f[l][k]*f[k+][r]) );
}
}
}
for(int i=;i<=n;i++)
//cout<<dp[i][i+n-1]<<" ";
ans=max(ans,dp[i][i+n-]);
printf("%lld\n",ans);
for(int i=;i<=n;i++)
if(ans==dp[i][i+n-])
printf("%d ",i);
return ;
}
/*
scanf("%d\n",&n);
scanf("%c %d ",&op[i],&a[i]);
和
scanf("%d",&n);
scanf("%c%d",&op[i],&a[i]);
*/
ovo
【IOI1998】Polygon 区间DP的更多相关文章
- IOI1998 Polygon [区间dp]
[IOI1998]Polygon 题意翻译 题目可能有些许修改,但大意一致 多边形是一个玩家在一个有n个顶点的多边形上的游戏,如图所示,其中n=4.每个顶点用整数标记,每个边用符号+(加)或符号*(乘 ...
- [IOI1998] Polygon (区间dp,和石子合并很相似)
题意: 给你一个多边形(可以看作n个顶点,n-1条边的图),每一条边上有一个符号(+号或者*号),这个多边形有n个顶点,每一个顶点有一个值 最初你可以把一条边删除掉,这个时候这就是一个n个顶点,n-2 ...
- POJ 1179 - Polygon - [区间DP]
题目链接:http://poj.org/problem?id=1179 Time Limit: 1000MS Memory Limit: 10000K Description Polygon is a ...
- IOI 98 (POJ 1179)Polygon(区间DP)
很容易想到枚举第一步切掉的边,然后再计算能够产生的最大值. 联想到区间DP,令dp[i][l][r]为第一步切掉第i条边后从第i个顶点起区间[l,r]能够生成的最大值是多少. 但是状态不好转移,因为操 ...
- 【POJ1179】Polygon 区间DP
这道题是典型的环形石子归并模型,破环成链后时间复杂度为\(O(n^3)\) 不过,因为题目中所给的数字可能是负数,仅仅记录区间内合并之后的最大值并不满足动态规划的最优子结构性质.因此,还需要额外记录下 ...
- POJ1179 Polygon 区间DP
题目大意: 多边形游戏,有N个顶点的多边形,3 <= N <= 50 ,多边形有N条边,每个顶点中有一个数字(可正可负),每条边上或者是“+”号,或者是“*”号.边从1到N编号,首先选择一 ...
- 【洛谷 P4342】[IOI1998]Polygon(DP)
题目链接 题意不再赘述. 这题和合并石子很类似,但是多了个乘法,而乘法是不满足"大大得大"的,因为两个非常小的负数乘起来也会很大,一个负数乘一个很大的整数会很小,所以我们需要添加一 ...
- 「IOI1998」「LuoguP4342」Polygon(区间dp
P4342 [IOI1998]Polygon - 洛谷 题意翻译 题目可能有些许修改,但大意一致 多边形是一个玩家在一个有n个顶点的多边形上的游戏,如图所示,其中n=4.每个顶点用整数标记,每个边用符 ...
- [IOI1998]Polygon(区间dp)
[IOI1998]Polygon 题意翻译 多边形是一个玩家在一个有n个顶点的多边形上的游戏,如图所示,其中n=4.每个顶点用整数标记,每个边用符号+(加)或符号*(乘积)标记. 第一步,删除其中一条 ...
随机推荐
- Servlet详解之两个init方法的作用
在Servlet中 javax.servlet.GenericServlet类 继承自java.lang.Object 实现了Serializable,,servlet ,ServletConfig ...
- JavaScript_5_对象
1. JavaScrip中所有事物都是对象:字符串.数字.日期.等等 2. 在javaScripe中,对象是拥有属性和方法的数据 <!DOCTYPE html> <html> ...
- HDU 3709 Balanced Number (数位DP)
题意: 找出区间内平衡数的个数,所谓的平衡数,就是以这个数字的某一位为支点,另外两边的数字大小乘以力矩之和相等,即为平衡数. 思路: 一开始以为需要枚举位数,枚举前缀和,枚举后缀和,一旦枚举起来就会M ...
- 关于日志造成的频繁的IO
记录日志可能消耗大量的IO [Q] 每次写入都是一个IO操作 即使是同一个文件 两次写入也要打开两次IO操作 [F] 设想有这样一个扩展 把php中要记录的日志 用文件名 和 内容的方式记录在内存中 ...
- UVA 11992 Fast Matrix Operations (降维)
题意:对一个矩阵进行子矩阵操作. 元素最多有1e6个,树套树不好开(我不会),把二维坐标化成一维的,一个子矩阵操作分解成多条线段的操作. 一次操作的复杂度是RlogC,很容易找到极端的数据(OJ上实测 ...
- UVA 1442 Cave 洞穴 (贪心+扫描)
题意:有一个洞穴,每个位置有一个底的高度p[i],和对应顶的高度s[i],要往里面尽量放燃料,要求燃料不能碰到顶,可以无限接近. 题解:制约燃料储放的就是顶的高度了,分别求出设当前储放位置的向两边的延 ...
- UVA1607 Gates 与非门电路 (二分)
题意:给你一个按发生时间的序列,表示与非门电路的输入,一开始全部输入是x,现在要改成尽量少的x,实现相同的功能. 题解:电路功能只有4中0,1,x,非x.那么如果一开始x改变了,输出结果不变,那么说明 ...
- iBatis for net 框架使用
简介:ibatis 一词来源于“internet”和“abatis”的组合,是一个由Clinton Begin在2001年发起的开放源代码项目,到后面发展的版本叫MyBatis但都是指的同一个东西.最 ...
- Your Ride Is Here
纯粹的水题= = /* ID:yk652321 LANG:C++ TASK:ride */ #include<iostream> #include<cstring> #incl ...
- Codeforces Round #513 (rated, Div. 1 + Div. 2)
前记 眼看他起高楼:眼看他宴宾客:眼看他楼坍了. 比赛历程 开考前一分钟还在慌里慌张地订正上午考试题目. “诶这个数位dp哪里见了鬼了???”瞥了眼时间,无奈而迅速地关去所有其他窗口,临时打了一个缺省 ...