首页 > 其他分享 >P1380 T型骨牌 题解

P1380 T型骨牌 题解

时间:2024-02-17 21:01:34浏览次数:20  
标签:sy sx int 题解 sum P1380 骨牌 vx vy

本题每个位置有 $5$ 种可能,据题中 $n,m$ 均小于五,所以可以用搜索直接过。

上代码

#include<cstdio>
using namespace std;
bool mp[15][15];
int n, m, ans;
int dt[4][5][2] = {{{-1, -1}, {0, -1}, {1, -1}, {0, 0}, {0, 1}}, {{-1, 0}, {0, 0}, {1, -1}, {1, 0}, {1, 1}}, {{0, -1}, {0, 0}, {-1, 1}, {0, 1}, {1, 1}}, {{-1, 1}, {-1, 0}, {-1, -1}, {0, 0}, {1, 0}}};
void dfs(int sx, int sy, int sum)
{
		
	if (sx == n)
	{
		if (sum > ans)
			ans = sum;
		return ;
	}
	for (int i = 0;i < 4; i++)
	{
		bool flag = 1;
		for (int j = 0;j < 5; j++)
		{
			int vx = sx + dt[i][j][0], vy = sy + dt[i][j][1];
			if (mp[vx][vy] || vx < 1 || vx > n || vy < 1 || vy > m)
			{
				flag = 0;
				break;
			}
		}
		if (!flag)
			continue;
		for (int j = 0;j < 5; j++)
		{
			int vx = sx + dt[i][j][0], vy = sy + dt[i][j][1];
			mp[vx][vy] = 1;
		}
		if (sy == m - 1)
			dfs(sx + 1, 2, sum + 1);
		else
			dfs(sx, sy + 1, sum + 1);
		for (int j = 0;j < 5; j++)
		{
			int vx = sx + dt[i][j][0], vy = sy + dt[i][j][1];
			mp[vx][vy] = 0;
		}
	}
	if (sy == m - 1)
		dfs(sx + 1, 2, sum);
	else
		dfs(sx, sy + 1, sum);
}
int main()
{
	scanf ("%d%d", &n, &m);
	if (n < 3 || m < 3)
	{
		puts("0");
		return 0;
	}
	dfs(2, 2, 0);
	printf ("%d", ans);
	return 0;
}

这样交上去后会有 $3$ 点是 $T$ 的。

重审代码,发现在如下代码可以进行剪枝。

		if (sy == m - 1)
			dfs(sx + 1, 2, sum + 1);
		else
			dfs(sx, sy + 1, sum + 1);

思考后发现只有现在的 $sum + 1$ 大于之前的值时,才有可能得到更大值。

AC代码便应运而生。

#include<cstdio>
using namespace std;
bool mp[15][15];
int dmp[15][15];//dmp[i][j] 用来记载上次以 i,j 为中心方 T 的最大值。
int n, m, ans;
int dt[4][5][2] = {{{-1, -1}, {0, -1}, {1, -1}, {0, 0}, {0, 1}}, {{-1, 0}, {0, 0}, {1, -1}, {1, 0}, {1, 1}}, {{0, -1}, {0, 0}, {-1, 1}, {0, 1}, {1, 1}}, {{-1, 1}, {-1, 0}, {-1, -1}, {0, 0}, {1, 0}}};
//dt[i][j][k] 表示第 i 种方法的第 j 个点和中心点的坐标差, k = 0 代表行, k = 1 代表列。
void dfs(int sx, int sy, int sum)
{
	if (sx == n)
	{
		if (sum > ans)
			ans = sum;
		return ;
	}
	for (int i = 0;i < 4; i++)
	//枚举4种放法
	{
		bool flag = 1;
		for (int j = 0;j < 5; j++)
		//判断是否能放
		{
			int vx = sx + dt[i][j][0], vy = sy + dt[i][j][1];
			if (mp[vx][vy] || vx < 1 || vx > n || vy < 1 || vy > m)
			{
				flag = 0;
				break;
			}
		}
		if (!flag)
			continue;
		for (int j = 0;j < 5; j++)
		//标记
		{
			int vx = sx + dt[i][j][0], vy = sy + dt[i][j][1];
			mp[vx][vy] = 1;
		}
		if (sum + 1 >= dmp[sx][sy])
		{
			if (sum > dmp[sx][sy])
				dmp[sx][sy] = sum;
			if (sy == m - 1)
				dfs(sx + 1, 2, sum + 1);
			else
				dfs(sx, sy + 1, sum + 1);
		}
		for (int j = 0;j < 5; j++)
		//回溯
		{
			int vx = sx + dt[i][j][0], vy = sy + dt[i][j][1];
			mp[vx][vy] = 0;
		}
	}
	if (sy == m - 1)
		dfs(sx + 1, 2, sum);
	else
		dfs(sx, sy + 1, sum);
}
int main()
{
	scanf ("%d%d", &n, &m);
	if (n < 3 || m < 3)
	{
		puts("0");
		return 0;
	}
	dfs(2, 2, 0);
	printf ("%d", ans);
	return 0;
}

有错误欢迎私信指出

标签:sy,sx,int,题解,sum,P1380,骨牌,vx,vy
From: https://www.cnblogs.com/Assassins-Creed/p/18018395

相关文章

  • P1686 挑战 题解
    本题就是要找到最短的捷径。注意事项:捷径必须是直线。要求捷径最短而非总路程最短。捷径不与原有的路重合既然在同一直线上,则该捷径的起点与终点的横坐标或纵坐标相等。要把横坐标或纵坐标相同的聚在一起只需要排个序即可。捷径最短的话(以横坐标相等举例),只需要以$x$为第......
  • P1541 [NOIP2010 提高组] 乌龟棋题解
    有两种方法,代码注释都很详细了直接上代码一:记忆化搜索#include<bits/stdc++.h>usingnamespacestd;intt[15];intn,m;inta[400];intmp[45][45][45][45];//mp[i][j][k][l]表示1号用i张,2号用j张,3号用k张,4号用l张的情况下,最多能拿多少分intdfs(intstep,intw)//step......
  • TopCoder SRM478C RandomApple 题解
    题意:有\(k\)种苹果和\(n\)个箱子,每个箱子中有一些苹果,先等概率选取\(n\)个箱子组成集合的非空子集,再从选出的苹果中随机选一个,问每种苹果被选中的概率是多少箱子\(i\)有\(a_{i,j}\)个第\(j\)种苹果,第\(i\)个箱子的总苹果数\(siz_i=\sum\limits_{j=1}^ka_{i,j}\),苹果总数\(sum=\su......
  • 无限酒店 题解
    题目链接由于间隔不变,对于下面\(3\)个操作,只需记录起始位置与间隔即可。对于无数人到达酒店:所有位置的起始点\(\times2\),间隔\(\times2\),新的团队起始点为\(1\),间隔为\(2\)。对于\(k\)个人到达酒店:所有点的起始点\(+k\),间隔不变,新的团队起始点为\(0\),间隔为\(1\)......
  • CF1365G Secure Password 题解
    Description本题是交互题。有一个固定的数组\(A\),同时通过数组\(A\)构造出数组\(P\),具体来讲,\(P_i\)是\(A\)中除\(A_i\)外的所有元素的按位或。你需要在最多\(13\)次询问中得到最后的\(P\)数组。\(2\leqn\leq1000\)。Solution首先有一个\(2\logn\)的是注......
  • 传纸条 题解
    题目描述小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题。一次素质拓展活动中,班上同学安排做成一个m行n列的矩阵,而小渊和小轩被安排在矩阵对角线的两端,因此,他们就无法直接交谈了。幸运的是,他们可以通过传纸条来进行交流。纸条要经由许多同学传到对方手里,小渊坐在矩阵......
  • 2024.2.17模拟赛T1题解
    先考虑\(q=(1...n)\)的情况:发现如果设\(divcnt(p)\)表示将\(p\)划分为极小值域连续段的个数,那满足\(divcnt(p)\gem\)的排列都是合法的。那现在要求出有多少个排列符合条件可以先算出长度为\(i\),\(divnct\)为\(1\)的排列个数,这可以用dp解决然后再背包一下,就能求......
  • P2042 [NOI2005] 维护数列 题解
    题目链接:维护数列比较不好码的题,首先确保自己会一种文艺平衡树的书写,这点因人而异,比较推荐初学者学\(fhq\)平衡树,坑比较少,比较好码,写平衡树合并、持久化类的题时,也比较好写。注意到空间需求比较大,还涉及删除,我们的删除用各种树类数据结构中最常用的回收标记用于新的节点开辟。......
  • CF1929E 题解
    题意:给定一棵\(n\)个节点数和\(k\)条路径\((a_i,b_i)\),求至少将多少条边染色,使得给定路径都至少有一条染色的边。\(n\le10^5,k\le20\)。思路:好题。显然状压\(dp\),\(dp[S]\)表示至少染多少条边使得\(S\)中的路径都满足条件。正常思路是枚举子集转移,考虑\(T\s......
  • 整数划分 题解
    题目描述如何把一个正整数N(N长度<20)划分为M(M>1)个部分,使这M个部分的乘积最大。N、M从键盘输入,输出最大值及一种划分方式。输入格式第一行一个正整数T(T<=10000),表示有T组数据。接下来T行每行两个正整数N,M。输出格式对于每组数据第一行输出最大值。第二行输出划分方案,将N按......