P7112 【模板】行列式求值 题解(高斯消元+辗转相除)

题目链接

题目思路

先介绍行列式的变化对于行列式值的改变

1 如果交换矩阵的两行,则行列式的符号要取反

2 如果用某数t乘以矩阵的一行,则行列式等于原行列式的t倍

3 三角阵的行列式等于对角线元素乘积

4 从矩阵的行k减去行i的l倍,行列式不会改变,即消元过程不改变行列式

根据上面的式子用高斯消元化简即可

注意这里不需要求逆元,由于行列的相互加减不会影响行列式的值

所以使用辗转相除法

代码

#include<bits/stdc++.h>
#define fi first
#define se second
#define debug cout<<"I AM HERE"<<endl;
using namespace std;
typedef long long ll;
const int maxn=600+5,inf=0x3f3f3f3f4;
const double eps=1e-6;
const ll INF=0x3f3f3f3f3f3f3f3f;
int n,mod;
ll a[maxn][maxn];
ll gs(int n){
    ll ans=1;
    bool flag=1;
    for(int i=1,pos;i<=n;i++){
        for(pos=i;pos<=n;pos++){
            if(a[pos][i]){
                break;
            }
        }
        if(pos==n+1){
            return 0;
        }
        if(pos!=i){
            flag^=1;
            swap(a[pos],a[i]);
        }
        for(int j=i+1;j<=n;j++){
            if(a[i][i]>a[j][i]){
                swap(a[i],a[j]);
                flag^=1;
            }
            while(a[j][i]){
                swap(a[i],a[j]);
                flag^=1;
                ll p=a[j][i]/a[i][i];
                for(int k=i;k<=n;k++){
                    a[j][k]=(a[j][k]+(mod-p)*a[i][k])%mod;
                }
            }
        }
        ans=ans*a[i][i]%mod;
    }
    if(!flag){
        ans=(-ans+mod)%mod;
    }
    return ans;
}
signed main(){
    scanf("%d%d",&n,&mod);
    for(int i=1;i<=n;i++){
        for(int j=1;j<=n;j++){
            scanf("%lld",&a[i][j]);
        }
    }
    ll ans=gs(n);
    printf("%lld\n",ans);
    return 0;
}
posted @ 2021-11-23 21:45  hunxuewangzi  阅读(208)  评论(0编辑  收藏  举报