CF175E Power Defence
CF175E Power Defence
题意
一个塔防游戏:给定一个无限长的数轴,一个无限血的敌人要从正无穷走到负无穷。你的任务是放置三种塔,包含两种攻击塔和一种寒冰塔,使得敌人受到的伤害最大。
其中,每种塔的攻击半径可能不同,每种攻击塔的攻击力也可能不同。而寒冰塔没有攻击力,它的作用是使范围内敌人的速度减速,即一段区间若有\(k\)个寒冰塔覆盖,敌人速度变为\(\frac{1}{k+1}\)。
敌人初始速度为1格每秒,攻击塔的伤害值也是以秒计算的。
思路
首先有几个基本结论:
- 若没有寒冰塔,整场游戏答案固定。
- 若有寒冰塔,让所有塔挤在一起必不会更劣。这里对挤在一起的定义是:所有塔至少有一端上下对齐,且左右两端距离最大不超过\(\lceil \frac{n}{2}\rceil\)。
- 寒冰塔的能力实际上是将范围内基础伤害增加一倍。注意这里的伤害是忽略其他寒冰塔效果的。
发现序列长度很短,排列的方式也很有限,于是我们就有了一个大胆的想法:模拟退火。
然后便是裸的模拟退火算法:随机出来两个位置进行交换,统计答案,若答案大于当前答案,接受之;否则以当前温度的概率接受之。参数调整得好的话多做几次便可。
然后是统计答案。我有一个\(n^2\)的想法:扫描序列,每扫到一个寒冰塔就再次扫描整个序列的攻击塔,计算出两个塔间相交的范围乘上该塔攻击力加入答案即可,当然最后别忘了加上不计算寒冰塔的基础伤害。正确性也显然,因为显然寒冰塔的贡献是可以对于每个其他攻击塔单独计算的。
具体实现我耍了个小聪明:将所有炮塔放进一个数组中,这样交换的时候就不用分上下位置了。统计答案的时候计算出该塔的实际位置就行。
#include<iostream>
#include<cstdio>
#include<cstring>
#include<cctype>
#include<algorithm>
#include<cmath>
#include<ctime>
#define T 10000
#define eps 0.99
#define kb 1.38e-23
#define endt 1e-12
#define INF 1e19
using namespace std;
inline int read(){
int w=0,x=0;char c=getchar();
while(!isdigit(c))w|=c=='-',c=getchar();
while(isdigit(c))x=(x<<3)+(x<<1)+(c^48),c=getchar();
return w?-x:x;
}
namespace star
{
const int maxn=45;
int n1,n2,n3,d1,d2,n,a[maxn],m;
double r1,r2,r3,res,ans;
inline double getsum(){
double sum=0;
for(int i=1;i<=n;i++){
if(a[i]!=3)continue;
int x=i;
if(x>m)x-=m;
double L3=x-r3,R3=x+r3;
for(int j=1;j<=n;j++){
if(a[j]==3)continue;
int y=j,t=a[j];
if(y>m)y-=m;
if(t==1){
double L1=y-r1,R1=y+r1;
sum+=max(0.0,d1*(min(R1,R3)-max(L1,L3)));
}else if(t==2){
double L2=y-r2,R2=y+r2;
sum+=max(0.0,d2*(min(R2,R3)-max(L2,L3)));
}
}
}
return res+sum;
}
inline void solve()
{
double t=T;
double ans=0;
while(t>endt)
{
int xx=rand()%n+1,yy=rand()%n+1;
while(a[xx]==a[yy])xx=rand()%n+1,yy=rand()%n+1;
swap(a[xx],a[yy]);
double zp=getsum();
if(zp>ans){
ans=zp;
star::ans=max(star::ans,ans);
}else if(rand()<exp((ans-zp)/t/kb) * RAND_MAX)swap(a[xx],a[yy]);
else ans=zp;
t*=eps;
}
}
inline void work(){
n1=read(),n2=read(),n3=read();
r1=read();
r1=sqrt(r1*r1-1);
r2=read();
r2=sqrt(r2*r2-1);
r3=read();
r3=sqrt(r3*r3-1);
d1=read(),d2=read();
res=2*n1*r1*d1+2*n2*r2*d2;
if(n3==0)return (void)printf("%.10lf",res);
n=n1+n2+n3;
for(int i=1;i<=n;i++)
if(n1) a[i]=1,n1--;
else if(n2) a[i]=2,n2--;
else if(n3) a[i]=3,n3--;
m=n/2+1;
n=m*2;
random_shuffle(a+1,a+1+n);
srand(time(0));
// while((double)clock()/CLOCKS_PER_SEC<2.7)
for(int i=1;i<=100;i++)
solve();
printf("%.10lf\n",ans);
}
}
signed main(){
star::work();
return 0;
}
总结
这题正解是DP,爆搜也能过。
codeforces的题目,如果我现场写模拟退火的话一定当场去世。毕竟我因为一些奇怪的原因交了好多回。
所以骗骗分还是可以的
by ysr
UPD:2020.12.29
发现这个好像是个爬山
CF175E Power Defence的更多相关文章
- PatentTips - Fast awake from low power mode
BACKGROUND Electronic devices, such as electronic book readers ("eBook reader devices"), c ...
- 一起学微软Power BI系列-使用技巧(5)自定义PowerBI时间日期表
1.日期函数表作用 经常使用Excel或者PowerBI,Power Pivot做报表,时间日期是一个重要的纬度,加上做一些钻取,时间日期函数表不可避免.所以今天就给大家分享一个自定义的做日期表的方法 ...
- 一起学微软Power BI系列-使用技巧(4)Power BI中国版企业环境搭建和帐号问题
千呼万唤的Power BI中国版终于落地了,相信12月初的微软技术大会之后已经铺天盖地的新闻出现了,不错,Power BI中国版真的来了,但还有些遗憾,国际版的一些重量级服务如power bi emb ...
- 一起学微软Power BI系列-使用技巧(3)Power BI安卓手机版安装与体验
Power BI有手机版,目前支持安卓,苹果和WP,不过没有WP手机,苹果在国内还不能用,要FQ和用就不测试了.安卓的我也也是费了九牛二虎之力才把app下载下来,把方法分享给大家. FQ太麻烦,所以建 ...
- Power BI官方视频(3) Power BI Desktop 8月份更新功能概述
Power BI Desktop 8月24日发布了更新版本.现将更新内容翻译整理如下,可以根据后面提供的链接下载最新版本使用. 1.主要功能更新 1.1 数据钻取支持在线版 以前的desktop中进行 ...
- 一起学微软Power BI系列-使用技巧(1)连接Oracle与Mysql数据库
说起Oracle数据库,以前没用过Oracle不知道,但是这1年用Oracle后,发现真的是想狂吐槽,特别是那个.NET驱动和链接字符串,特别奇葩.总归是和其他数据库不一样,标新立异,不知道为何.另外 ...
- 千呼万唤始出来,微软Power BI简体中文版官网终于上线了,中文文档也全了。。
前几个月时间,研究微软Power BI技术,由于没有任何文档和资料,只能在英文官网瞎折腾,同时也发布了英文文档的相关文章:系列文章,刚好上周把文章发布完,结果简体中文版上线了.哈哈,心里有苦啊,早知道 ...
- 微软新神器-Power BI横空出世,一个简单易用,还用得起的BI产品,你还在等什么???
在当前互联网,由于大数据研究热潮,以及数据挖掘,机器学习等技术的改进,各种数据可视化图表层出不穷,如何让大数据生动呈现,也成了一个具有挑战性的可能,随之也出现了大量的商业化软件.今天就给大家介绍一款逆 ...
- 免费的精品: Productivity Power Tools 动画演示
Productivity Power Tools 是微软官方推出的 Visual Studio 扩展,被用以提高开发人员生产率.它的出现一定程度上弥补和完善了 Visual Studio 自身的不足, ...
随机推荐
- 自动化工具之Appium工具简单介绍
背景 自动化,性能测试,接口测试,开发平台等工作,到底测试的价值在哪里,其实价值来源不断充实与为大众服务,今天简单介绍ui小工具appium攻击. 简单介绍 Appium 是一个自动化测试开源工具,支 ...
- MySQL:如何使用MyCAT实现分库分表?
分库分表介绍 随着微服务这种架构的兴起,我们应用从一个完整的大的应用,切分为很多可以独立提供服务的小应用.每个应用都有独立的数据库. 数据的切分分为两种: 垂直切分:按照业务模块进行切分,将不同模块的 ...
- 连接过的WiFi改了密码之后再次连接不让输入新密码还是用旧密码一直显示连接失败
设置---网络和Internet---WLAN----管理已知网络----忘记 根据这个步骤就能忘记密码,重新输入新密码了.
- 解决SpringMVC重复提交的问题
方法一:通过重定向采取请求转发的方式完成表单内容的添加会造成内容的重复插入.当向Servlet发送一条增加记录的请求后,servlet首先向数据库增加一条记录,然后又从数据库中查询出所有数据,接着转发 ...
- (Vue中)cehart在同一个dom上画图图切换时饼图有折线图的坐标系
网上都是别人转载的,下面是转载的代码,在Vue中根本不适用 var echartrunningstate = null; if (echartrunningstate && echar ...
- Java并发之Semaphore源码解析(二)
在上一章,我们学习了信号量(Semaphore)是如何请求许可证的,下面我们来看看要如何归还许可证. 可以看到当我们要归还许可证时,不论是调用release()或是release(int permit ...
- 线上BUG:MySQL死锁分析实战
原文链接:线上BUG:MySQL死锁分析实战 1 线上告警 我们不需要关注截图中得其他信息,只要能看到打印得org.springframework.dao.DeadlockLoserDataAcces ...
- ps 合并两张图片
1.ps 打开第一张图片,2.打开另一张图片为图层.3.选中图层,创建蒙版: 4.点击蒙版+按Alt键,打开蒙版:5.打开蒙版后选中渐变工具,途中黑色为不显示区域,(注意渐变模式要改为正常):6.调整 ...
- POJ 3087 Shuffle'm Up 模拟,看着不像搜索啊
题意:给定s1,s1两副扑克,顺序从下到上.依次将s2,s1的扑克一张一张混合.例如s1,ABC; s2,DEF. 则第一次混合后为DAEBFC. 然后令前半段为s1, 后半段为s2. 如果可以变换成 ...
- Git常用命令超级详细(全网最详细)
1.新建代码库 1.1在当前目录新建一个 Git 代码库 $ git init 1.2新建一个目录,将其初始化为 Git 代码库 $ git init [project-name] 1.3下载一个项目 ...