[TCO2013]DirectionBoard
题意:给一个网格,每个格子有一个方向表示在这个格子上要往哪个方向走,你可以改变某些格子的方向,问最少多少次操作使得从任意格子出发都能回到这个格子
woc这都不会我还是回家种田去吧...
题目的要求是改变某些点的出边使得整个图变成许多环,注意到在满足要求的图中,每个点只有一条入边和一条出边,所以我们对每个格子建两个点代表它的出边和入边,相当于是要找到一组入边和出边的匹配,如果一个入边和出边匹配是原来的方向,那么费用为$0$,否则费用为$1$,跑最小费用最大流即可
#include<stdio.h>
#include<string>
#include<vector>
#include<queue>
#include<string.h>
using namespace std;
const int inf=2147483647;
int h[510],nex[3010],to[3010],cap[3010],cos[3010],M=1,S,T;
void ins(int a,int b,int c,int d){
M++;
to[M]=b;
cap[M]=c;
cos[M]=d;
nex[M]=h[a];
h[a]=M;
}
void add(int a,int b,int c,int d){
ins(a,b,c,d);
ins(b,a,0,-d);
}
int dis[510],n,sum,cost;
bool aug[510];
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;
bool bfs(){
int x,i;
memset(dis,63,sizeof(dis));
dis[T]=0;
q.push(T);
while(!q.empty()){
t=q.top();
q.pop();
x=t.x;
if(dis[x]!=t.d)continue;
for(i=h[x];i;i=nex[i]){
if(cap[i^1]&&dis[x]-cos[i]<dis[to[i]]){
dis[to[i]]=dis[x]-cos[i];
q.push(to[i]);
}
}
}
sum+=dis[S];
for(x=1;x<=n;x++){
for(i=h[x];i;i=nex[i])cos[i]+=dis[to[i]]-dis[x];
}
return dis[S]!=dis[0];
}
int dfs(int x,int f){
if(f==0)return 0;
if(x==T){
cost+=f*sum;
return f;
}
int us=0,i,t;
aug[x]=1;
for(i=h[x];i;i=nex[i]){
if(!cos[i]&&cap[i]&&!aug[to[i]]){
t=dfs(to[i],min(cap[i],f-us));
cap[i]-=t;
cap[i^1]+=t;
us+=t;
if(us==f)break;
}
}
if(us==f)aug[x]=0;
return us;
}
int costflow(){
do{
do{
memset(aug,0,sizeof(aug));
}while(dfs(S,inf));
}while(bfs());
return cost;
}
const int g[4][2]={{0,-1},{0,1},{-1,0},{1,0}};
int type(char c){
if(c=='L')return 0;
if(c=='R')return 1;
if(c=='U')return 2;
return 3;
}
class DirectionBoard{
public:
int getMinimum(vector<string>mp){
int n,m,i,j,k,x,y,t;
n=mp.size();
m=mp[0].length();
S=n*m*2+1;
T=S+1;
::n=T;
for(i=0;i<n;i++){
for(j=0;j<m;j++){
add(S,i*m+j+1,1,0);
t=type(mp[i][j]);
for(k=0;k<4;k++){
x=(i+g[k][0]+n)%n;
y=(j+g[k][1]+m)%m;
add(i*m+j+1,x*m+y+1+n*m,1,k!=t);
}
add(i*m+j+1+n*m,T,1,0);
}
}
return costflow();
}
};
/*
int main(){
vector<string>vt;
char s[20];
DirectionBoard cl;
while(~scanf("%s",s))vt.push_back(s);
printf("%d",cl.getMinimum(vt));
}
*/
[TCO2013]DirectionBoard的更多相关文章
- [TCO2013]TrickyInequality
$\newcommand{stirf}[2]{{{#1}\brack{#2}}}$$\newcommand{stirs}[2]{{{#1}\brace{#2}}}$题意:$\sum\limits_{i ...
- [TCO2013]Block3Checkers
题意:一个网格上有一些障碍和$3$个在网格边界上的棋子,你要添加一些障碍使得没有两个棋子四连通,问最少添加多少个障碍 官方题解——一张图教你做人... 三个棋子将网格边界分成三段,添加障碍后网格中一定 ...
- [TCO2013]LitPanels
题意:一个$n\times m$的无色网格,你可以在其中选择两个$x\times y$的子矩形并在其中将其中任意的格子涂上颜色,问最终能得到多少种不同的网格 做这题会用到一个概念叫包围盒(boundi ...
随机推荐
- 【洛谷 P2761】 软件补丁问题(状态压缩,最短路)
题目链接 第四题. 初看题目很懵,网络流这么厉害的吗,毫无头绪去看题解.. 所以这和网络流有什么关系呢? 把规则用二进制保存下来,然后跑最短路救星了. 在线跑,离线连边太慢了. (以后干脆不管什么题直 ...
- 使用idea+Tomcat搭建servlet服务器
1.使用java 搭建一个简单的Servlet 服务器 https://blog.csdn.net/qq_35164169/article/details/76146655
- hdu 1081 To The Max(dp+化二维为一维)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1081 To The Max Time Limit: 2000/1000 MS (Java/Others ...
- 64_d1
DSDP-5.8-15.fc26.i686.rpm 13-Feb-2017 22:06 658926 DSDP-5.8-15.fc26.x86_64.rpm 13-Feb-2017 22:09 653 ...
- 64_c2
coin-or-Bcp-1.4.3-3.fc26.i686.rpm 22-May-2017 21:07 250866 coin-or-Bcp-1.4.3-3.fc26.x86_64.rpm 22-Ma ...
- Lambda 表达式 in java 8
Lambda 表达式 in Java 8 Lambda表达式是java 8 新增的特性 Lambda表达式主要作用:支持将代码块作为方法参数,允许使用更简洁的代码创建函数式接口的实例,是匿名内部类的一 ...
- Winform利用委托进行窗体间的传值
在form1.cs中 1.委托的定义 //定义一个委托 public delegate void AddUsrEventHandler(object sender, AddUsrEventHandle ...
- Java初次见面
1.Java语言特点(运行环境JRE[操作系统,api,dll]): a.跨平台:Java自带的虚拟机很好地实现了跨平台性.Java源程序代码经过编译后生成二进制的字节码是与平台无关的,但是可被Jav ...
- 关于 拼接 url 连接 参数的问题(爬虫)。
比如这里 我找的 后台请求的json的链接: 第一页: http://www.igoldenbeta.com:8080/cn-jsfund-server-mobile/bkt/api?appkey=1 ...
- awk处理之案例五:awk匹配字段2包含字段1的文本
编译环境 本系列文章所提供的算法均在以下环境下编译通过. [脚本编译环境]Federa 8,linux 2.6.35.6-45.fc14.i686 [处理器] Intel(R) Core(TM)2 Q ...