骑士放置 (匈牙利算法 最大独立集)

描述

给定一个 N*M 的棋盘,有一些格子禁止放棋子。问棋盘上最多能放多少个不能互相攻击的骑士(国际象棋的“骑士”,类似于中国象棋的“马”,按照“日”字攻击,但没有中国象棋“别马腿”的规则)。N, M<=100。

输入

第一行为n,m,t(表示有t个禁止的格子) 第二行到t+1行为x,y,分别表示禁止格子所在的位置,x为第x行,y为第y列,行列编号从1开始。

输出

一个整数,表示最多能放多少个骑士。

样例输入

2 3 0

样例输出

4

思路:

1.任意点(x1,y1),走日后,(x2,y2) 一定会从奇数变成偶数,偶数变成奇数,因此可以 划分奇数点和偶数点,显然可以用二分图来解决

2.要求多少个互不攻击的马,可以把每个格子看作一个点,在可以互相攻击的点间建立边, 二分图查找最大匹配数(即最小顶点数后),求得最大独立集即为最后结果。

3.最后的结果:最大独立集=总点数-最大匹配数-禁止点数

参考代码,详见注释:

#include 
#include 
#include 
#include 
#define LL long long
#define IO ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);
using namespace std;
const int MOD = 1e7+7;
const int INF = 0x3f3f3f3f;
const int N = 101;
int n,m,t; 
bool ma[N][N]; //存储图信息 
bool vis[N][N]; //是否访问过该点 
pair p[N][N];//存放奇数点图对应的偶数点 
int dir[8][2]={{-2,1},{-1,2},{1,2},{2,1},{2,-1},{1,-2},{-1,-2},{-2,-1}};  
/*
最小顶点覆盖 = 最大匹配数,而最大独立集 = 顶点数 - 最小顶点覆盖 
1.任意点(x1,y1),走日后,(x2,y2) 一定会从奇数变成偶数,偶数变成奇数,因此可以
划分奇数点和偶数点,显然可以用二分图来解决 
2.要求多少个互不攻击的马,可以把每个格子看作一个点,在可以互相攻击的点间建立边,
二分图查找最大匹配数(即最小顶点数后),求得最大独立集即为最后结果。 
3.最后的结果:最大独立集=总点数-最大匹配数-禁止点数 
*/
bool get(int x,int y)
{
	int nx,ny,cx,cy;
	for(int i=0;i<8;i++)
	{ 
		//跳跃的下个点,一定为奇数点 
		nx=x+dir[i][0];
		ny=y+dir[i][1];
		//1.越界判断    2.禁止点   3.访问过的点 
		if(nx<1 || nx>n || ny<1 || ny>m || ma[nx][ny] || vis[nx][ny])continue;

		vis[nx][ny]=1; 

		cx=p[nx][ny].first;
		cy=p[nx][ny].second;
		//1.若当前奇数点没有对应的偶数点  2.或者当前偶数点能有其他的奇数点可达,则让给他 
		if( (cx==0 && cy==0) || get(cx,cy) )
		{
			p[nx][ny]={x,y};
			return true;
		}
	}
	return false;
 } 
int main()
{
	IO;
	int x,y;
	cin>>n>>m>>t;
	for(int i=0;i>x>>y;
		ma[x][y]=1;
	}
	int ans=0;
	
	//枚举偶数点 
	for(int i=1;i<=n;i++)
	{
		for(int j=1;j<=m;j++)
		{
			//1.禁止点  2.奇数点 。 跳过枚举 
			if(ma[i][j] || ((i+j)&1) ) continue;
			memset(vis,0,sizeof(vis));
			if(get(i,j))ans++;
		}
	}
	//最大独立集=总点数-最大匹配数-禁止点数 
	cout<

你可能感兴趣的:(算法,图论)