ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

JOI2016 サンドイッチ

2020-11-19 19:32:43  阅读:303  来源: 互联网

标签:吃掉 JOI2016 三明治 int 直角 对角线 include


题目链接

题意简述

\(JOI\)参加\(IOI\)联谊会,会场有一张桌子,桌子上有\(R\times C\)个三明治被摆成\(R\)行\(C\)列,每个三明治被沿着主对角线或副对角线切成两个小三明治。

\(JOI\)要吃三明治,他会按照一定的规则吃三明治,如果一个小三明治同时满足以下两种条件,他就不会吃掉那个小三明治:

  1. 与该小三明治在同一个三明治中的另一个三明治没有被吃掉。
  2. 与该小三明治两条直角边相邻的另外两个小三明治中有一个没有被吃掉。

现在他想知道,他吃掉每一个三明治时,最少已经吃掉了多少个小三明治?

如果吃不到,输出$ -1$

题目解析

这道题想得我好饿啊

首先,一对小三明治肯定是连续拿走的,因为不连续拿走没有意义,这样做既没有暴露一条新的直角边出来让\(JOI\)君可以再吃一个,也不会对答案有任何优化。所以可以以一个大三明治为单位讨论问题。

考虑到吃掉一个三明治之后会让另一个三明治的一条直角边暴露出来,为方便叙述,设吃掉\(i\)之后会暴露\(j\)的一条直角边。如果我们钦定先吃\(j\)的吃了\(i\)之后暴露出来的直角边所属的那一半小三明治(好绕),容易发现只有吃掉\(i\)之后,\(j\)才能被吃掉。可以枚举每个三明治先吃哪一半,然后用\(O(n^2)\)的复杂度爆搜算答案,总复杂度\(O(n^4)\).

但是可以根据这个图的一些特殊性质进行优化:由于是网格图,要吃掉一个三明治,那么这个三明治的左右两边的三明治至少有一个会被吃掉。还是钦定某一半三明治先被吃掉,假设是左半三明治先被吃掉,那么左边三明治的左半三明治也先被吃掉,因为如果左边三明治的右半三明治先被吃掉的话,那么当前三明治要先被吃掉才行,这样就有环了啊(即:我要吃掉我自己,必须要先吃掉我自己)所以从左往右搜一遍,信息可以不用清空,复杂度降了一维,\(O(n^3)\)可以接受。


►Code View

#include<cstdio>
#include<algorithm>
#include<cstring>
#include<vector>
#include<queue>
using namespace std;
#define LL long long
#define N 405
#define INF 0x3f3f3f3f
int rd()
{
	int x=0,f=1;char c=getchar();
	while(c<'0'||c>'9'){if(c=='-')f=-1; c=getchar();}
	while(c>='0'&&c<='9'){x=(x<<3)+(x<<1)+(c^48); c=getchar();}
	return f*x;
}
int n,m;
char s[N][N];
int ans[N][N],tot,vis[N][N];
const int dx[]={0,-1,0,1},dy[]={-1,0,1,0};//左上右下
bool check(int x,int y)
{
	if(x<1||x>n||y<1||y>m) return 0;
	return 1;
}
void dfs(int x,int y,int d)
{
	if(!check(x,y)) return ;
	if(vis[x][y])
	{
		if(vis[x][y]==-1) tot=INF;//成环 
		return ;
	}
	tot+=2;
	vis[x][y]=-1;
	dfs(x+dx[d],y+dy[d],d);
	if(s[x][y]=='N') d^=3;
	else d^=1;
	/*
	这里方向的转换写得很妙啊 
	0^3=3 要吃掉左半边 主对角线 要先吃左边的三明治 还必须吃掉下面的三明治
	0^3=1 要吃掉左半边 副对角线 要先吃左边的三明治 还必须吃掉上面的三明治
	2^3=1 要吃掉右半边 主对角线 要先吃右边的三明治 还必须吃掉上面的三明治
	2^1=3 要吃掉右半边 副对角线 要先吃右边的三明治 还必须吃掉下面的三明治
	上下转换同理 
	*/
	dfs(x+dx[d],y+dy[d],d);
	vis[x][y]=1; 
}
int main()
{
	n=rd(),m=rd();
	for(int i=1;i<=n;i++)
		scanf("%s",s[i]+1);
	memset(ans,INF,sizeof(ans));
	for(int i=1;i<=n;i++)
	{
		memset(vis,0,sizeof(vis));
		tot=0;
		for(int j=1;j<=m;j++)
		{
			dfs(i,j,0);
			ans[i][j]=min(ans[i][j],tot);
		}
		memset(vis,0,sizeof(vis));
		tot=0;
		for(int j=m;j>=1;j--)
		{
			dfs(i,j,2);
			ans[i][j]=min(ans[i][j],tot);
		}
		for(int j=1;j<=m;j++)
			printf("%d ",ans[i][j]==INF?-1:ans[i][j]);
		puts("");
	}
	return 0;
}
碎碎念

不知道为啥\(LOJ\)可以过,但是某谷不行...

而且发现某谷把我的题解拒绝掉了,理由是排版不好看...呃,这算啥理由,难道是因为我顺带吐槽了一下某谷?

标签:吃掉,JOI2016,三明治,int,直角,对角线,include
来源: https://www.cnblogs.com/lyttt/p/14007435.html

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有