[CF936D]World of Tank

World of Tank

题解

相当有趣的 d p dp dp题。

显然,如果我们要开炮击毁一个建筑,我们是可以提前在许多个位置开炮的,只要 CD 好了就可以开炮。
如果我们要将 CD 放到状态中的话,是相当麻烦的,毕竟 CD 是 1 0 9 10^9 109级别的。
事实上这里有一种更好地方法,我们可以不这样记录,我们记录我们可以存储多少的 CD 不用,等着撞到了障碍物,再将这些存着的 CD 消耗掉,表示在这些 CD 满时就打炮,再继续存储剩余的 CD
这样就可以将我们的 CD 放到 d p dp dp之里面,显然 CD 值是屯的越多越好的,所以我们取得肯定是 d p dp dp值的最大值。
在换道的时候,由于你没法提前开炮了,只能冷却你的炮,所以这时候需要将 d p dp dp值与 t t t去 m i n min min。
最后我们可以通过 d p dp dp值还原出我们的操作路径。

但这样 d p dp dp状态还是太多了, n n n是 1 0 9 10^9 109级别的。
我们发现我们转道肯定只会在障碍物之后转,而开炮有与状态无关。
所以我们只需要将所有的障碍物和其后一个位置加入 d p dp dp状态,离散化一下就只有 m 1 + m 2 m1+m2 m1+m2级别的状态了。

时间复杂度 O ( ( m 1 + m 2 ) log ⁡   ( m 1 + m 2 ) ) O\left((m1+m2)\log\,(m1+m2)\right) O((m1+m2)log(m1+m2))。

源码

#include<bits/stdc++.h>
using namespace std;
#define MAXN 2000005
#define lowbit(x) (x&-x)
#define reg register
#define pb push_back
#define mkpr make_pair
#define fir first
#define sec second
typedef long long LL;
typedef unsigned long long uLL; 
typedef long double ld;
typedef pair<int,int> pii;
const LL INF=0x7f7f7f7f7f7f7f7f;
const int mo=1e9+7;
const int inv2=5e8+4;
const int jzm=23333;
const int zero=20000;
const int lim=1000000;
const int orG=3,ivG=334845270;
const double Pi=acos(-1.0);
const double eps=1e-12;
template<typename _T>
_T Fabs(_T x){return x<0?-x:x;}
template<typename _T>
void read(_T &x){
	_T f=1;x=0;char s=getchar();
	while(s>'9'||s<'0'){if(s=='-')f=-1;s=getchar();}
	while('0'<=s&&s<='9'){x=(x<<3)+(x<<1)+(s^48);s=getchar();}
	x*=f;
}
template<typename _T>
void print(_T x){if(x<0){x=(~x)+1;putchar('-');}if(x>9)print(x/10);putchar(x%10+'0');}
int gcd(int a,int b){return !b?a:gcd(b,a%b);}
int add(int x,int y,int p){return x+y<p?x+y:x+y-p;}
void Add(int &x,int y,int p){x=add(x,y,p);}
int qkpow(int a,int s,int p){int t=1;while(s){if(s&1)t=1ll*t*a%p;a=1ll*a*a%p;s>>=1;}return t;}
int n,m1,m2,b[MAXN],tot,cnt,pre[MAXN][2],dp[MAXN][2],t;
bool maze[MAXN][2];
vector<int>ans1;
vector<pii>ans2;
struct ming{int x,y;}s[MAXN];
signed main(){
	read(n);read(m1);read(m2);read(t);b[++tot]=1;
	for(int i=1,x;i<=m1;i++)read(x),s[++cnt]=(ming){x,0},b[++tot]=x,b[++tot]=x+1;
	for(int i=1,x;i<=m2;i++)read(x),s[++cnt]=(ming){x,1},b[++tot]=x,b[++tot]=x+1;
	sort(b+1,b+tot+1);tot=unique(b+1,b+tot+1)-b-1;
	for(int i=1;i<=cnt;i++)s[i].x=lower_bound(b+1,b+tot+1,s[i].x)-b,maze[s[i].x][s[i].y]=1;
	memset(dp,-0x3f,sizeof(dp));
	dp[0][0]=0;
	for(int i=1;i<=tot;i++){
		int dif=b[i]-b[i-1];
		for(int j=0;j<2;j++){
			if(!maze[i][j]&&dp[i-1][j]>=0){
				dp[i][j]=max(dp[i][j],dp[i-1][j]+dif);
				if(dp[i-1][j]+dif==dp[i][j])pre[i][j]=0;
			}
			if(maze[i][j]&&dp[i-1][j]+dif>t){
				dp[i][j]=max(dp[i][j],dp[i-1][j]-t+dif);
				if(dp[i-1][j]-t+dif==dp[i][j])pre[i][j]=0;	
			}
			if(!maze[i][j]&&!maze[i-1][j]&&dp[i-1][j^1]>=0){
				dp[i][j]=max(dp[i][j],min(dp[i-1][j^1],t)+dif);
				if(min(dp[i-1][j^1],t)+dif==dp[i][j])pre[i][j]=1;
			}
			if(maze[i][j]&&!maze[i-1][j]&&dp[i-1][j^1]+dif>t){
				dp[i][j]=max(dp[i][j],min(dp[i-1][j^1],t)+dif-t);
				if(min(dp[i-1][j^1],t)+dif-t==dp[i][j])pre[i][j]=1;
			}
		}
	}
	if(dp[tot][0]<0&&dp[tot][1]<0){puts("No");return 0;}
	int now=dp[tot][0]>=0?0:1,pos=tot,siz;
	while(pos){
		int nxt=now^pre[pos][now];if(nxt^now)ans1.pb(b[pos-1]);
		if(maze[pos][now])ans2.pb(mkpr(b[pos]-dp[pos][now],2-(now^1)));
		now=nxt;pos--;
	}
	puts("Yes");
	printf("%d\n",siz=ans1.size());reverse(ans1.begin(),ans1.end());
	for(int i=0;i<siz;i++)printf("%d ",ans1[i]);puts("");
	printf("%d\n",siz=ans2.size());reverse(ans2.begin(),ans2.end());
	for(int i=0;i<siz;i++)printf("%d %d\n",ans2[i].fir,ans2[i].sec);
	return 0;  
}

谢谢!!!

上一篇:拍照时如何有效预防抖动?8个简单实用方法教你有效防抖


下一篇:springboot简介:hello,world