题意:输入n、m、p然后是n头牛塔坐标、m个草垛坐标,有p次指令机会。
每次指令可以向上下左右其中一种方向,让所有牛塔都移动一单位。
然后每经过草垛/牛塔次,就有1个权值。
求权值最大值及使权值最大的移动指令序列(字典序最小)。
题解:f[i][j][k]表示第i次,移动序列x轴坐标为j,y轴k,最大权值。
然后倒着做以保证字典序。
代码:
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
#define N 1010
#define M 32
#define inf 0x3f3f3f3f
using namespace std;
int n,m,p;
int map[N][N];
int cx[N],cy[N];
int lx=inf,rx,ly=inf,ry;
int f[M][M<<1][M<<1],ans;
char opt[M][M<<1][M<<1];
int main()
{
// freopen("test.in","r",stdin);
int i,j,k;
int a,b,c;
int x,y;
scanf("%d%d%d",&n,&m,&p);
for(i=1;i<=n;i++)scanf("%d%d",&cx[i],&cy[i]);
for(i=1;i<=m;i++)
{
scanf("%d%d",&x,&y);
map[x][y]=1;
lx=min(lx,x),rx=max(rx,x);
ly=min(ly,y),ry=max(ry,y);
}
for(i=p;i>=0;i--)// 走i步
{
for(j=-i;j<=i;j++)// x轴位移
{
int remain=i-abs(j);
for(k=-remain;k<=remain;k++)
{
int noxus=-1,demacia=0;
char tt;
for(int adc=1;adc<=n;adc++)
{
x=cx[adc]+j,y=cy[adc]+k;
if(lx<=x&&x<=rx&&ly<=y&&y<=ry&&map[x][y])demacia++;
}
if(f[i+1][j+1+M][k+M]>noxus)noxus=f[i+1][j+1+M][k+M],tt='E';
if(f[i+1][j+M][k+1+M]>noxus)noxus=f[i+1][j+M][k+1+M],tt='N';
if(f[i+1][j+M][k-1+M]>noxus)noxus=f[i+1][j+M][k-1+M],tt='S';
if(f[i+1][j-1+M][k+M]>noxus)noxus=f[i+1][j-1+M][k+M],tt='W';
f[i][j+M][k+M]=noxus+demacia;
opt[i][j+M][k+M]=tt;
}
}
}
printf("%d\n",f[0][0+M][0+M]);
x=y=0+M;
for(i=0;i<p;i++)
{
printf("%c",opt[i][x][y]);
if(opt[i][x][y]=='E')x++;
else if(opt[i][x][y]=='W')x--;
else if(opt[i][x][y]=='N')y++;
else if(opt[i][x][y]=='S')y--;
}
return 0;
}