洛谷P1263 || 巴蜀2311 宫廷守卫
题目描述
从前有一个王国,这个王国的城堡是一个矩形,被分为M×N个方格。一些方格是墙,而另一些是空地。这个王国的国王在城堡里设了一些陷阱,每个陷阱占据一块空地。
一天,国王决定在城堡里布置守卫,他希望安排尽量多的守卫。守卫们都是经过严格训练的,所以一旦他们发现同行或同列中有人的话,他们立即向那人射 击。因此,国王希望能够合理地布置守卫,使他们互相之间不能看见,这样他们就不可能互相射击了。守卫们只能被布置在空地上,不能被布置在陷阱或墙上,且一 块空地只能布置一个守卫。如果两个守卫在同一行或同一列,并且他们之间没有墙的话,他们就能互相看见。(守卫就像象棋里的车一样)
你的任务是写一个程序,根据给定的城堡,计算最多可布置多少个守卫,并设计出布置的方案。
输入输出格式
输入格式:
第一行两个整数M和N(1≤M,N≤200),表示城堡的规模。
接下来M行N列的整数,描述的是城堡的地形。第i行j列的数用ai,j表示。
ai,j=0,表示方格[i,j]是一块空地;
ai,j=1,表示方格[i,j]是一个陷阱;
ai,j=2,表示方格[i,j]是墙。
输出格式:
第一行一个整数K,表示最多可布置K个守卫。
此后K行,每行两个整数xi和yi,描述一个守卫的位置。
输入输出样例
3 4
2 0 0 0
2 2 2 1
0 1 0 2
2
1 2
3 3
说明
样例数据如图5-2(黑色方格为墙,白色方格为空地,圆圈为陷阱,G表示守卫)

刚开始的写法是BFS出每个联通块,单独处理。
↑明显不对
后来发现可以把被墙隔开的同一行当成两行处理,列同理。
然后二分图匹配坐标即可。
对二分图的理解还是不透彻啊……
(代码仅提供思路,不保证正确性)(没有SPJ程序)
(巴蜀OJ上只要求输出最多守卫数量,已AC)
/*By SilverN*/
#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<algorithm>
#define LL long long
using namespace std;
const int mxn=;
int read(){
int x=,f=;char ch=getchar();
while(ch<'' || ch>''){if(ch=='-')f=-;ch=getchar();}
while(ch>='' && ch<=''){x=x*+ch-'';ch=getchar();}
return x*f;
}
struct edge{
int v,nxt;
}e[mxn*];
int hd[mxn*],mct=;
void add_edge(int u,int v){
e[++mct].v=v;e[mct].nxt=hd[u];hd[u]=mct;return;
}
int mp[mxn][mxn];
int idx,idy;
int hs[mxn][mxn];
int aidx[mxn*10],aidy[mxn*10];
int link[mxn*10],vis[mxn*10];
//
int n,m,ans=;
bool DFS(int u){
for(int i=hd[u];i;i=e[i].nxt){
int v=e[i].v;
if(!vis[v]){
vis[v]=;
if(link[v]==- || DFS(link[v])){
link[v]=u;
return ;
}
}
}
return ;
}
void solve(){
memset(link,-,sizeof link);
for(int i=;i<=idx;i++){
memset(vis,,sizeof vis);
if(DFS(i))ans++;
}
return;
}
void init(){
int i,j;
bool flag=;
for(i=;i<=m;i++){
flag=;
for(j=;j<=n;j++){
if(mp[i][j]==){flag=;continue;}
if(flag){
++idx;
aidx[idx]=i;
flag=;
}
hs[i][j]=idx;
}
}
idy=idx;
for(j=;j<=n;j++){
flag=;
for(i=;i<=m;i++){
if(mp[i][j]==){flag=;continue;}
if(flag){
++idy;
aidy[idy]=j;
flag=;
}
if(mp[i][j])continue;
add_edge(idy,hs[i][j]);
add_edge(hs[i][j],idy);
// printf("%d %d:%d\n",i,j,idy);
}
}
return;
}
int main(){
m=read();n=read();
int i,j;
for(i=;i<=m;i++)
for(j=;j<=n;j++)
mp[i][j]=read();
init();
solve();
/* for(i=1;i<=m;i++){
for(j=1;j<=n;j++){
printf("%d ",hs[i][j]);
}
printf("\n");
}*/
printf("%d\n",ans);
for(i=idx+;i<=idy;i++){
if(link[i]!=-)printf("%d %d\n",aidx[link[i]],aidy[i]);
}
return ;
}
洛谷P1263 || 巴蜀2311 宫廷守卫的更多相关文章
- 洛谷P1263 宫廷守卫
		P1263 宫廷守卫 题目描述 从前有一个王国,这个王国的城堡是一个矩形,被分为M×N个方格.一些方格是墙,而另一些是空地.这个王国的国王在城堡里设了一些陷阱,每个陷阱占据一块空地. 一天,国王决定在 ... 
- 洛谷 P1263 宫廷守卫
		被这道题折腾了 \(2\) 个小时. 按照题意,每个守卫的上下左右四个方向上应当都是墙,而不能出现其他的守卫. 如图是一个合法的放置方案.每个守卫四个方向上都是墙(包括宫廷外墙). 如图是一个非法的放 ... 
- BZOJ5324 & 洛谷4563 & LOJ2545:[JXOI2018]守卫——题解
		https://www.lydsy.com/JudgeOnline/problem.php?id=5324 https://www.luogu.org/problemnew/show/P4563 ht ... 
- 【题解】宫廷守卫 [P1263]
		[题解]宫廷守卫 [P1263] 传送门:宫廷守卫 \([P1263]\) [题目描述] 给出一个 \(n*m\) 的方格图,分别用整数 \(0,1,2\) 表示空地.陷阱.墙,空地上可以放置守卫,如 ... 
- 洛谷1640 bzoj1854游戏 匈牙利就是又短又快
		bzoj炸了,靠离线版题目做了两道(过过样例什么的还是轻松的)但是交不了,正巧洛谷有个"大牛分站",就转回洛谷做题了 水题先行,一道傻逼匈牙利 其实本来的思路是搜索然后发现写出来类 ... 
- 洛谷P1352  codevs1380 没有上司的舞会——S.B.S.
		没有上司的舞会 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 题目描述 Description Ural大学有N个职员,编号为1~N.他们有 ... 
- 洛谷P1108 低价购买[DP | LIS方案数]
		题目描述 “低价购买”这条建议是在奶牛股票市场取得成功的一半规则.要想被认为是伟大的投资者,你必须遵循以下的问题建议:“低价购买:再低价购买”.每次你购买一支股票,你必须用低于你上次购买它的价格购买它 ... 
- 洛谷 P2701 [USACO5.3]巨大的牛棚Big Barn Label:二维数组前缀和 你够了 这次我用DP
		题目背景 (USACO 5.3.4) 题目描述 农夫约翰想要在他的正方形农场上建造一座正方形大牛棚.他讨厌在他的农场中砍树,想找一个能够让他在空旷无树的地方修建牛棚的地方.我们假定,他的农场划分成 N ... 
- 洛谷P1710 地铁涨价
		P1710 地铁涨价 51通过 339提交 题目提供者洛谷OnlineJudge 标签O2优化云端评测2 难度提高+/省选- 提交 讨论 题解 最新讨论 求教:为什么只有40分 数组大小一定要开够 ... 
随机推荐
- 分层开发MySchool总结
			由于分层之间存在各层之间的关系窗体之间的方法跳转,故有需要者可以进行下载本地文件 MySchool.rar 3304KB 5/22/2016 9:43:28 AM ,代码中有注释, 
- android stuio eclipse映射下的快捷键
			转:关于 android stuio eclipse映射下的快捷键 http://www.cnblogs.com/0616--ataozhijia/p/3870064.html 会持续更新)这边讲的常 ... 
- brainfuck
			/阅读这样的代码就像在强奸你的大脑 #include<stdio.h> #include<ctype.h> #include<stdlib.h> #include ... 
- CSS3中的字体rem
			rem和em都是相对单位,em相对父元素的font-size来计算,而rem是根据文档根元素(html)的font-size大小来计算的 通常将html的字体大小设为62.5%(等于10px),当然也 ... 
- VS 2013 中如何自定义代码片段
			1.菜单 工具->代码段管理器 
- centos一键优化脚本
			centos一键优化脚本:细节:http://oldboy.blog.51cto.com/2561410/1336488网络状态优化:http://oldboy.blog.51cto.com/2561 ... 
- string与stringBuilder的效率与内存占用实测
			using UnityEngine; using System.Diagnostics; using System.Text; using UnityEngine.UI; public class s ... 
- Sql复习之安全性与权限管理+vmware增加硬盘容量
			参考资料: http://www.cnblogs.com/Jackeyzhang/archive/2011/05/18/2049621.html VmWare虚拟机增加硬盘容量的方法 http://b ... 
- Android tab导航的几种方法:ActionBar tab +fragment,Viewpager+pagerTitleStrip,开源框架ViewPageIndicator 和 ViewPager
			action来实现tab标签 并跟fragment结合 因为要写新闻客户端这个tab导航是必须的 这里我写几个小练习,希望大家融会贯通. 1actionbar设置tab +fragment 布局是个l ... 
- 学习Shell脚本编程(第3期)_在Shell程序中使用的参数
			位置参数 内部参数 如同ls命令可以接受目录等作为它的参数一样,在Shell编程时同样可以使用参数.Shell程序中的参数分为位置参数和内部参数等. 3.1 位置参数 由系统提供的参数称为位置参数.位 ... 
