【贪心】Atcoder 3869 Tiling

题意:

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


分析:

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

但如果就这么交,是要WA的,因为一种特殊情况:
这里写图片描述
这种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();
            else
                PF("NO");
        }
    else{
        PF("NO");
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值