HDOJ-2258 Continuous Same Game (1) dfs

12 篇文章 0 订阅

题目链接:here

题意:

        坑人啊!!我被英文给坑人。。写了两天!!才看出是题意理解错了!!弄明白之后果断AC。。。

        题意是这样的,就是这种游戏。。大家都玩过。一个n*m的方格,1-5代表5种颜色的小格子。碰到2个,或者2个以上在一起的小格子就自动消掉。。这个人比较笨,只会一个一个的消掉,先消掉数量最大的,当数量相同的时候,取i小的那个,i 相同就取  j 小的那个,当下面的没有了,就下移。。一整竖排没有的时候,左边的右移(我一开始以为左移!!)。。。然后每次消掉 n 个连续的,得分就是  n*(n+1)。。。算最后的了多少分。

分析:

        先 dfs 那些一样格子的数量,然后消掉。。最后判断需不需要下移或者右移。。然后搞定。

代码:

#include <iostream>
#include <cstdio>
#include <cstring>
using namespace std;

const int maxn = 22;
int dir[4][2] = {{1,0},{-1,0},{0,1},{0,-1}};

int n, m;
char map[maxn][maxn];
int vis[maxn][maxn];	//标记每次是否访问,并且访问的是第几个块
int maxnum[maxn*maxn];	//记录每个标记的块数
int len;
int s;		//左边第几排开始

void dfs(int x, int y, char ch, int num)
{
	int nx, ny;
	for (int i=0; i<4; i++)
	{
		nx = x + dir[i][0];
		ny = y + dir[i][1];
		if (map[nx][ny] == '0' || map[nx][ny] != ch || vis[nx][ny]) continue;
		vis[nx][ny] = num;
		maxnum[num] ++;
		dfs(nx, ny, ch, num);
	}
}

int main()
{
	while (scanf("%d %d", &n, &m) != EOF)
	{
		memset(map, '0', sizeof(map));
		getchar();
		s = 1;
		int i, j;
		for (i=1; i<=n; i++)
		{
			for (j=1; j<=m; j++)
			{
				char c = getchar();
				map[i][j] = c;
			}
			getchar();
		}
		int ans = 0;
		while (true)
		{
			memset(vis, 0, sizeof(vis));
			memset(maxnum, 0, sizeof(maxnum));
			len = 1;

			for (i=1; i<=n; i++)
				for (j=s; j<=m; j++)
					if (vis[i][j] == 0 && map[i][j] != '0')
					{
						vis[i][j] = len;
						maxnum[len] ++;
						dfs(i, j, map[i][j], len);
						len ++;
					}
			int Max = 0, iMax = 0;
			for (i=1; i<len; i++)
				if (maxnum[i] > Max)
					Max = maxnum[i], iMax = i;
			if (Max <= 1) break;
			ans += Max * Max - Max;
			for (i=1; i<=n; i++)
				for (j=s; j<=m; j++)
					if (vis[i][j] == iMax)
						for (int k=i; map[k][j] != '0'; k--)
							map[k][j] = map[k-1][j];
			bool flag = true;
			while (flag)
			{
				int mark = 0;
				for (int j=m; j>=s; j--)
				{
					flag = true;
					for (int i=1; i<=n; i++)
						if (map[i][j] != '0')
							flag = false;
					if (flag)
					{
						mark = j;
						break;
					}
				}
				if (flag)
				{
					for (i=1; i<=n; i++)
						for (int k=mark; k >= s; k--)
							map[i][k] = map[i][k-1];
					s++;
				}
				if (m < 1 || s >= m) flag = false;
			}
		}
		printf("%d\n", ans);
	}
	return 0;
}

样例输入:

8 8
11223344
55112233
44551122
33445511
22334455
11223344
55112233
44551122

9 9
111222333
444555111
222333444
555111222
333444555
111222333
444555111
222333444
555111222

5 5
12345
54321
11122
22333
12345

6 6
112233
222333
444441
221111
555222
222211

8 8
12345432
12345321
32145231
13542311
23541351
13135421
31354212
31354132

样例输出:

64
162
44
122
120




  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值