[BZOJ]1059 矩阵游戏(ZJOI2007)
虽然说是一道水题,但小C觉得还是挺有意思的,所以在这里mark一下。
Description
Input
Output
输出文件应包含T行。对于每一组数据,如果该关卡有解,输出一行Yes;否则输出一行No。
Sample Input
2
2
0 0
0 1
3
0 0 1
0 1 0
1 0 0
Sample Output
No
Yes
HINT
对于100%的数据,N ≤ 200。
Solution
拿到这道题目肯定要思考,什么样的局面通过交换行列可以变成主对角线上都为黑格的情况?
由于交换是可逆的,我们倒过来思考,主对角线上都是黑格的情况通过交换行列可以变成什么局面。
这样就一目了然了。一开始是1~n的排列,每次交换行列相当于交换两个位置上的数字。
这样问题就转化成:是否能从每行选出恰好一个黑格,n行黑格代表的列数正好组成一个1~n的全排列。
正解是二分图匹配。全排列可以看作是行和列的匹配,思路很清晰。时间复杂度O(n^2)(匈牙利算法)/O(n^1.5)(网络流)。
但是小C这里要介绍的,是另一种做法。
想到矩阵和排列,我们会很自然地想起行列式。
我们思考一下行列式的计算公式:。
发现如果一个排列上的所有数都不为0,就一定会对答案有贡献!
所以我们就直接求这个矩阵的行列式的值就可以了???复杂度O(n^3)。
当然不行,这些排列对答案的贡献有正有负,矩阵的值全为0或1的话很容易凑出0。例如{{0,1,1},{0,1,1},{1,0,0}}。
那怎么办?
很简单啊,我们给矩阵里的黑格都随机一个权值就好了嘛。
什么?还是WA?
我们来看一看行列式等于0还有什么条件:
1.有一行或一列全为0的情况;
2.有两行或两列数值成比例的情况;
3.行向量之间或列向量之间有相关的情况;
4.逆矩阵不存在的情况:
5.行列式对应的矩阵的秩小于行列式的阶数的情况……
够了!似乎第二条看起来特别扎眼,于是我们给矩阵里的黑格都随机一个质数权值如何?
过了卧槽。
二分图匹配:
#include <cstdio>
#include <cstring>
#include <algorithm>
#define MN 205
using namespace std;
bool u[MN],mp[MN][MN];
int mat[][MN];
int t,n; inline int read()
{
int n=,f=; char c=getchar();
while (c<'' || c>'') {if(c=='-')f=-; c=getchar();}
while (c>='' && c<='') {n=n*+c-''; c=getchar();}
return n*f;
} bool dfs(int x)
{
if (u[x]) return false;
u[x]=true;
for (register int i=;i<=n;++i)
{
if (!mp[x][i]||mat[][x]==i) continue;
if (!mat[][i]||dfs(mat[][i])) {mat[][x]=i; mat[][i]=x; return true;}
}
return false;
} int main()
{
register int i,j;
t=read();
while (t--)
{
n=read();
for (i=;i<=n;++i)
for (j=;j<=n;++j) mp[i][j]=read();
memset(mat,,sizeof(mat));
for (i=;i<=n;++i)
{
memset(u,,sizeof(u));
if (!dfs(i)) break;
}
if (i<=n) puts("No"); else puts("Yes");
}
}
高斯消元求行列式:
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cstdlib>
#define mod 1000000007
#define MP 2000005
#define MN 205
using namespace std;
int a[MN][MN],p[MP];
bool u[MP];
int t,n,pin; inline int read()
{
int n=,f=; char c=getchar();
while (c<'' || c>'') {if(c=='-')f=-; c=getchar();}
while (c>='' && c<='') {n=n*+c-''; c=getchar();}
return n*f;
} int mi(int x,int y)
{
register int z=;
for (;y;x=1LL*x*x%mod,y>>=) if (y&) z=1LL*z*x%mod;
return z;
} void init()
{
register int i,j;
for (i=;i<MP;++i)
{
if (!u[i]) p[++pin]=i;
for (j=;i*p[j]<MP;++j)
{
u[i*p[j]]=true;
if (i%p[j]==) break;
}
}
} int main()
{
register int i,j,k,x,mxi,lt;
srand(); init();
t=read();
while (t--)
{
n=read();
for (i=;i<=n;++i)
for (j=;j<=n;++j)
if (read()&) a[i][j]=p[(rand()*rand()%pin+rand())%pin+];
else a[i][j]=;
for (i=;i<n;++i)
{
mxi=i;
for (j=i+;j<=n;++j) if (a[j][i]>a[mxi][i]) mxi=j;
if (mxi!=i) swap(a[mxi],a[i]);
if (!a[i][i]) break;
for (j=i+;j<=n;++j)
for (lt=1LL*mi(a[i][i],mod-)*a[j][i]%mod,k=i;k<=n;++k)
a[j][k]=(a[j][k]-1LL*a[i][k]*lt%mod+mod)%mod;
}
if (i<n||!a[n][n]) puts("No"); else puts("Yes");
}
}
Last Word
行列式还能这么用.jpg。
这题又让小C了解了一下行列式的一点点性质,好像还练习了一下乱搞技巧?
小C的n^3做法似乎成功拿到了bzoj那一题的垫底:
[BZOJ]1059 矩阵游戏(ZJOI2007)的更多相关文章
- BZOJ 1059 矩阵游戏
Description 小Q是一个非常聪明的孩子,除了国际象棋,他还很喜欢玩一个电脑益智游戏--矩阵游戏.矩阵游戏在一个\(N \times N\)黑白方阵进行(如同国际象棋一般,只是颜色是随意的). ...
- BZOJ 1059 矩阵游戏 二分图匹配
题目链接: https://www.lydsy.com/JudgeOnline/problem.php?id=1059 题目大意: 小Q是一个非常聪明的孩子,除了国际象棋,他还很喜欢玩一个电脑益智游戏 ...
- [bzoj]1059矩阵游戏<二分图匹配*匈牙利算法>
题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=1059 初见此题,我觉得这是水题,我认为只要每一行和每一列至少存在一个黑格就可以出现对角线, ...
- 洛谷 P1129 BZOJ 1059 cogs 660 [ZJOI2007]矩阵游戏
题目描述 小Q是一个非常聪明的孩子,除了国际象棋,他还很喜欢玩一个电脑益智游戏――矩阵游戏.矩阵游戏在一个N*N黑白方阵进行(如同国际象棋一般,只是颜色是随意的).每次可以对该矩阵进行两种操作: 行交 ...
- 矩阵游戏|ZJOI2007|BZOJ1059|codevs1433|luoguP1129|二分图匹配|匈牙利算法|Elena
1059: [ZJOI2007]矩阵游戏 Time Limit: 10 Sec Memory Limit: 162 MB Description 小Q是一个非常聪明的孩子,除了国际象棋,他还很喜欢玩 ...
- BZOJ 3240 矩阵游戏
Description 婷婷是个喜欢矩阵的小朋友,有一天她想用电脑生成一个巨大的\(n\)行\(m\)列的矩阵(你不用担心她如何存储).她生成的这个矩阵满足一个神奇的性质:若用\(F[i][j]\)来 ...
- 【BZOJ】【1059】【ZJOI2007】矩阵游戏
二分图完美匹配/匈牙利算法 如果a[i][j]为黑点,我们就连边 i->j ,然后跑二分图最大匹配,看是否有完美匹配. <_<我们先考虑行变换:对于第 i 行,如果它第 j 位是黑点 ...
- bzoj 1059: [ZJOI2007]矩阵游戏 二分图匹配
1059: [ZJOI2007]矩阵游戏 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 1891 Solved: 919[Submit][Statu ...
- BZOJ 1059 [ZJOI2007]矩阵游戏
1059: [ZJOI2007]矩阵游戏 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 2707 Solved: 1322[Submit][Stat ...
随机推荐
- Linux 帳號管理與 ACL 權限設定
1. Linux 的账号与群组1.1 使用者识别: UID 与 GID1.2 使用者账号:/etc/passwd, /etc/shadow1.3 关于群组: 有效与初始群组. groups, newg ...
- MySQL InnoDB锁机制
概述: 锁机制在程序中是最常用的机制之一,当一个程序需要多线程并行访问同一资源时,为了避免一致性问题,通常采用锁机制来处理.在数据库的操作中也有相同的问题,当两个线程同时对一条数据进行操作,为了保证数 ...
- EasyUI 中easyui-textbox和easyui-searchbox文本框的点击事件。
html: <input id="txtsearch" class="easyui-textbox" data-options="buttonT ...
- machine learning 之 导论 一元线性回归
整理自Andrew Ng 的 machine learnig 课程 week1. 目录: 什么是机器学习 监督学习 非监督学习 一元线性回归 模型表示 损失函数 梯度下降算法 1.什么是机器学习 Ar ...
- B树和B+树的插入、删除图文详解
简介:本文主要介绍了B树和B+树的插入.删除操作.写这篇博客的目的是发现没有相关博客以举例的方式详细介绍B+树的相关操作,由于自身对某些细节也感到很迷惑,通过查阅相关资料,对B+树的操作有所顿悟,写下 ...
- C# 使用 GDI+ 画图
最近做一个微信公众号服务,有一些简单的图片处理功能.主要就是用户在页面操作,前端做一些立刻显示的效果,然后提交保存时后端真正修改原图. 我们的后端是 ASP.NET,也就是 C# 语言了,C# 本身处 ...
- Struts2 之值栈
值栈(ValueStack) http://www.cnblogs.com/bgzyy/p/8639893.html 这是我的有关 struts2 的第一篇文章,对于里面我们说到的一个 struts2 ...
- SpringCloud的Hystrix(一) 一个消费者内的两个服务监控
一.概念与定义 1.服务雪崩 在微服务架构中,整个系统按业务拆分出一个个服务,这些服务之间可以相互调用(RPC),为了保证服务的高可用,单个服务通常会集群部署. 但是由于网络原因或自身原因,服务并不能 ...
- 访问远程的docker
docker version vim /etc/default/docker DOCKER_OPTS=" -Label name=dockerserver2" docke ...
- python网络爬虫与信息提取 学习笔记day1
Day1: 安装python之后,为其配置requests第三方库,并爬取百度主页内容. 语句解释: r.status_code检测请求的状态码,如果状态码为200,则说明访问成功,否则,则说明访问失 ...