[CF1105D]Kilani and the Game

[CF1105D]Kilani and the Game

题目大意:给出一个$n\times m(n,m\leqslant10^3)$的地图,有$k(k\leqslant9)$个玩家,第$i$个玩家速度为$s_i$。地图中$\#$代表障碍;$.$ 代表空地;数字代表是一名编号为此数字的玩家的城堡。每个玩家按编号轮流操作,每次操作把自己城堡周围$s_i$格内的空地变成自己城堡。直到没有玩家能操作为止。输出每名玩家城堡的个数。

题解:$bfs$,显然发现每个点只会扩展一次,用$k$个$queue$保存每个人的可扩展的城堡,模拟扩展即可,$s_i$可以每次扩展一格,扩展$s_i$次来解决。复杂度$O(nm)$

卡点:

 

C++ Code:

#include <algorithm>
#include <cctype>
#include <cstdio>
#include <queue>
#define maxn 1005
#define maxk 10
const int D[2][4] = {
    {1, 0, -1, 0}, {0, 1, 0, -1}};

struct Point {
	int x, y;
	Point() { }
	Point(int __x, int __y) : x(__x), y(__y) { }
} ;
std::queue<Point> q[maxk];

bool used[maxn][maxn], Continue[maxk];
int n, m, k, len[maxk], ans[maxk];

inline bool over_range(int x, int y) {
	return x < 1 || x > n || y < 1 || y > m || used[x][y];
}

int main() {
	scanf("%d%d%d", &n, &m, &k);
	for (int i = 0; i < k; ++i) scanf("%d", len + i);
	for (int i = 1; i <= n; ++i) {
		static char s[maxn];
		scanf("%s", s + 1);
		for (int j = 1; j <= m; ++j) if (s[j] != '.') {
			used[i][j] = true;
			if (isdigit(s[j])) {
				int pos = (s[j] & 15) - 1;
				++ans[pos];
				q[pos].push(Point(i, j));
			}
		}
	}
	for (int now = 0, num = k; num; now += 1 - k, now += now >> 31 & k) {
		static std::queue<Point> Q;
		std::queue<Point> &q = ::q[now];
		for (int Tim = len[now]; Tim; --Tim) {
			if (q.empty()) {
				num -= !Continue[now];
				Continue[now] = true;
				break;
			}
			while (!q.empty()) {
				Point u = q.front(); q.pop();
				for (int i = 0, x, y; i < 4; ++i) {
					x = u.x + D[0][i], y = u.y + D[1][i];
					if (!over_range(x, y)) {
						++ans[now];
						used[x][y] = true;
						Q.push(Point(x, y));
					}
				}
			}
			std::swap(q, Q);
		}
	}
	for (int i = 0; i < k; ++i) printf("%d ", ans[i]); puts("");
	return 0;
}

  

转载于:https://www.cnblogs.com/Memory-of-winter/p/10350986.html

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/101137.html原文链接:https://javaforall.net

(0)
上一篇 2021年6月29日 下午7:00
下一篇 2021年6月29日 下午8:00


相关推荐

  • Ubuntu中的dpkg命令「建议收藏」

    Ubuntu中的dpkg命令「建议收藏」Linux命令学习系列之dpkg命令详解   普通dpkg用法   dpkg-i安装一个Debian包裹文件,如你手动下载的文件。   dpkg-c列出的内容。   dpkg-I从中提取包裹信息。   dpkg-r移除一个已安装的包裹。   dpkg-L列出安装的所有文件清单。同时请看dpkg-c来检查一个.deb文件的

    2022年5月22日
    38
  • wireshark分析ICMP数据包

    wireshark分析ICMP数据包1 理解 ICMP 数据包的内容 1 类型 8 位字段 ICMP 报文类型 2 代码 8 位字段 记录发送特定类型的 ICMP 报文的原因 3 检验和 16 位字段 4 首部其余部分 32 位字段 5 数据部分 包括差错报文的数据部分携带的信息可以找出引起差错的原始报文 查询报文的数据部分携带了基于查询数据的额外信息 2 tracetwww baidu com 分析出错的数据包的源地址和目标地

    2026年3月19日
    2
  • centos开启关闭防火墙(电脑防火墙关闭有什么影响)

    CentOS6:1)永久性生效,重启后不会复原开启:chkconfigiptableson关闭:chkconfigiptablesoff 2)即时生效,重启后复原service iptables status    查看防火墙状态service iptables start     开启防火墙service iptab…

    2022年4月13日
    57
  • win10自动更新有效强制永久关闭怎么办_win10怎么不自动更新

    win10自动更新有效强制永久关闭怎么办_win10怎么不自动更新网上的一些Win10彻底关闭WindowsUpdate自动更新的方法,主要是通过一些如设置流量计费或借助一些专门的小工具来实现,比如360来限速,但往往会发现,Win10自动更新就像打不死的小强,不管怎么关闭,之后还是会自动更新,让不少小伙伴颇为不爽。今天为大家带来了这篇教程,通过服务、注册表、组策略、计划任务中,全方位设置,彻底关闭Win10自动更新,感兴趣的小伙伴不妨试试吧。服务中关闭Wi…

    2025年6月16日
    5
  • easyui textbox 密码框的prompt提示语显示*号问题的解决办法

    easyui textbox 密码框的prompt提示语显示*号问题的解决办法easyuitextbox密码框的prompt提示语显示*号问题的解决办法

    2022年7月25日
    23
  • windows10命令行进入指定目录_命令行返回上一级目录

    windows10命令行进入指定目录_命令行返回上一级目录假定我们想进入指定目录E:\Software\apache-tomcat\apache-tomcat\bin在命令行输入cdE:\Software\apache-tomcat\apache-tomcat\bin+回车发现……啊哦没反应改为先输入E:+回车再输入cdSoftware\apache-tomcat\apache-tomcat\bin+回车问题解决…

    2022年10月15日
    6

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号