洛谷4055 [JSOI2009]游戏(二分图博弈)
例题:在N×M的迷宫中有一个棋子,小 AA 首先任意选择棋子放置的位置。然后,小 YY 和小 AA 轮流将棋子移动到相邻的格子里。游戏的规则规定,在一次游戏中,同一个格子不能进入两次,且不能将棋子移动到某些格子中去。当玩家无法继续移动棋子时,游戏结束,最后一个移动棋子的玩家赢得了游戏。
输入格式:输入数据首先输入两个整数 N,M,表示了迷宫的边长。接下来 N行,每行 M个字符,描述了迷宫。
输出格式:若小 AA 能够赢得游戏,则输出一行 WIN,然后输出所有可以赢得游戏的起始位置,按行优先顺序输出,每行一个。否则输出一行 LOSE
分析:发现每次移动一定是从一个黑点到达一个白点,或者反过来。所以可以对于棋盘进行染色然后连边。
考虑一下必胜策略。如果选择从一个匹配点开始走,另外一个人沿着匹配点走,那么就输了,因为匹配点不一定有出边了。如果选择从一个非匹配点开始走,另外一个人无论怎么走都只能走到一个匹配点(或者无路可走),因为如果另外一个人可以走到一个非匹配点,意味着这两个点可以匹配,所以不存在这种情况。那么,先手只需要沿着匹配边走就一定能够做到必胜。所以黑白染色之后连边,找到所有不一定在最大匹配中的点就好了。跑网络流就会快很多。但是怎么判断一个点是否在不一定在最大匹配中呢?把所有和S相连的(还有剩余流量的边连接的)点全部扣下来,这些点一定满足条件。为什么呢?首先不在当前的这个匹配中的点一定会被计算。如果一个点在匹配中,但是他被连上了,那么一定是通过一个没有被匹配上的点,到达当前点的匹配点,在连回来的,这样子意味着可以交换匹配。和T相连的点同理解决即可。
#include<cstdio>
#include<cstring>
#include<queue>
using namespace std;
const int M = 105;
const int N = 1e4 + 50; typedef long long ll; int S, T, cnt = 1, maxn, tot, dn, dm, ans;
int dep[N], head[N], temp[N], id[M][M], kk[4][2] = {{-1, 0}, {1, 0}, {0, -1}, {0, 1}};
char s[M][M];
bool vis[N], is[N], color[N]; struct Edge{
int nex, to, val;
}e[N << 3]; inline int min(int a, int b) { return a < b ? a : b; }
inline void add(int a, int b, int c){
e[++cnt] = {head[a], b, c}, head[a] = cnt;
e[++cnt] = {head[b], a, 0}, head[b] = cnt;
} bool bfs(){
memset(dep, 0, sizeof(dep));
queue<int> q;
q.push(S); dep[S] = 1;
while(!q.empty()){
int u = q.front(); q.pop();
for(int i = head[u]; i; i = e[i].nex){
int to = e[i].to;
if(!dep[to] && e[i].val){
dep[to] = dep[u]+1;
q.push(to);
}
}
}
return dep[T];
} ll dfs(int u, int flow){
if(u == T) return flow;
ll out = 0, k;
for(int i = temp[u]; i && flow; i = e[i].nex){
temp[u] = i;
if(!e[i].val) continue;
int to = e[i].to;
if(dep[to] == dep[u]+1 && (k = dfs(to,min(flow,e[i].val)))){
e[i].val -= k;
e[i^1].val += k;
flow -= k; out += k;
}
}
if(!out) dep[u] = 0;
return out;
} void Dinic(){
while(bfs()){
for(int i = S; i <= T; ++i) temp[i] = head[i];
maxn += dfs(S, 1e9);
}
} void DFS(int u, int k){
vis[u] = 1;
if(color[u] == k) ++ans, is[u] = 1;
for(int i = head[u]; i; i = e[i].nex){
int to = e[i].to;
if(e[i].val == k && !vis[to]) DFS(to, k);
}
} int main(){
scanf("%d%d", &dn, &dm);
for(int i = 1; i <= dn; ++i) scanf("%s", s[i] + 1);
for(int i = 1; i <= dn; ++i)
for(int t = 1; t <= dm; ++t)
if(s[i][t] != '#') id[i][t] = ++tot;
T = tot + 1;
for(int i = 1; i <= dn; ++i)
for(int t = 1; t <= dm; ++t)
if(s[i][t] == '.'){
if(((i + t) & 1)){
add(S, id[i][t], 1), color[id[i][t]] = 1;
for(int j = 0; j < 4; ++j){
int dx = i + kk[j][0], dy = t + kk[j][1];
if(dx >= 1 && dx <= dn && dy >= 1 && dy <= dm)
add(id[i][t], id[dx][dy], 1);
}
}
else add(id[i][t], T, 1);
}
Dinic(); DFS(S, 1); memset(vis, 0, sizeof(vis)); DFS(T, 0);
if(ans){
puts("WIN");
for(int i = 1; i <= dn; ++i)
for(int t = 1; t <= dm; ++t)
if(is[id[i][t]]) printf("%d %d\n", i, t);
}
else puts("LOSE");
return 0;
}
洛谷4055 [JSOI2009]游戏(二分图博弈)的更多相关文章
- [JSOI2009]游戏 二分图博弈
		题面 题面 题解 二分图博弈的模板题,只要会二分图博弈就可以做了,可以当做板子打. 根据二分图博弈,如果一个点x在某种方案中不属于最大匹配,那么这是一个先手必败点. 因为对方先手,因此我们就是要找这样 ... 
- 洛谷 P2197 nim游戏
		洛谷 P2197 nim游戏 题目描述 甲,乙两个人玩Nim取石子游戏. nim游戏的规则是这样的:地上有n堆石子(每堆石子数量小于10000),每人每次可从任意一堆石子里取出任意多枚石子扔掉,可以取 ... 
- 洛谷 P1965 转圈游戏
		洛谷 P1965 转圈游戏 传送门 思路 每一轮第 0 号位置上的小伙伴顺时针走到第 m 号位置,第 1 号位置小伙伴走到第 m+1 号位置,--,依此类推,第n − m号位置上的小伙伴走到第 0 号 ... 
- 洛谷 P1000 超级玛丽游戏
		P1000 超级玛丽游戏 题目背景 本题是洛谷的试机题目,可以帮助了解洛谷的使用. 建议完成本题目后继续尝试P1001.P1008. 题目描述 超级玛丽是一个非常经典的游戏.请你用字符画的形式输出超级 ... 
- 【流水调度问题】【邻项交换对比】【Johnson法则】洛谷P1080国王游戏/P1248加工生产调度/P2123皇后游戏/P1541爬山
		前提说明,因为我比较菜,关于理论性的证明大部分是搬来其他大佬的,相应地方有注明. 我自己写的部分换颜色来便于区分. 邻项交换对比是求一定条件下的最优排序的思想(个人理解).这部分最近做了一些题,就一起 ... 
- $loj10156/$洛谷$2016$ 战略游戏 树形$DP$
		洛谷loj Desription Bob 喜欢玩电脑游戏,特别是战略游戏.但是他经常无法找到快速玩过游戏的方法.现在他有个问题. 现在他有座古城堡,古城堡的路形成一棵树.他要在这棵树的节点上放置最少数 ... 
- 洛谷P1000 超级玛丽游戏(洛谷新手村1-1-1)
		题目背景 本题是洛谷的试机题目,可以帮助了解洛谷的使用. 建议完成本题目后继续尝试P1001.P1008. 题目描述 超级玛丽是一个非常经典的游戏.请你用字符画的形式输出超级玛丽中的一个场景. *** ... 
- 洛谷P1080 国王游戏 python解法 - 高精 贪心 排序
		洛谷的题目实在是裹脚布 还编的像童话 这题要 "使得获得奖赏最多的大臣,所获奖赏尽可能的少." 看了半天都觉得不像人话 总算理解后 简单说题目的意思就是 根据既定的运算规则 如何排 ... 
- 洛谷P1290欧几里德游戏
		题目地址 题目大意: 两个人st和ol博弈 有两个整数n,m 每次轮到一个人时候,需要选择用大的那个数减去小的那个数的倍数(不能减为负数) 最后得到0的为胜利者 思路: (以下讨论均在n<m的条 ... 
- 洛谷1640 bzoj1854游戏 匈牙利就是又短又快
		bzoj炸了,靠离线版题目做了两道(过过样例什么的还是轻松的)但是交不了,正巧洛谷有个"大牛分站",就转回洛谷做题了 水题先行,一道傻逼匈牙利 其实本来的思路是搜索然后发现写出来类 ... 
随机推荐
- C# OpenCVSharp图像入门_给绿幕图片视频加背景
			OpenCV的全称是Open Source Computer Vision Library,是一个跨平台的计算机视觉库.OpenCV是由英特尔公司发起并参与开发,以BSD许可证授权发行,可以在商业和研 ... 
- C++欧几里得算法求最大公约数和最小公倍数
			定义 最大公约数即为 Greatest Common Divisor,常缩写为 gcd. 一组整数的公约数,是指同时是这组数中每一个数的约数的数. 一组整数的最大公约数,是指所有公约数里面最大的一个. ... 
- 基于C#的无边框窗体阴影绘制方案 - 开源研究系列文章
			今天介绍无边框窗体阴影绘制的内容. 上次有介绍使用双窗体的方法来显示阴影,这次介绍使用API函数来进行绘制.这里使用的是Windows API函数,操作系统的窗体也是用的这个来进行的绘制. 1. 项目 ... 
- docker网络 bridge 与overlay 模式
			转载请注明出处: 1.bridge网络模式 工作原理: 在Bridge模式中,Docker通过创建一个虚拟网络桥接器(bridge)将容器连接到主机上的物理网络接口.每个容器都会被分配一个IP地址, ... 
- wineqq中接收文件的查看与移动
			在Ubuntu等linux系统中安装QQ都需要安装wine支持,而在使用时,会遇到qq接收到的文件无法直接进行操作等问题. 这时,我们发现直接对文件进行复制后,无法在Ubuntu目录中进行粘贴. 其实 ... 
- 关于 LLM 和图数据库、知识图谱的那些事
			本文整理自 NebulaGraph 布道师 wey 在「夜谈 LLM」主题分享上的演讲,主要包括以下内容: 背景 LLM RAG Graph 知识抽取 Text2Cypher Graph RAG 未来 ... 
- 一行命令即可启动 Walrus丨入门教程
			应用管理平台 Walrus 已正式开源,本文将介绍如何上手安装 Walrus 以及如何借助 Walrus 进行应用部署. 开源地址:https://github.com/seal-io/walrus ... 
- Unity TextMeshPro 添加中文字体遇见的问题以及解决方案
			前言 按标准官方教程为 Unity TextMeshPro 添加中文字体时出现了各种奇奇怪怪的问题,于是有了这篇随笔. 中文字体解决方案 以下步骤适用于 TextMeshPro 3.0.6. 字符数量 ... 
- Android13深入了解 Android 小窗口模式和窗口类型
			Android13深入了解 Android 小窗口模式和窗口类型 小窗模式,作为一种在移动设备上的多任务处理方式,为用户带来了便捷和高效的体验,尤其在一些特定场景下,其价值愈发凸显.以下是为什么需要小 ... 
- k8s 入门到实战--部署应用到 k8s
			背景 最近这这段时间更新了一些 k8s 相关的博客和视频,也收到了一些反馈:大概分为这几类: 公司已经经历过服务化改造了,但还未接触过云原生. 公司部分应用进行了云原生改造,但大部分工作是由基础架构和 ... 
