题面(骑士共存问题)

在一个 \(n \times n\) 个方格的国际象棋棋盘上,马(骑士)可以攻击的棋盘方格如图所示。棋盘上某些方格设置了障碍,骑士不得进入。

对于给定的 \(n \times n\) 个方格的国际象棋棋盘和 \(m\) 个障碍标志,计算棋盘上最多可以放置多少个骑士,使得它们彼此互不攻击。

对于全部的测试点,保证 \(1 \leq n \leq 200\),\(0 \leq m \lt n^2\)。

思路

二分图的最大独立集。

先对这个棋盘进行黑白染色,然后发现,同颜色的格子的马永远无法互相攻击。

首先,先创立源点 \(s\),汇点 \(t\)。

然后对于白点 \((i,j)\),连边 \((s,(i,j),1)\)。对于黑点 \((k,l)\),连边 \(((k,l),t,1)\),对于可以互相攻击的点 \((a,b),(c,d)\),连边 \(((a,b),(c,d),+\infty)\)。

然后求最大独立集。

代码

P3355 骑士共存问题

#include <bits/stdc++.h>
#define int long long
using namespace std; int n,m,s,t; struct edge{
int from,to,val;
}e[2000001];int head[2000001],cur[2000001],siz=1;
void addedge(int x,int y,int z){
e[++siz].to=y,e[siz].val=z;
e[siz].from=head[x],head[x]=siz;
}
void add(int x,int y,int z){
addedge(x,y,z);
addedge(y,x,0);
} int gap[2000001];
bool bfs(){
memset(gap,0,sizeof(gap));
fill(gap+1,gap+1+n,0);
queue<int> q;
q.push(s);
gap[s]=1;
while(!q.empty()){
int now=q.front();
q.pop();
for(int i=head[now];i;i=e[i].from){
int u=e[i].to;
if(e[i].val&&!gap[u]){
gap[u]=gap[now]+1;
q.push(u);
}
}
}
return (gap[t]);
}
int dfs(int now,int val){
if(now==t) return val;
for(int &i=cur[now];i;i=e[i].from){
int u=e[i].to;
if(e[i].val&&gap[now]+1==gap[u]){
int F=dfs(u,min(e[i].val,val));
if(F){
e[i].val-=F;
e[i^1].val+=F;
return F;
}
}
}
return 0;
}
int dinic(){
int ret=0;
while(bfs()){
copy(head,head+1+n,cur);
int F=0;
while(F=dfs(s,10000000000000)){
ret+=F;
}
}
return ret;
} int knight[100000][1000]; int mk(int x,int y){
x--;
return (x*n+y);
} int delta[][2]={{1,2},{1,-2},{-1,2},{-1,-2},{2,1},{2,-1},{-2,1},{-2,-1}}; bool valid(int x){
return (x>=1)&&(x<=n);
} signed main(){
cin>>n>>m;
for(int i=1;i<=m;i++){
int x,y;
cin>>x>>y;
knight[x][y]=1;
}
s=0,t=n*n+1; for(int i=1;i<=n;i++){
for(int j=1;j<=n;j++){
if((i+j)%2&&!knight[i][j]){
add(s,mk(i,j),1);
}
if(!((i+j)%2)&&!knight[i][j]){
add(mk(i,j),t,1);
}
}
}
for(int i=1;i<=n;i++){
for(int j=1;j<=n;j++){
if((i+j)%2==0){
continue;
}
for(int k=0;k<8;k++){
int tx=i+delta[k][0];
int ty=j+delta[k][1];
if(valid(tx)&&valid(ty)){
if(!knight[tx][ty]){
add(mk(i,j),mk(tx,ty),INT_MAX);
}
}
}
}
}
n=n*n;
cout<<n-m-dinic()<<'\n';
return 0;
}

P4304 [TJOI2013]攻击装置

#include <bits/stdc++.h>
#define int long long
using namespace std; int n,m,s,t; struct edge{
int from,to,val;
}e[200001];int head[200001],cur[200001],siz=1;
void addedge(int x,int y,int z){
e[++siz].to=y,e[siz].val=z;
e[siz].from=head[x],head[x]=siz;
}
void add(int x,int y,int z){
addedge(x,y,z);
addedge(y,x,0);
} int gap[200001];
bool bfs(){
memset(gap,0,sizeof(gap));
fill(gap+1,gap+1+n,0);
queue<int> q;
q.push(s);
gap[s]=1;
while(!q.empty()){
int now=q.front();
q.pop();
for(int i=head[now];i;i=e[i].from){
int u=e[i].to;
if(e[i].val&&!gap[u]){
gap[u]=gap[now]+1;
q.push(u);
}
}
}
return (gap[t]);
}
int dfs(int now,int val){
if(now==t) return val;
for(int &i=cur[now];i;i=e[i].from){
int u=e[i].to;
if(e[i].val&&gap[now]+1==gap[u]){
int F=dfs(u,min(e[i].val,val));
if(F){
e[i].val-=F;
e[i^1].val+=F;
return F;
}
}
}
return 0;
}
int dinic(){
int ret=0;
while(bfs()){
copy(head,head+1+n,cur);
int F=0;
while(F=dfs(s,10000000000000)){
ret+=F;
}
}
return ret;
} int knight[10000][1000]; int mk(int x,int y){
x--;
return (x*n+y);
} int delta[][2]={{1,2},{1,-2},{-1,2},{-1,-2},{2,1},{2,-1},{-2,1},{-2,-1}}; bool valid(int x){
return (x>=1)&&(x<=n);
} int gc(){
char ch;
do{
ch=getchar();
}while(!isdigit(ch));
return ch-'0';
} signed main(){
cin>>n;
for(int i=1;i<=n;i++){
for(int j=1;j<=n;j++){
knight[i][j]=gc();
m+=knight[i][j];
}
}
s=0,t=n*n+1; for(int i=1;i<=n;i++){
for(int j=1;j<=n;j++){
if((i+j)%2&&!knight[i][j]){
add(s,mk(i,j),1);
}
if(!((i+j)%2)&&!knight[i][j]){
add(mk(i,j),t,1);
}
}
}
for(int i=1;i<=n;i++){
for(int j=1;j<=n;j++){
if((i+j)%2==0){
continue;
}
for(int k=0;k<8;k++){
int tx=i+delta[k][0];
int ty=j+delta[k][1];
if(valid(tx)&&valid(ty)){
if(!knight[tx][ty]){
add(mk(i,j),mk(tx,ty),INT_MAX);
}
}
}
}
}
n=n*n;
cout<<n-m-dinic()<<'\n';
return 0;
}

习题:P5030 长脖子鹿放置

题面

众周所知,在西洋棋中,我们有城堡、骑士、皇后、主教和长脖子鹿。

如图所示,西洋棋的“长脖子鹿”,类似于中国象棋的马,但按照“目”字攻击,且没有中国象棋“别马腿”的规则。(因为长脖子鹿没有马腿)

给定一个\(N * M\),的棋盘,有 \(k\) 个格子禁止放棋子。问棋盘上最多能放多少个不能互相攻击的长脖子鹿。

对于\(100\)%的数据,\(1 ≤ N,M ≤ 200\)

代码

#include <bits/stdc++.h>
#define int long long
using namespace std; int n,m,s,t; struct edge{
int from,to,val;
}e[2000001];int head[2000001],cur[2000001],siz=1;
void addedge(int x,int y,int z){
e[++siz].to=y,e[siz].val=z;
e[siz].from=head[x],head[x]=siz;
}
void add(int x,int y,int z){
addedge(x,y,z);
addedge(y,x,0);
} int gap[2000001];
bool bfs(){
memset(gap,0,sizeof(gap));
fill(gap+1,gap+1+n,0);
queue<int> q;
q.push(s);
gap[s]=1;
while(!q.empty()){
int now=q.front();
q.pop();
for(int i=head[now];i;i=e[i].from){
int u=e[i].to;
if(e[i].val&&!gap[u]){
gap[u]=gap[now]+1;
q.push(u);
}
}
}
return (gap[t]);
}
int dfs(int now,int val){
if(now==t) return val;
for(int &i=cur[now];i;i=e[i].from){
int u=e[i].to;
if(e[i].val&&gap[now]+1==gap[u]){
int F=dfs(u,min(e[i].val,val));
if(F){
e[i].val-=F;
e[i^1].val+=F;
return F;
}
}
}
return 0;
}
int dinic(){
int ret=0;
while(bfs()){
copy(head,head+1+n,cur);
int F=0;
while(F=dfs(s,10000000000000)){
ret+=F;
}
}
return ret;
} int knight[100000][1000]; int mk(int x,int y){
x--;
return (x*m+y);
} int delta[][2]={{1,3},{1,-3},{-1,3},{-1,-3},{3,1},{3,-1},{-3,1},{-3,-1}}; bool valid(int x){
return (x>=1)&&(x<=n);
}
bool validy(int x){
return (x>=1)&&(x<=m);
} int kkk,k; signed main(){
cin>>n>>m>>kkk;
for(int i=1;i<=kkk;i++){
int x,y;
cin>>x>>y;
if(!knight[x][y]){
k++;
}
knight[x][y]=1;
}
s=0,t=n*m+1;
for(int i=1;i<=n;i++){
for(int j=1;j<=m;j++){
if((i)%2){
add(s,mk(i,j),1);
}
if(!((i)%2)){
add(mk(i,j),t,1);
}
}
}
for(int i=1;i<=n;i++){
for(int j=1;j<=m;j++){
if(knight[i][j]){
continue;
}
if((i)%2==0){
continue;
}
for(int k=0;k<8;k++){
int tx=i+delta[k][0];
int ty=j+delta[k][1];
if(valid(tx)&&validy(ty)){
if(!knight[tx][ty]){
add(mk(i,j),mk(tx,ty),1);
}
}
}
}
}
n=n*m;
cout<<n-k-dinic()<<'\n';
return 0;
}

网络流棋盘模型 | P3355 骑士共存问题 P4304 [TJOI2013]攻击装置的更多相关文章

  1. P4304 [TJOI2013]攻击装置

    传送门 看到棋盘先黑白染色冷静一下 然后发现...攻击的时候同种颜色不会相互攻击 这样就是个网络流经典套路了,关于这个套路我以前好像写过几题,那边有解释一下:传送门 #include<iostr ...

  2. 洛谷P4304 [TJOI2013]攻击装置 题解

    题目链接: https://www.luogu.org/problemnew/show/P4304 分析: 最大独立集 最大独立集=总点数-最大匹配数 独立集:点集,图中选一堆点,这堆点两两之间没有连 ...

  3. 洛谷P4304 TJOI2013 攻击装置 (二分图匹配)

    题目大意:一个矩阵,一些点被拿掉,在棋盘上马走日,马之间不能落在同一点,求最多放几匹马. 采用对矩阵黑白染色,画个图可以发现:马可以走到的位置和他所处的位置颜色不同,将马和他可以走到的位置连边,最多可 ...

  4. 洛咕 P4304 [TJOI2013]攻击装置

    把坐标按照(x+y)%2染色可以发现这是个二分图 二分图最大独立集=点数-最大匹配 于是就是个算匹配的傻逼题了 // luogu-judger-enable-o2 #include<bits/s ...

  5. P4304 [TJOI2013]攻击装置 最小割

    $ \color{#0066ff}{ 题目描述 }$ 给定一个01矩阵,其中你可以在0的位置放置攻击装置. 每一个攻击装置(x,y)都可以按照"日"字攻击其周围的8个位置(x-1, ...

  6. 【洛谷】4304:[TJOI2013]攻击装置【最大点独立集】【二分图】2172: [国家集训队]部落战争【二分图/网络流】【最小路径覆盖】

    P4304 [TJOI2013]攻击装置 题目描述 给定一个01矩阵,其中你可以在0的位置放置攻击装置. 每一个攻击装置(x,y)都可以按照“日”字攻击其周围的8个位置(x-1,y-2),(x-2,y ...

  7. P3355 骑士共存问题

    P3355 骑士共存问题 题目描述 在一个 n*n (n <= 200)个方格的国际象棋棋盘上,马(骑士)可以攻击的棋盘方格如图所示.棋盘上某些方格设置了障碍,骑士不得进入 对于给定的 n*n ...

  8. loj #6226. 「网络流 24 题」骑士共存问题

    #6226. 「网络流 24 题」骑士共存问题   题目描述 在一个 n×n\text{n} \times \text{n}n×n 个方格的国际象棋棋盘上,马(骑士)可以攻击的棋盘方格如图所示.棋盘上 ...

  9. P3355 骑士共存问题 二分建图 + 当前弧优化dinic

    P3355 骑士共存问题 题意: 也是一个棋盘,规则是“马”不能相互打到. 思路: 奇偶点分开,二分图建图,这道题要注意每个点可以跑八个方向,两边都可以跑,所以边 = 20 * n * n. 然后di ...

  10. P3355 骑士共存问题 网络流

    骑士共存 题目描述 在一个 n*n个方格的国际象棋棋盘上,马(骑士)可以攻击的棋盘方格如图所示.棋盘上某些方格设置了障碍,骑士不得进入 对于给定的 n*n 个方格的国际象棋棋盘和障碍标志,计算棋盘上最 ...

随机推荐

  1. 【番外篇】Rust环境搭建+基础开发入门+Rust与.NET6、C++的基础运算性能比较

    前言:突然想打算把Rust作为将来自己主要的副编程语言.当然,主语言还是C#,毕竟.NET平台这么强大,写起来就是爽.缘起:之前打算一些新的产品或者新的要开发的东西,由于没有历史包袱,就想重新选型一下 ...

  2. 题解 CF327A Flipping Game

    前言 数据水的一批,\(\mathcal{O}(n^3)\) 给过我觉得是不应该的. 题意 有一个由 \(0\) 和 \(1\) 组成的序列 \(a_1,a_2,a_3,a_4....,a_n\) . ...

  3. VBA---Basic

    题记: 之前用VBA做过几个小工具,用来实现办公自动化的.在编写过程中也遇到了一些问题,但最终都通过网友们的分享予以解决,现对其中的一些知识点进行总结. common sense 取消文件刷新: Ap ...

  4. 怎么实现无痛刷新token

    最近遇到这个需求,前端登录后,后端返回  access_token 和 refresh_token ,当token 过期时用旧的 refresh_token 去获取新的token,前端要不痛去刷新to ...

  5. 带你从0到1开发AI图像分类应用

    摘要:通过一个垃圾分类应用的开发示例,介绍AI Gallery在AI应用开发流程中的作用. 本文分享自华为云社区<AI Gallery:从0到1开发AI图像分类应用>,作者: yd_269 ...

  6. 如何使用ModelBox快速提升AI应用性能?

    摘要:在开发初期开发者往往聚焦在模型的精度上,性能关注较少,但随着业务量不断增加,AI应用的性能往往成为瓶颈,此时对于没有性能优化经验的开发者来说往往需要耗费大量精力做优化性能,本文为开发者介绍一些常 ...

  7. __init__、__all__

    在python中 引用模块包的时候,要先进入此模块的__init__.py中畅游一遍,因此,我们多次需要一个语句的时候,就可以将这些语句写入到__init__.py中: 在使用*号的时候我们可以用__ ...

  8. .NET 7 的 AOT 到底能不能扛反编译?

    一:背景 1.讲故事 在B站,公众号上发了一篇 AOT 的文章后,没想到反响还是挺大的,都称赞这个东西能抗反编译,可以让破解难度极大提高,可能有很多朋友对逆向不了解,以为用 ILSpy,Reflect ...

  9. vue 中使用 this 更新数据的一次大坑

    情景说明: 之前用 vue 做数据绑定更新时,发现一个莫名奇妙的问题. 我在 vue 实例中声明了一个数组属性如 books: [],在异步请求的回调函数中使用 this.books = res.da ...

  10. 实践案例:同程艺龙网的 Dubbo 升级经验总结

    本篇为同程艺龙旅行网 Apache Dubbo 的实践案例总结.感兴趣的朋友可以访问官网了解更多详情,或搜索关注官方微信公众号 Apache Dubbo 跟进最新动态. 作者信息: 严浩:同程艺龙高级 ...