二分图——洛谷P2825 [HEOI2016]游戏
来源:互联网 发布:python 字节流 编辑:程序博客网 时间:2024/06/05 08:57
https://www.luogu.org/problem/show?pid=2825
这个显然就是以行列来二分图;
因为二分图的话每个点最多连接一条边相当于这一行里放一个炸弹;
建图
遇到#就分段
比如 * * xx # * 这一行就分成两段;
**xxxx这一行直接就是一段;
然后大力二分;
我一开始写法;
#include<bits/stdc++.h>using namespace std;const int N=3e3;struct qu{int z,x,y;}d[N*2];struct cs{int to,nxt;}a[N*N];int A[N],B[N],aa,bb;int head[N],ll,link[N];int n,m,top,ans;bool vi[N];char c[55][55];void init(int x,int y){ a[++ll].to=y; a[ll].nxt=head[x]; head[x]=ll;}bool check(qu a,qu b){ if(c[a.z][b.z]!='*')return 0; if(a.z<b.x||b.y<a.z)return 0; if(b.z<a.x||a.y<b.z)return 0; return 1;}void make(){ int last=0,now=0; for(int i=1;i<=n;i++) for(int j=1;j<=m+1;j++) if(j==m+1||c[i][j]=='#'){ if(!now)continue; d[++top].z=i; d[top].x=last; d[top].y=now; A[++aa]=top; last=now=0; }else{if(!last)last=j;now=j;} for(int j=1;j<=m;j++) for(int i=1;i<=n+1;i++) if(i==n+1||c[i][j]=='#'){ if(!now)continue; d[++top].z=j; d[top].x=last; d[top].y=now; B[++bb]=top; last=now=0; }else{if(!last)last=i;now=i;} for(int i=1;i<=aa;i++) for(int j=1;j<=bb;j++) if(check(d[A[i]],d[B[j]]))init(i,j);}bool dfs(int x){ for(int k=head[x];k;k=a[k].nxt) if(!vi[a[k].to]){ vi[a[k].to]=1; if(!link[a[k].to]||dfs(link[a[k].to])){ link[a[k].to]=x;return 1; } }return 0;}int main(){ scanf("%d%d",&n,&m); for(int i=1;i<=n;i++) for(int j=1;j<=m;j++) cin>>c[i][j]; make(); for(int i=1;i<=aa;i++){ memset(vi,0,sizeof vi); if(dfs(i))ans++; } printf("%d",ans);}
将近模拟;
然后用上dp的巧妙思想;
#include<bits/stdc++.h>using namespace std;const int N=3e3;struct qu{int z,x,y;}d[N*2];struct cs{int to,nxt;}a[N*N];int A[55][55],B[55][55],aa,bb;int head[N],ll,link[N];int n,m,top,ans;bool vi[N];char c[55][55];void init(int x,int y){ a[++ll].to=y; a[ll].nxt=head[x]; head[x]=ll;}void make(){ for(int i=1;i<=n;i++) for(int j=1;j<=m;j++){ if(i!=1&&c[i-1][j]!='#')A[i][j]=A[i-1][j];else A[i][j]=++aa; if(j!=1&&c[i][j-1]!='#')B[i][j]=B[i][j-1];else B[i][j]=++bb; } for(int i=1;i<=n;i++) for(int j=1;j<=m;j++) if(c[i][j]=='*')init(A[i][j],B[i][j]);}bool dfs(int x){ for(int k=head[x];k;k=a[k].nxt) if(!vi[a[k].to]){ vi[a[k].to]=1; if(!link[a[k].to]||dfs(link[a[k].to])){ link[a[k].to]=x;return 1; } }return 0;}int main(){ scanf("%d%d",&n,&m); for(int i=1;i<=n;i++) for(int j=1;j<=m;j++) cin>>c[i][j]; make(); for(int i=1;i<=aa;i++){ memset(vi,0,sizeof vi); if(dfs(i))ans++; } printf("%d",ans);}
阅读全文
1 0
- 二分图——洛谷P2825 [HEOI2016]游戏
- 洛谷 P2825 [HEOI2016] 游戏
- 洛谷 P2825 [HEOI2016]游戏
- 洛谷P2825 [HEOI2016]游戏
- 【HEOI2016】游戏 二分图
- 洛谷2825 HEOI2016 游戏 二分图匹配
- 二分图匹配——BZOJ4554/Luogu2825 [Tjoi2016&Heoi2016]游戏
- HEOI2016 游戏 二分图匹配
- 【bzoj4554】【Tjoi2016&Heoi2016】【游戏】【二分图】
- 【bzoj4554】【Tjoi2016】【Heoi2016】【游戏】【二分图匹配】
- [BZOJ4554][Tjoi2016&Heoi2016][二分图匹配]游戏
- [Tjoi2016&Heoi2016]游戏 二分图最大匹配
- 二分图 BZOJ4554 [Tjoi2016&Heoi2016]游戏
- 【BZOJ】4554 [Tjoi2016&Heoi2016]游戏 二分图
- bzoj4554 [HEOI2016]游戏 二分匹配
- BZOJ4554(Tjoi2016&Heoi2016)[游戏]--二分图最大匹配
- 【二分图最大匹配】BZOJ4554 [Tjoi2016&Heoi2016]游戏
- [BZOJ4554][TJOI2016&HEOI2016]游戏(二分图匹配)
- Python中关于CSV文件中的I/O
- Crawler——链接爬虫
- Scraping_regex
- 冒泡排序——java语言
- Scraper——BeautifulSoup and LXML
- 二分图——洛谷P2825 [HEOI2016]游戏
- grunt 新手一日入门
- 第三章 组合逻辑
- 自定义Flash背景的相关设置方法以及其与目录下的文件的对应关系
- C语言程序设计(24)
- PKUSC 2017 酱油记
- Scraper_compare( 'NoneType' object has no attribute 'group')
- 数据挖掘——亲和性分析
- python算法演练——One Rule 算法