题目描述

经过小FF的研究,他发现蚂蚁们每次都走同一条长度为n个单位的路线进攻, 且蚂蚁们的经过一个单位长度所需的时间为T秒。也就是说,只要小FF在条路线上布防且给蚂蚁造成沉痛伤害就能阻止蚂蚁的进军。

SCV擅长制造的防御塔有三种,分别是激光塔,放射塔和干扰塔, 他们可以在一个单位长度内修建一座防御塔。三种防御塔的作用如下:

激光塔: 使用高能激光,当蚂蚁从塔前经过时每秒对蚂蚁造成r点伤害。

放射塔: 释放放射性元素, 当蚂蚁经过这座塔后,每一秒受到g点伤害。

干扰塔: 干扰塔负责干扰蚂蚁们的信息素,使得蚂蚁在经过这座塔后,经过之后每一个单位长度的时间变成T+b。

当然, 放射塔和干扰塔的效果是可以叠加的, 也就是说如果敌人经过x座放射塔,那么敌人每秒钟会受到x * g点伤害; 同理,如果敌人经过y座干扰塔, 那么敌人经过一个单位长度的时间将变为T+y * b。

现在距离蚂蚁的下一轮进攻还有足够长的时间,你这个“NewBe_One”计划的首席工程师现在被任命为战略总参谋长, 因此你必须设计一个给蚂蚁们造成最大伤害的布塔方案。

输入格式

输入数据仅一行, 5个整数 n, r, g, b, T中间用一个空格隔开。 它们分别表示你可以布防的总长度, 激光塔的效果、 放射塔的效果和干扰塔的效果。

对于30%的数据: 1<=n<=20;

对于60%的数据: 1<=n<=1024;0<=r, g, b<=65536;0<=T<=3;

对于另外40%的数据:1<=n<=400;0<=r, g, b<=2^31-1;0<=t<=1000.

输出格式

输出仅一个整数, 代表你的方案给敌人带来的最大伤害值。


我们先逮着400的数据下手:

这道题最好想的就是动态规划了,几个状态一开出来直接转移即可。首先设dp(i,j,k,l,0/1/2),其中i表示当前在第i个位置,已经放置了j个放射塔,并且放了k个干扰塔和l个激光塔,0表示i这个位置放放射塔,1表示干扰塔,2表示激光塔。先不管爆内存的事,列出状态转移方程:

\[dp[i][j][k][l][0]=Max_{0≤x≤2}{\{}dp[i-1][j-1][k][l][x]{\}}+g*(j-1)*(t+k*b)\\
dp[i][j][k][l][1]=Max_{0≤x≤2}{\{}dp[i-1][j][k-1][l][x]{\}}+g*j*(t+(k-1)*b)\\
dp[i][j][k][l][2]=Max_{0≤x≤2}{\{}dp[i-1][j][k][l-1][x]{\}}+(g*j+r)*(t+k*b)
\]

可以发现每个状态都由上一个状态的最大值更新而来,所以我们可以省掉最后一维,让dp数组自带Max:

\[ans0=dp[i-1][j-1][k][l]+g*(j-1)*(t+k*b)\\
ans1=dp[i-1][j][k-1][l]+g*j*(t+(k-1)*b)\\
ans2=dp[i-1][j][k][l-1]+(g*j+r)*(t+k*b)\\
dp[i][j][k][l]=Max{\{}ans0,ans1,ans2{\}}
\]

然后我们发现,更新dp数组时我们只需要用到第i-1和第i位的信息,i-2及之前的地方都浪费了,所以我们可以把它滚掉:

\[ans0=dp[!d][j-1][k][l]+g*(j-1)*(t+k*b)\\
ans1=dp[!d][j][k-1][l]+g*j*(t+(k-1)*b)\\
ans2=dp[!d][j][k][l-1]+(g*j+r)*(t+k*b)\\
dp[d][j][k][l]=Max{\{}ans0,ans1,ans2{\}},d{\in}[0,1]
\]

然后我们又发现,每个位置必须放一个塔,不然显然没有防塔的优。那么当我们枚举到i这个位置时,我们可以由i-j-k推出l的值。所以我们只需要三层循环就够了。并且由于j,k确定之后,l也是确定的,那么我们可以干脆掐掉l这一维:

\[ans0=dp[!d][j-1][k]+g*(j-1)*(t+k*b)\\
ans1=dp[!d][j][k-1]+g*j*(t+(k-1)*b)\\
ans2=dp[!d][j][k]+(g*j+r)*(t+k*b)\\
dp[d][j][k]=Max{\{}ans0,ans1,ans2{\}},d{\in}[0,1]
\]

然后经过提交发现,d这一维要不要都无所谓。其实感性地证明一下也是可以得出这个结论的。那么:

\[ans0=dp[j-1][k]+g*(j-1)*(t+k*b)\\
ans1=dp[j][k-1]+g*j*(t+(k-1)*b)\\
ans2=dp[j][k]+(g*j+r)*(t+k*b)\\
dp[j][k]=Max{\{}ans0,ans1,ans2{\}}
\]

那么最初级的算法就设计出来了,时间复杂度为O(N^3)。然后发现开O2可以过?


N^3显然不是正解。

我们看到上面暴力的最终转移方程,可以发现我们的dp数组只与放置的塔的个数有关,与当前在哪个位置无关,那么可以这么认为:上面的暴力算法中枚举位置时其实是在确定激光塔的数量。所以说我们只需要枚举j和k,而l可以直接n-j-k得出,所以我们只需要两层循环。不过伤害值怎么算?

我们分析一下题目:

如果我们在中间放下激光塔,即它的后面还有放射塔和干扰塔,那么可以发现我们把随便一个后面的放射塔和干扰塔拿来给激光塔换一下都会更优。进一步我们得出一个结论:激光塔必定放在末尾的位置,并且是连续的一串激光塔。

所以我们可以枚举了j和k之后直接根据j和k算出激光塔的伤害了。具体看代码:

#include<iostream>
#include<cstring>
#include<cstdio>
#define maxn 1025
using namespace std; __int128 dp[maxn][maxn];
__int128 n,r,g,b,t;
__int128 ans; inline __int128 read(){
register __int128 x(0),f(1); register char c(getchar());
while(c<'0'||'9'<c){ if(c=='-') f=-1; c=getchar(); }
while('0'<=c&&c<='9') x=(x<<1)+(x<<3)+(c^48),c=getchar();
return x*f;
} void print(__int128 x){
if(!x) return;
print(x/10);
putchar(x%10+'0');
} int main(){
n=read(),r=read(),g=read(),b=read(),t=read();
//先计算放射塔和干扰塔的伤害,激光塔最后算
for(register int i=1;i<=n;i++){
for(register int j=0;i+j<=n;j++){
if(!j) dp[i][j]=dp[i-1][j]+g*(i-1)*(t+j*b);
else dp[i][j]=max(dp[i-1][j]+g*(i-1)*(t+j*b),dp[i][j-1]+g*i*(t+(j-1)*b));
}
}
//找最大值作为答案,并且加上激光塔的伤害
for(register int i=0;i<=n;i++){
for(register int j=0;i+j<=n;j++){
register int k=n-i-j;
if(k) ans=max(ans,dp[i][j]+(g*i+r)*k*(t+j*b));
}
}
print(ans);
return 0;
}

* 数据需要开高精,为了可读性(和懒)代码用了__int128。

luogu P2198 杀蚂蚁的更多相关文章

  1. Luogu2586 [ZJOI2008]杀蚂蚁 ---- 模拟

    Luogu2586 [ZJOI2008]杀蚂蚁 题意 还是一道大模拟 https://www.luogu.org/problemnew/show/P2586 大概就是炮塔大蚂蚁的故事 下载这个游戏ht ...

  2. 【BZOJ 1033】 [ZJOI2008]杀蚂蚁antbuster

    Description 最近,佳佳迷上了一款好玩的小游戏:antbuster.游戏规则非常简单:在一张地图上,左上角是蚂蚁窝,右下角是蛋糕,蚂蚁会源源不断地从窝里爬出来,试图把蛋糕搬回蚂蚁窝.而你的任 ...

  3. BZOJ 1033 杀蚂蚁

    Description 最近,佳佳迷上了一款好玩的小游戏:antbuster.游戏规则非常简单:在一张地图上,左上角是蚂蚁窝,右下角是蛋糕,蚂蚁会源源不断地从窝里爬出来,试图把蛋糕搬回蚂蚁窝.而你的任 ...

  4. bzoj千题计划121:bzoj1033: [ZJOI2008]杀蚂蚁antbuster

    http://www.lydsy.com/JudgeOnline/problem.php?id=1033 经半个下午+一个晚上+半个晚上 的 昏天黑地调代码 最终成果: codevs.洛谷.tyvj上 ...

  5. [BZOJ 1033][ZJOI2008]杀蚂蚁antbuster

    1033: [ZJOI2008]杀蚂蚁antbuster Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1200  Solved: 507[Submi ...

  6. BZOJ1033:[ZJOI2008]杀蚂蚁antbuster(模拟)

    Description 最近,佳佳迷上了一款好玩的小游戏:antbuster.游戏规则非常简单:在一张地图上,左上角是蚂蚁窝,右 下角是蛋糕,蚂蚁会源源不断地从窝里爬出来,试图把蛋糕搬回蚂蚁窝.而你的 ...

  7. [bzoj1033] [ZJOI2008]杀蚂蚁antbuster

    Description 最近,佳佳迷上了一款好玩的小游戏:antbuster.游戏规则非常简单:在一张地图上,左上角是蚂蚁窝,右下角是蛋糕,蚂蚁会源源不断地从窝里爬出来,试图把蛋糕搬回蚂蚁窝.而你的任 ...

  8. BZOJ1033:[ZJOI2008]杀蚂蚁

    我对模拟的理解:https://www.cnblogs.com/AKMer/p/9064018.html 题目传送门:https://www.lydsy.com/JudgeOnline/problem ...

  9. [ZJOI2008]杀蚂蚁

    题意翻译 注意在(0,0)已经有蚂蚁的时候是不会生成新蚂蚁的 还有如果有蚂蚁扛着蛋糕,但是不在某个炮的范围内,炮仍然会打最近的蚂蚁 题目描述 最近,佳佳迷上了一款好玩的小游戏:antbuster. 游 ...

随机推荐

  1. Cloudera Manager API使用

    Cloudera Manager提供了很多API供用户进行访问,因此可以利用其提供的API进行大数据集群中各种服务的状态检测.监控以及服务的启停操作. 版本信息 Cloudera Manager 6. ...

  2. Maven史上最全的pom.xml详解

    下面主要是借鉴 官网的资料 收集而来 主要是为了讲解,用到的很少,但是还是需要了解 ,重点是方便查验资料 <project xmlns="http://maven.apache.org ...

  3. .net MVC 微信公众号 点击菜单拉取消息时的事件推送

    官方文档:https://mp.weixin.qq.com/wiki?t=resource/res_main&id=mp1421141016&token=&lang=zh_CN ...

  4. Excel 冻结窗口

    1.冻结前五行 鼠标选中第六行,点击视图----> 冻结窗口 ----> 冻结拆分窗口 2.冻结第一列窗口 鼠标选中第1列,点击视图----> 冻结窗口 ----> 冻结首列窗 ...

  5. python初学者-从键盘获取信息

    name = input(">>> 姓名:") QQ = input(">>>QQ: ") phone_num = inpu ...

  6. 拥抱 C/C++ : Android JNI 的使用

    编译工具 CMake 以及 Android 上 JNI 的使用介绍. 编译工具 CMake 在Android Studio 2.2 之后,工具中增加了 CMake 的支持,于是我们有两种选择来编译 c ...

  7. jsp页面技术总结

    1.日期框只能选择不能输入属性设置 data-options="editable:false" 2.列表中多出一列空白 .datagrid-view1{ width: 0px!im ...

  8. ESP8266系列图片外观 Wi-Fi模块一共有01~14十多款模块

  9. Java对象赋值与引用

    当需要创建多个相同类型的对象且有某些字段的值是相同的,如果直接 get,set 的话,属性多的时候代码会很长,于是乎,以下代码产生了( java 基础差没搞清楚赋值与引用) 复制代码 1 User u ...

  10. Light Probes

    LightProbes (光照探针,光探测器?) 提供了一种方法用于捕获和使用 穿过场景中空白空间的 光(light)的信息. 和光照贴图相似(lightmaps),Light Probes也存储关于 ...