• BZOJ 3168 Luogu P4100 [HEOI2013]钙铁锌硒维生素 (矩阵求逆、二分图匹配)


    线性代数+图论好题。

    题目链接: (bzoj) https://www.lydsy.com/JudgeOnline/problem.php?id=3168

    (luogu) https://www.luogu.org/problemnew/show/P4100

    题解: 首先(A)矩阵必须满秩。有一个结论是,设矩阵(C)满足(CA=B), 则(A)的第(i)行可以被(B)的第(j)行来替代当且仅当(C_{j,i} e 0).

    (B_j)可以用(A)除了(i)之外的行向量线性表示,那么(B_j)无法替换(A_i). 若(C_{j,i}=0)代表用(A)矩阵的行向量表示(B_j)的系数向量中(A_i)这一项的系数为(0).

    那么(CA=B)可以推出(CAA^{-1}=BA^{-1}, C=BA^{-1})

    数学被各种吊打啊……

    然后我们就在(O(n^3))时间内求出了对于每一个(i,j), (A)中第(i)行是否能被(B)中第(j)行替换

    问题转化成了,给一张二分图,保证有完美匹配,求一个完美匹配使得(A)中的每个点在(B)中的匹配点构成的排列的字典序最小。

    这个东西,貌似必须用匈牙利算法。。先跑一边随便求出一个完美匹配,然后从(1)号到(n)号每个点再匹配尽量小的,如果能找到使当前(i)号点更小,且不影响(i)号点前面的点的交错路,那么就可以更新答案了。

    时间复杂度(O(n^3)).

    代码

    #include<cstdio>
    #include<cstdlib>
    #include<cstring>
    #include<cassert>
    #include<iostream>
    #define llong long long
    using namespace std;
    
    inline int read()
    {
    	int x=0; bool f=1; char c=getchar();
    	for(;!isdigit(c);c=getchar()) if(c=='-') f=0;
    	for(; isdigit(c);c=getchar()) x=(x<<3)+(x<<1)+(c^'0');
    	if(f) return x;
    	return -x;
    }
    
    const int N = 300;
    const int P = 942030731;
    llong quickpow(llong x,llong y)
    {
    	llong cur = x,ret = 1ll;
    	for(int i=0; y; i++)
    	{
    		if(y&(1ll<<i)) {y-=(1ll<<i); ret = ret*cur%P;}
    		cur = cur*cur%P;
    	}
    	return ret;
    }
    llong mulinv(llong x) {return quickpow(x,P-2);}
    struct Matrix
    {
    	llong a[N+3][N+3]; int n;
    	Matrix() {}
    	Matrix(int _n) {n = _n; for(int i=1; i<=n; i++) for(int j=1; j<=n; j++) a[i][j] = 0ll;}
    	void read(int _n)
    	{
    		n = _n;
    		for(int i=1; i<=n; i++) for(int j=1; j<=n; j++) scanf("%lld",&a[i][j]);
    	}
    	void write()
    	{
    		printf("%d
    ",n);
    		for(int i=1; i<=n; i++) {for(int j=1; j<=n; j++) printf("%lld ",a[i][j]); puts("");}
    	}
    	Matrix operator *(const Matrix &arg)
    	{
    		Matrix ret = Matrix(n);
    		for(int i=1; i<=n; i++)
    		{
    			for(int k=1; k<=n; k++)
    			{
    				for(int j=1; j<=n; j++)
    				{
    					ret.a[i][j] = (ret.a[i][j]+a[i][k]*arg.a[k][j])%P;
    				}
    			}
    		}
    		return ret;
    	}
    	Matrix inv()
    	{
    		Matrix ret = Matrix(n); for(int i=1; i<=n; i++) ret.a[i][i] = 1ll;
    		for(int i=1; i<=n; i++)
    		{
    			if(a[i][i]==0)
    			{
    				bool found = false;
    				for(int j=i+1; j<=n; j++)
    				{
    					if(a[j][i])
    					{
    						for(int k=1; k<=n; k++) {swap(a[i][k],a[j][k]),swap(ret.a[i][k],ret.a[j][k]);}
    						found = true; break;
    					}
    				}
    				if(found==false) {ret.a[0][0] = P; return ret;}
    			}
    			for(int j=i+1; j<=n; j++)
    			{
    				llong coe = (P-a[j][i]*mulinv(a[i][i])%P)%P;
    				for(int k=1; k<=n; k++)
    				{
    					a[j][k] = (a[j][k]+coe*a[i][k])%P;
    					ret.a[j][k] = (ret.a[j][k]+coe*ret.a[i][k])%P;
    				}
    			}
    		}
    		for(int i=1; i<=n; i++)
    		{
    			llong coe = mulinv(a[i][i]);
    			for(int j=1; j<=n; j++) {a[i][j] = a[i][j]*coe%P; ret.a[i][j] = ret.a[i][j]*coe%P;}
    		}
    //		write();
    //		ret.write();
    		for(int i=n; i>=1; i--)
    		{
    			for(int j=n; j>=i+1; j--)
    			{
    				llong coe = (P-a[i][j]*mulinv(a[j][j])%P)%P;
    				a[i][j] = 0ll;
    				for(int k=1; k<=n; k++) ret.a[i][k] = (ret.a[i][k]+coe*ret.a[j][k])%P;
    			}
    		}
    		return ret;
    	}
    } a,b,aux,c;
    int g[N+3][N+3];
    int vis[N+3];
    int match1[N+3],match2[N+3];
    int n;
    
    bool dfs1(int u)
    {
    	for(int i=1; i<=n; i++)
    	{
    		if(g[i][u]==true && vis[i]==false)
    		{
    			vis[i] = true;
    			if(match2[i]==0 || dfs1(match2[i])==true)
    			{
    				match2[i] = u; match1[u] = i;
    				return true;
    			}
    		}
    	}
    	return false;
    }
    
    bool dfs2(int u,int u0)
    {
    	for(int i=1; i<=n; i++)
    	{
    		if(g[i][u]==true && vis[i]==false)
    		{
    			vis[i] = true;
    			if(match2[i]==u0 || (match2[i]>u0 && dfs2(match2[i],u0)==true))
    			{
    				match2[i] = u; match1[u] = i;
    				return true;
    			}
    		}
    	}
    	return false;
    }
    
    int main()
    {
    	scanf("%d",&n);
    	a.read(n); b.read(n);
    	aux = a.inv();
    	if(aux.a[0][0]==P) {printf("NIE"); return 0;}
    	c = b*aux;
    	for(int i=1; i<=n; i++)
    	{
    		for(int j=1; j<=n; j++) g[i][j] = c.a[i][j]==0 ? 0 : 1;
    	}
    //	for(int i=1; i<=n; i++) {for(int j=1; j<=n; j++) printf("%d",g[i][j]); puts("");}
    	int mf = 0;
    	for(int i=1; i<=n; i++)
    	{
    		for(int j=1; j<=n; j++) vis[j] = false;
    		mf += dfs1(i);
    	}
    	if(mf<n) {printf("NIE"); return 0;}
    	printf("TAK
    ");
    	for(int i=1; i<=n; i++)
    	{
    		for(int j=1; j<=n; j++) vis[j] = false;
    		dfs2(i,i);
    	}
    	for(int i=1; i<=n; i++) printf("%d
    ",match1[i]);
    	return 0;
    }
    
  • 相关阅读:
    0.嵌入式系统 Boot Loader 技术内幕
    JAVA_SE基础——25.面向对象练习
    JAVA_SE基础——24.面向对象的内存分析
    JAVA_SE基础——23.类的定义
    深入理解java的static关键字
    JAVA_SE基础——22.面向对象的概念
    JAVA_SE基础——21.二维数组的定义
    Java常用排序算法/程序员必须掌握的8大排序算法
    JAVA_SE基础——20.数组的常见操作
    JAVA_SE基础——19.数组的定义
  • 原文地址:https://www.cnblogs.com/suncongbo/p/11155155.html
Copyright © 2020-2023  润新知