$n \leq 1e9$,$n*2$的网格里有$m_1+m_2 \leq 1e6$个障碍物,现有一坦克从$(0,1)$出发要到$(n+1,1/2)$,他每秒可以换行(纵坐标1变2或2变1)也可以发炮弹,然后会强制往前走一格(横坐标+1),一次炮弹打一个障碍,炮弹装载时间$t \leq n$,一开始炮弹没装好。问是否可以到达终点,若可以输出过程中的所有转弯和打炮操作的位置。

比较接地气的一个题。虽然没做出来但大爱。

首先可以把打炮看成“攒步数”,怎么讲呢?比如说坦克在某段时间一直往前走不换行,走到一个障碍物就不得不打他一下,那么他在撞到这障碍物前至少有$t$步空闲。类似地,两个障碍物就$2t$,$k$个就$kt$,就好比在攒步数,然后到障碍物处花掉。而换了行的话,由于不可以换行立即猛发炮弹(尽管之前攒了很多步数但刚换行最多只能打一发),因此步数将变成$min(当前步数,t)$。基于此可以进行一个dp:$f(i,j)$表示在格子$(i,j)$攒的步数,无法到达即$-1$。注意分是否是障碍物转移,因为一行有障碍物时另一行的坦克不可能直接撞上来。这样就得到了$O(n)$的解法。

$However$,这里$n$大大的!必须把复杂度与障碍物数联系起来!

其实dp的主要瓶颈在换行,如果不换行完全可以枚举每个障碍物,到那里$f(i,j)-=t$即可。那换行咋整啊?

yy可得:存在一最优策略使得换行操作一定在某个障碍物后进行。如下图所示,黑色块是某个转弯前的最后一个块,绿色的线路和红色线路攒的步数是一样的,但由于换行之后dp数组会对$t$取$min$,因此不如红色线路。

是一个贪心策略。这种dp+贪心是比较新颖的套路。

转移时注意,因为我表示的状态是“每个障碍物的下一列”,如果下一列有障碍物要提前打掉。

 //#include<iostream>
#include<cstring>
#include<cstdlib>
#include<cstdio>
//#include<map>
#include<math.h>
//#include<time.h>
//#include<complex>
#include<algorithm>
using namespace std; int n,m1,m2,t;
#define maxn 2000011
bool turn[][maxn]; int list[][maxn],f[][maxn];
struct BO{int pos,type;}bo[maxn],ans[maxn]; int lb=;
int lans,tt[maxn],ltt;
bool cmp(const BO a,const BO b) {return a.pos<b.pos;} int main()
{
scanf("%d%d%d%d",&n,&m1,&m2,&t);
for (int i=;i<=m1;i++) scanf("%d",&list[][i]);
for (int i=;i<=m2;i++) scanf("%d",&list[][i]);
for (int i=,j=;;)
{
if (i>m1) {for (;j<=m2;j++) bo[++lb]=(BO){list[][j],}; break;}
if (j>m2) {for (;i<=m1;i++) bo[++lb]=(BO){list[][i],}; break;}
if (list[][i]<list[][j]) bo[++lb]=(BO){list[][i++],};
else if (list[][i]>list[][j]) bo[++lb]=(BO){list[][j++],};
else bo[++lb]=(BO){list[][i++],},j++;
} f[][]=f[][]=; bo[].pos=-; bo[lb+].pos=n+; bo[lb+].type=;
for (int i=;i<=lb;i++)
{
if (bo[i-].pos!=bo[i].pos-)
{
f[][i]=f[][i-]==-?-:f[][i-]+(bo[i].pos-bo[i-].pos)-;
f[][i]=f[][i-]==-?-:f[][i-]+(bo[i].pos-bo[i-].pos)-;
if (bo[i].type== || bo[i].type==) f[][i]=max(-,f[][i]-t);
if (bo[i].type== || bo[i].type==) f[][i]=max(-,f[][i]-t);
f[][i]=f[][i]==-?-:f[][i]+;
f[][i]=f[][i]==-?-:f[][i]+;
}
else f[][i]=f[][i-],f[][i]=f[][i-];
if (bo[i+].pos==bo[i].pos+)
{
if (bo[i+].type== || bo[i+].type==) f[][i]=max(-,f[][i]-t);
if (bo[i+].type== || bo[i+].type==) f[][i]=max(-,f[][i]-t);
}
f[][i]=f[][i]==-?-:f[][i]+;
f[][i]=f[][i]==-?-:f[][i]+;
if (bo[i+].pos!=bo[i].pos+ || bo[i+].type== || bo[i+].type==)
{if (f[][i]<min(t,f[][i])) turn[][i]=,f[][i]=min(t,f[][i]);}
if (bo[i+].pos!=bo[i].pos+ || bo[i+].type== || bo[i+].type==)
{if (f[][i]<min(t,f[][i])) turn[][i]=,f[][i]=min(t,f[][i]);}
}
if (f[][lb]==- && f[][lb]==-) {puts("No"); return ;}
puts("Yes");
int now=; if (f[][lb]==-) now=; lans=ltt=; int cnt=;
for (int i=lb;i;i--)
{
if (turn[now][i])
{
tt[++ltt]=bo[i].pos+;
for (int j=bo[i].pos++t-f[now][i];cnt;j+=t,cnt--) ans[++lans]=(BO){j,now};
now^=; if (bo[i+].pos==bo[i].pos+ && (bo[i+].type==now || bo[i+].type==)) cnt++;
}
if (bo[i].type==now || bo[i].type==) cnt++;
}
if (now==) tt[++ltt]=;
for (int j=t;cnt;j+=t,cnt--) ans[++lans]=(BO){j,now};
printf("%d\n",ltt);
for (int i=ltt;i;i--) printf("%d ",tt[i]); puts("");
sort(ans+,ans++lans,cmp);
printf("%d\n",lans);
for (int i=;i<=lans;i++) printf("%d %d\n",ans[i].pos,ans[i].type+);
return ;
}

Codeforces936D. World of Tank的更多相关文章

  1. [ASE]项目介绍及项目跟进——TANK BATTLE·INFINITE

    童年的记忆,大概是每周末和小伙伴们围坐在电视机前,在20来寸的电视机屏幕里守卫着这个至今都不知道是什么的白色大鸟. 当年被打爆的坦克数量估计也能绕地球个三两圈了吧. 十几年过去了,游戏从2D-3D,画 ...

  2. poj3635Full Tank?[分层图最短路]

    Full Tank? Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 7248   Accepted: 2338 Descri ...

  3. Codeigniter整合Tank Auth权限类库的教程

    Codeigniter整合Tank Auth权限类库的教程一开始找了很多CodeIgniter的类库,觉得都不怎么样,后来干脆自己通过CI的钩子系统写了权限管理.但是还是不怎么满意,后来有人推荐tan ...

  4. poj 3635 Full Tank? ( bfs+dp思想 )

    Full Tank? Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 5857   Accepted: 1920 Descri ...

  5. hdu 4445 Crazy Tank (暴力枚举)

    Crazy Tank Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total ...

  6. Aquarium Tank(csu1634+几何+二分)Contest2087 - 湖南多校对抗赛(2015.05.24)-G

    Aquarium Tank Time Limit: 1 Sec  Memory Limit: 128 MBSubmit: 15  Solved: 4[Submit][Status][Web Board ...

  7. Full Tank? POJ - 3635 (bfs | 最短路)

    After going through the receipts from your car trip through Europe this summer, you realised that th ...

  8. POJ 3635 - Full Tank? - [最短路变形][手写二叉堆优化Dijkstra][配对堆优化Dijkstra]

    题目链接:http://poj.org/problem?id=3635 题意题解等均参考:POJ 3635 - Full Tank? - [最短路变形][优先队列优化Dijkstra]. 一些口胡: ...

  9. POJ 3635 - Full Tank? - [最短路变形][优先队列优化Dijkstra]

    题目链接:http://poj.org/problem?id=3635 Description After going through the receipts from your car trip ...

随机推荐

  1. AJPFX浅谈Java 性能优化之字符串过滤实战

    ★一个简单的需求 首先描述一下需求:给定一个 String 对象,过滤掉除了数字(字符'0'到'9')以外的其它字符.要求时间开销尽可能小.过滤函数的原型如下: String filter(Strin ...

  2. iframe及其引出的页面跳转问题

    前提:在前一段的工作中碰到了一些页面跳转,子页面跳到父页面上的等等问题,当时页面总是跳不对,或者跳错,要不就是不需要重新打开窗口,却又重新打开一个了,特此搜寻网上各大博客论坛,加上项目经验整理一篇文章 ...

  3. ["1", "2", "3"].map(parseInt)

    为什么["1", "2", "3"].map(parseInt) 为 1,NaN,NaN; parseInt() parseInt() 函数 ...

  4. Android学习笔记(十六) ContentProvider

    1.相关概念 ContentProvider:不同应用程序之间进行数据交换的标准API:程序“暴露”数据的方法. ContentResolver:一个程序访问另一个程序被“暴露”的数据的方法. Uri ...

  5. Linux系统结构与终端控制台

    Linux系统结构与终端控制台 作者:Vashon 时间:20150418 以下主要是对Linux系统终端控制台切换及基本操作的范例,其他的理论就不多说了,直接进入实践部分. Starting.... ...

  6. element ui select组件和table做分页完整功能和二级联动效果

    <template> <div class="index_box"> <div class="search_box"> &l ...

  7. laravel composer 扩展包开发(超详细)

    laravel composer 扩展包开发(超详细) 置顶 2018年02月05日 11:09:16 Simael__Aex 阅读数:10396    版权声明:转载请注明出处:http://blo ...

  8. CAD交互绘制带颜色宽度的直线(网页版)

    用户可以在CAD控件视区任意位置绘制直线. 主要用到函数说明: _DMxDrawX::DrawLine 绘制一个直线.详细说明如下: 参数 说明 DOUBLE dX1 直线的开始点x坐标 DOUBLE ...

  9. 关于apache access log 统计的那些事儿

    统计APACHE ACCESS.LOG IP访问记录 可以根据自己的需要,统计很多,每个IP访问多少个页面等等! cat access.log-20090904 |awk '{print $3}'|s ...

  10. python爬虫---实现项目(二) 分析Ajax请求抓取数据

    这次我们来继续深入爬虫数据,有些网页通过请求的html代码不能直接拿到数据,我们所需的数据是通过ajax渲染到页面上去的,这次我们来看看如何分析ajax 我们这次所使用的网络库还是上一节的Reques ...