zoukankan      html  css  js  c++  java
  • 【知识点】高斯消元&线性基

    高斯消元:

    解$n$元一次方程组的通用方法,大部分时候用于解决没有明显转移顺序的dp。

    考虑将方程组列成一个$n imes (n+1)$的矩阵$A$,然后依次枚举每一个未知数$j$(第$j$列):

    1. 从上往下找到第一个$i$,满足$igeq j,A_{i,j} eq 0$。
    2. 如果找不到则该方程组无解,退出。否则把第$i$行与第$j$行交换。
    3. 对于任意$k eq j$,用第$j$行消第$k$行,使其满足$A_{k,j}=0$。

    显然最后消出的是一个对角线矩阵(只在$A_{i,i}$处不为0),此时方程组的解$x_{i}=frac{A_{i,n+1}}{A_{i,i}}$。

    复杂度$O(n^{3})$。

    #include<bits/stdc++.h>
    #define maxn 505
    #define maxm 500005
    #define inf 0x7fffffff
    #define ll long long
    #define rint register int
    #define debug(x) cerr<<#x<<": "<<x<<endl
    #define fgx cerr<<"--------------"<<endl
    #define dgx cerr<<"=============="<<endl
    
    using namespace std;
    int n; double A[maxn][maxn];
    
    inline int read(){
        int x=0,f=1; char c=getchar();
        for(;!isdigit(c);c=getchar()) if(c=='-') f=-1;
        for(;isdigit(c);c=getchar()) x=x*10+c-'0';
        return x*f;
    }
    
    inline bool Gauss(){
        for(int j=1;j<=n;j++){
            for(int i=j;i<=n;i++){
                if(A[i][j]==0) continue;
                for(int k=1;k<=n+1;k++) swap(A[i][k],A[j][k]); 
                break;
            }
            if(A[j][j]==0) return 0;
            for(int i=1;i<=n;i++){
                if(i==j || A[i][j]==0) continue;
                double x=A[i][j]/A[j][j];
                for(int k=1;k<=n+1;k++) A[i][k]-=A[j][k]*x;
            }
        }
        return 1;
    }
    
    int main(){
        n=read();
        for(int i=1;i<=n;i++)
            for(int j=1;j<=n+1;j++)
                A[i][j]=read();
        if(!Gauss()) printf("No Solution
    ");
        else for(int i=1;i<=n;i++) printf("%.2lf
    ",A[i][n+1]/A[i][i]);
        return 0;
    }
    高斯消元

    线性基:

    用若干个线性基底表示一个向量空间,一般用来解决异或问题。

    例如:你有一个序列A,支持插入操作和查询操作,每次询问在A中选一个子集的最大异或和是多少。

    考虑把序列中的每个数拆成m位二进制,构成一个$n imes m$的01矩阵,我们只要把这个矩阵消成一个上三角矩阵就行了。

    用高斯消元写的话每次插入都要重构一遍,不够优秀。

    考虑不需要重构的支持插入的做法,设当前插入的数为x,从高位到低位枚举二进制位p:

    • 如果x在p这一位为0则跳过。
    • 如果x在p这一位不为0且$B_{p}$有值,则令$xoplus B_p$。
    • 否则,令$B_{p}=x$,结束枚举。

    这样消出的是一个可能有重复列的上三角矩阵。

    如果希望消出一个没有重复列的上三角矩阵,只需要在结束枚举时:

    • 对于所有$i<p$,用$B_i$更新$B_{p}$,使$B_p$与$B_i$没有重复列。
    • 对于所有$i>p$,用$B_{p}$更新$B_{i}$,使$B_i$与$B_p$没有重复列。

    复杂度$O(nlog{a_{i}})$。

    #include<algorithm>
    #include<iostream>
    #include<cstring>
    #include<cstdio>
    #define maxn 100005
    #define maxm 50
    #define inf 0x7fffffff
    #define ll long long
    using namespace std;
    ll N,A[maxn],p[maxm+5];
    inline ll read(){
        ll x=0,f=1; char c=getchar();
        for(;!isdigit(c);c=getchar()) if(c=='-') f=-1;
        for(;isdigit(c);c=getchar()) x=x*10+c-'0';
        return x*f;
    }
    int main(){
        N=read(); for(ll i=1;i<=N;i++) A[i]=read();
        for(ll i=1;i<=N;i++) for(ll j=maxm;j>=0;j--)
            if(A[i]&(1LL<<j)){
                if(!p[j]){ p[j]=A[i];
                    for(ll k=j-1;k>=0;k--) if(p[k]&&(p[j]&(1LL<<k))) p[j]^=p[k];
                    for(ll k=j+1;k<=maxm;k++) if(p[k]&(1LL<<j)) p[k]^=p[j];
                break; } else A[i]^=p[j];
            } ll ans=0;
        for(ll i=maxm;i>=0;i--)
            if((ans^p[i])>ans) ans^=p[i];
        printf("%lld
    ",ans);    
        return 0;
    }
    线性基

    树上高斯消元:

    设$f_i =a_i f_{father_{i}}+b_i$,从下往上递推即可。

  • 相关阅读:
    学习HTML-第一章HTML基础
    javascript中apply 和 call 的区别
    数据库
    贝叶斯公式的直观理解(先验概率/后验概率)
    4-决策树
    消息队列
    RESTful api 设计规范
    MySql索引
    MySQL并发控制
    MySQL三层逻辑架构
  • 原文地址:https://www.cnblogs.com/YSFAC/p/13224003.html
Copyright © 2011-2022 走看看