当前位置: 代码迷 >> 综合 >> 【贪心】Atcoder 3869 Tiling
  详细解决方案

【贪心】Atcoder 3869 Tiling

热度:26   发布时间:2023-09-27 08:12:18.0

题意:

给出一个N?MN?M的方阵,在方阵中放入A个1?21?2的方块,以及B个2?12?1的方块,求是否能够放下,如果能,输出方案。
N,M1000N,M≤1000
A,B500000A,B≤500000


分析:

很容易想到一点,从左上角开始,依次拆分成2*2的方格,每个方格中放入两个横向\纵向,一定能够使得空余的位置尽量少。当然,我们要注意一下放入的种类:
首先,如果N为奇数,那么在2*2分解后,就会剩下一行,那一行只能放横向的。
如果M为奇数,分解后会剩下一列,那一列只能放纵向的。
我们将那些只能放入某种方块的数量统计出来,如果当前要放入的方块数大于那个值,就一定需要放入2*2格子中,否则就不用放入。

但如果就这么交,是要WA的,因为一种特殊情况:
【贪心】Atcoder 3869 Tiling
这种3*3的格子中,能够放入2个横向,2个纵向的方格,但这只能在N,M均为奇数时才会出现这种情况。所以如果N,M均为大于1的奇数,且A,B均为大于1的整数,那么就可以考虑在右下角加入一个这种形状的摆放方式。分别check一下即可。

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#define SF scanf
#define PF printf
#define MAXN 1010
using namespace std;
int ans[MAXN][MAXN];
char w[10]={
   '.','<','>','^','v'};
int n,m;
int ablx,ably;
bool work(int a,int b,int flag){if(n%2==1)ablx=m/2;if(m%2==1)ably=n/2;if(flag==1){ablx--;ably--;}for(int i=1;i+1<=n;i+=2)for(int j=1;j+1<=m;j+=2){if(ans[i][j]||ans[i][j+1]||ans[i+1][j]||ans[i+1][j+1])continue;if(a>ablx){ans[i][j]=1;ans[i][j+1]=2;a--;if(a>0){ans[i+1][j]=1;ans[i+1][j+1]=2;a--;}}else if(b>ably){ans[i][j]=3;ans[i+1][j]=4;b--;if(b>0){ans[i][j+1]=3;ans[i+1][j+1]=4;b--;}}}if(n%2==1){for(int j=1;j+1<=m;j+=2)if(a>0&&ans[n][j]==0&&ans[n][j+1]==0){ans[n][j]=1;ans[n][j+1]=2;a--;}}if(m%2==1){for(int i=1;i+1<=n;i+=2)if(b>0&&ans[i][m]==0&&ans[i+1][m]==0){ans[i][m]=3;ans[i+1][m]=4;b--;}}return a||b;
}
void print(){PF("YES\n");for(int i=1;i<=n;i++){for(int j=1;j<=m;j++)PF("%c",w[ans[i][j]]);PF("\n");}
}
int main(){int a,b;SF("%d%d%d%d",&n,&m,&a,&b);if(work(a,b,0)==0){print();}else if(n%2==1&&m%2==1&&n>=3&&m>=3&&a>=2&&b>=2){memset(ans,0,sizeof ans);ans[n][m]=4;ans[n-1][m]=3;ans[n-2][m]=2;ans[n-2][m-1]=1;ans[n-2][m-2]=3;ans[n-1][m-2]=4;ans[n][m-2]=1;ans[n][m-1]=2;if(work(a-2,b-2,1)==0)print();elsePF("NO");}else{PF("NO");}
}
  相关解决方案