[TCO2013]Block3Checkers
题意:一个网格上有一些障碍和$3$个在网格边界上的棋子,你要添加一些障碍使得没有两个棋子四连通,问最少添加多少个障碍
官方题解——一张图教你做人...
三个棋子将网格边界分成三段,添加障碍后网格中一定存在一个点使得它可以到这三段(只走障碍的路径,八连通)
所以找出这三段后分别以它们为起点跑最短路即可,经过障碍权值为$0$,经过空地权值为$1$
#include<stdio.h> #include<queue> #include<string.h> #include<vector> #include<string> using namespace std; int h[410],nex[7010],to[7010],v[7010],M; void add(int a,int b,int c){ M++; to[M]=b; v[M]=c; nex[M]=h[a]; h[a]=M; } int dis[410]; struct pr{ int x,d; pr(int u=0){x=u;d=dis[u];} }t; bool operator<(pr a,pr b){return a.d>b.d;} priority_queue<pr>q; void dijk(int x){ int i; memset(dis,63,sizeof(dis)); dis[x]=0; q.push(x); while(!q.empty()){ t=q.top(); q.pop(); x=t.x; if(t.d!=dis[x])continue; for(i=h[x];i;i=nex[i]){ if(dis[x]+v[i]<dis[to[i]]){ dis[to[i]]=dis[x]+v[i]; q.push(to[i]); } } } } const int g4[4][2]={{0,1},{0,-1},{1,0},{-1,0}},g8[8][2]={{-1,-1},{-1,0},{-1,1},{0,-1},{0,1},{1,-1},{1,0},{1,1}}; int d[3][410],n,m; vector<string>mp; bool ok(int x,int y){ return 0<=x&&x<n&&0<=y&&y<m; } bool edge(int x,int y){ return ok(x,y)&&(x==0||x==n-1||y==0||y==m-1); } void dfs(int fx,int fy,int x,int y,int s){ int i,tx,ty; if(mp[x][y]=='A')return; add(s,x*m+y+1,0); add(x*m+y+1,s,mp[x][y]=='.'); for(i=0;i<4;i++){ tx=x+g4[i][0]; ty=y+g4[i][1]; if((tx!=fx||ty!=fy)&&edge(tx,ty))dfs(x,y,tx,ty,s); } } class Block3Checkers{ public: int blockThem(vector<string>mp){ int i,j,k,x,y,f,s[3],ans; ::mp=mp; n=mp.size(); m=mp[0].length(); s[0]=n*m+1; s[1]=n*m+2; s[2]=n*m+3; f=0; for(i=0;i<m;i++){ if(mp[0][i]=='A'){ if((i&&mp[0][i-1]=='A')||(i<m-1&&mp[0][i+1]=='A')||mp[1][i]=='A')return 100; if(i==0) dfs(0,0,1,0,s[f]); else dfs(0,i,0,i-1,s[f]); f++; } if(mp[n-1][i]=='A'){ if((i&&mp[n-1][i-1]=='A')||(i<m-1&&mp[n-1][i+1]=='A')||mp[n-2][i]=='A')return 100; if(i==m-1) dfs(n-1,m-1,n-2,m-1,s[f]); else dfs(n-1,i,n-1,i+1,s[f]); f++; } } for(i=1;i<n-1;i++){ if(mp[i][0]=='A'){ if(mp[i-1][0]=='A'||mp[i+1][0]=='A')return 100; dfs(i,0,i+1,0,s[f]); f++; } if(mp[i][m-1]=='A'){ if(mp[i-1][m-1]=='A'||mp[i+1][m-1]=='A')return 100; dfs(i,m-1,i-1,m-1,s[f]); f++; } } for(i=0;i<n;i++){ for(j=0;j<m;j++){ if(mp[i][j]!='A'){ for(k=0;k<8;k++){ x=i+g8[k][0]; y=j+g8[k][1]; if(ok(x,y)&&mp[x][y]!='A')add(i*m+j+1,x*m+y+1,mp[i][j]=='.'); } } } } for(i=0;i<3;i++){ dijk(s[i]); memcpy(d[i],dis,sizeof(dis)); } ans=n*m; for(i=0;i<n;i++){ for(j=0;j<m;j++){ if(mp[i][j]!='A'){ x=i*m+j+1; ans=min(ans,d[0][x]+d[1][x]+d[2][x]+(mp[i][j]=='.')); } } } return ans; } }; /* int main(){ vector<string>vt; char s[30]; Block3Checkers cl; while(~scanf("%s",s))vt.push_back(s); printf("%d",cl.blockThem(vt)); } */
[TCO2013]Block3Checkers的更多相关文章
- [TCO2013]TrickyInequality
$\newcommand{stirf}[2]{{{#1}\brack{#2}}}$$\newcommand{stirs}[2]{{{#1}\brace{#2}}}$题意:$\sum\limits_{i ...
- [TCO2013]LitPanels
题意:一个$n\times m$的无色网格,你可以在其中选择两个$x\times y$的子矩形并在其中将其中任意的格子涂上颜色,问最终能得到多少种不同的网格 做这题会用到一个概念叫包围盒(boundi ...
- [TCO2013]DirectionBoard
题意:给一个网格,每个格子有一个方向表示在这个格子上要往哪个方向走,你可以改变某些格子的方向,问最少多少次操作使得从任意格子出发都能回到这个格子 woc这都不会我还是回家种田去吧... 题目的要求是改 ...
随机推荐
- 21、python操作redis的模块?
什么是redis? redis是一个key-value存储系统.和Memcached类似,它支持存储的value类型相对更多,包括string(字符串).list(链表).set(集合).zset(s ...
- No 'Access-Control-Allow-Origin' Ajax跨域访问解决方案
No 'Access-Control-Allow-Origin' header is present on the requested resource. 当使用ajax访问远程服务器时,请求失败,浏 ...
- poj 2104 可持久化线段树
我们先离散化,然后根据权值建立线段树,假设我们现在有一颗权值线段树,表示在区间1-n中每个数出现了几次,那么我们可以二分的求出来这个区间的k大值,类似sbt的select操作,那么因为点的权值插入是无 ...
- Python2.7.3 Tkinter Entry(文本框) 说明
Python学习记录--关于Tkinter Entry(文本框)的选项.方法说明,以及一些示例. 属性(Options) background(bg) borderwidth(bd) cursor ...
- 自动化测试===unittest和requests接口测试案例,测试快递查询api(二)
在原来基础上生成测试报告: 首先需要 HTMLTestRunner.py 的unittest生成报告文件 (源码,自动化测试===unittest配套的HTMLTestRunner.py生成html ...
- 2017中国大学生程序设计竞赛 - 网络选拔赛 HDU 6154 CaoHaha's staff 思维
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6154 题意:在笛卡尔坐标系下,画一个面积至少为 n 的简单多边形,每次只能画一条边或者一个格子的对角 ...
- C json实战引擎 二 , 实现构造部分
引言 这篇博文和前一篇 C json实战引擎一,实现解析部分设计是相同的,都是采用递归下降分析. 这里扯一点 假如你是学生 推荐一本书 给 大家 自制编程语言 http://baike.baidu.c ...
- date 时间确定
获取当前时间: var date = new Date(); var year = date.getFullYear(); var month = date.getMonth() + 1; var d ...
- leetcode 之Set Matrix Zeroes(10)
设置两个布尔数组,记录行和列是否存在0.需要注意的是如何将行或列设为0. void setZeros(vector<vector<int>> &matrix) { in ...
- linux命令(18):chmod命令
1. 命令格式: chmod [-cfvR] [--help] [--version] mode file 2. 命令功能: 用于改变文件或目录的访问权限,用它控制文件或目录的访问权限. 3. 命令参 ...