4761. 【NOIP2016提高A组模拟9.7】鼎纹

Description

Input

Output

Sample Input

2
3 4 4 2
1100
0110
1100
10
01
10
00
2 2 2 2
11
11
01
10

Sample Output

YES
NO

Data Constraint

 

Solution 

一个显然的结论:每次覆盖最左上⻆的能覆盖的,并把它删除。最后如果还有 1 剩 余就表示不可⾏。证明? 显然最左上⻆的现在不去覆盖,以后就没机会覆盖了……

然后我们就可以暴力扫一遍了,时间复杂度O(nmab),得分:70。

如何优化?我们发现0的位置是没有用的,所以我们只用记录1的位置,然后一个一个跳到下一个1的位置就好了。但这还不够,因为有些已经删除的点没必要再看是否能覆盖了,所以我们可以想到一个支持删除操作的双向链表的算法,将鼎面用双向链表储存,将铜模用各种东西乱搞,可以用链表老存也可以记录其他所有的点与最左上方的点的位置关系,然后再一个一个跳着匹配。得分:100。

Code1

#include<cstdio> 
#include<cstring>
#include<algorithm>
#define N 1002
using namespace std;
int T,n,m,a,b,x,y,i,j,k,l,bz,f[N*N],g[N*N],h[N*N][2],t;
char c[N][N],d[N][N];
int main(){
	scanf("%d\n",&T);
	while(T--){
		scanf("%d%d%d%d\n",&n,&m,&a,&b);
		for(i=1;i<=n;i++){
			for(j=1;j<=m;j++) c[i][j]=getchar();
			scanf("\n");
		}
		for(i=1;i<=a;i++){
			for(j=1;j<=b;j++) d[i][j]=getchar();
			scanf("\n");
		}
		x=0;
		for(i=1;i<=n;i++){
			for(j=1;j<=m;j++){
				if(c[i][j]=='1'){
					int s=(i-1)*m+j;
					f[x]=s;g[s]=x;x=s;
				}
			}
		}
		f[x]=t=0;
		for(i=1;i<=a;i++){
			for(j=1;j<=b;j++){
				if(d[i][j]=='1'){
					if(!t){
						h[++t][0]=i;
						h[t][1]=j;	
					}
					else{
						h[++t][0]=i-h[1][0];
						h[t][1]=j-h[1][1];
					}
				}
			}
		}
		x=f[0];bz=1;
		while(x>0){
			i=x/m+(x%m>0);
			j=x-(i-1)*m;
			c[i][j]='0';
			for(k=2;k<=t;k++){
				int xx=i+h[k][0],yy=j+h[k][1],z=(xx-1)*m+yy;
				if(c[xx][yy]=='0'){
					bz=0;break;
				}
				c[xx][yy]='0';
				f[g[z]]=f[z];
				g[f[z]]=g[z];
			}
			if(!bz) break;
			x=f[x];
		}
		if(bz) printf("YES\n");else printf("NO\n");
	}
	return 0;
}

Code2 

#include <bits/stdc++.h>
#define fi first
#define se second
#define mp make_pair
#define fo(i,a,b) for(i=a;i<=b;i++)
using namespace std;
typedef pair<int,int> P;

const int N=1100;
int T,i,j,k,n,m,a,b,tot,x,y,xx,yy;
char d[N][N],t[N][N];
P o[N*N];

int main()
{
	for(scanf("%d",&T);T--;)
	{
		scanf("%d%d%d%d",&n,&m,&a,&b);
		fo(i,1,n) scanf("%s",d[i]+1);
		tot=0;
		fo(i,1,a)
		{
			scanf("%s",t[i]+1);
			fo(j,1,b) if(t[i][j]=='1') o[++tot]=mp(i,j);
		}
		fo(i,1,n)
			fo(j,1,m)
				if(d[i][j]=='1')
				{
					d[i][j]='0';
					x=i-o[1].fi, y=j-o[1].se;
					fo(k,2,tot)
					{
						xx=x+o[k].fi, yy=y+o[k].se;
						if(xx<1||xx>n||yy<1||yy>m||d[xx][yy]=='0') {puts("NO"); goto endl;}
						d[xx][yy]='0';
					}
				}
		puts("YES");
		endl:;
	}
}


作者:zsjzliziyang 
QQ:1634151125 
转载及修改请注明 
本文地址: https://blog.csdn.net/zsjzliziyang/article/details/86514781

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值