消元
参考资料
简介
高斯消元法(Gaussian Elimination)是求解线性方程组的经典算法,它在当代数学中有着重要的地位和价值,是线性代数课程教学的重要组成部分。
算法对比
| 算法名称 | 时间复杂度 | 需要回代 | 区分无解和无穷多解 | 代码长度 |
|---|---|---|---|---|
| 高斯–约旦消元法 | 否 | 困难 | 约 行 | |
| 高斯消元法 | 是 | 容易 | 约 行 |
高斯–约旦消元法
bool gauss(int n)
{
for(int i=1;i<=n;i++)
{
int t=i;
for(int j=i+1;j<=n;j++)
{
if(fabs(a[j][i])>fabs(a[t][i]))t=j;
}
if(fabs(a[t][i])<eps)return 0;
swap(a[i],a[t]);
for(int j=n+1;j>=i;j--)a[i][j]/=a[i][i];
for(int j=1;j<=n;j++)
{
if(i==j)continue;
for(int k=n+1;k>i;k--)a[j][k]-=a[i][k]*a[j][i];
}
}
return 1;
}
高斯消元法
无唯一解:某行前 个数均为 。():
- 无解:最后结果不为 。()
- 无穷多解:最后结果为 。()
int gauss(int n)
{
int r,c;
for(r=1,c=1;r<=n&&c<=n;c++)
{
int t=r;
for(int j=r+1;j<=n;j++)
{
if(fabs(a[j][c])>fabs(a[t][c]))t=j;
}
swap(a[r],a[t]);
if(fabs(a[r][c])<eps)continue;
for(int j=r+1;j<=n;j++)
{
if(fabs(a[j][c])<eps)continue;
double f=a[j][c]/a[r][c];
for(int k=c;k<=n+1;k++)a[j][k]-=a[r][k]*f;
a[j][c]=0;
}
r++;
}
for(int i=r;i<=n;i++)
{
if(fabs(a[i][n+1])>eps)return -1;
}
if(r<=n)return n-r+1;
for(int i=n;i>=1;i--)
{
for(int j=i+1;j<=n;j++)a[i][n+1]-=a[i][j]*a[j][0];
a[i][0]=a[i][n+1]/a[i][i];
}
return 0;
}
行列式求值
#include <bits/stdc++.h>
using namespace std;
using ll=long long;
const int N=605;
ll a[N][N];
int main()
{
ios::sync_with_stdio(false);
cin.tie(nullptr);
int n,mod;
cin>>n>>mod;
for(int i=1;i<=n;i++)
{
for(int j=1;j<=n;j++)
{
cin>>a[i][j];
}
}
ll ans=1;
for(int i=1;i<=n;i++)
{
for(int j=i+1;j<=n;j++)
{
while(a[i][i])
{
int f=a[j][i]/a[i][i];
for(int k=i;k<=n;k++)
{
a[j][k]=(a[j][k]-a[i][k]*f%mod)%mod;
}
swap(a[i],a[j]);
ans=-ans;
}
swap(a[i],a[j]);
ans=-ans;
}
}
for(int i=1;i<=n;i++)ans=ans*a[i][i]%mod;
cout<<(ans+mod)%mod<<'\n';
return 0;
}
例题
给定一个线性方程组,对其求解。
Code (1)
#include <bits/stdc++.h>
using namespace std;
const double eps=1e-8;
const int N=105;
double a[N][N];
bool gauss(int n)
{
for(int i=1;i<=n;i++)
{
int t=i;
for(int j=i+1;j<=n;j++)
{
if(fabs(a[j][i])>fabs(a[t][i]))t=j;
}
if(fabs(a[t][i])<eps)return 0;
swap(a[i],a[t]);
for(int j=n+1;j>=i;j--)a[i][j]/=a[i][i];
for(int j=1;j<=n;j++)
{
if(i==j)continue;
for(int k=n+1;k>i;k--)a[j][k]-=a[i][k]*a[j][i];
}
}
return 1;
}
int main()
{
ios::sync_with_stdio(false);
cin.tie(nullptr);
int n;
cin>>n;
for(int i=1;i<=n;i++)
{
for(int j=1;j<=n+1;j++)
{
cin>>a[i][j];
}
}
if(gauss(n)==0)
{
cout<<"No Solution"<<'\n';
return 0;
}
for(int i=1;i<=n;i++)
{
cout<<fixed<<setprecision(2)<<a[i][n+1]<<'\n';
}
return 0;
}
已知 元线性一次方程组。
请根据输入的数据,编程输出方程组的解的情况。Code (1)
#include <bits/stdc++.h>
using namespace std;
const double eps=1e-8;
const int N=55;
double a[N][N];
int gauss(int n)
{
int r,c;
for(r=1,c=1;r<=n&&c<=n;c++)
{
int t=r;
for(int j=r+1;j<=n;j++)
{
if(fabs(a[j][c])>fabs(a[t][c]))t=j;
}
swap(a[r],a[t]);
if(fabs(a[r][c])<eps)continue;
for(int j=r+1;j<=n;j++)
{
if(fabs(a[j][c])<eps)continue;
double f=a[j][c]/a[r][c];
for(int k=c;k<=n+1;k++)a[j][k]-=a[r][k]*f;
a[j][c]=0;
}
r++;
}
for(int i=r;i<=n;i++)
{
if(fabs(a[i][n+1])>eps)return -1;
}
if(r<=n)return n-r+1;
for(int i=n;i>=1;i--)
{
for(int j=i+1;j<=n;j++)a[i][n+1]-=a[i][j]*a[j][0];
a[i][0]=a[i][n+1]/a[i][i];
}
return 0;
}
int main()
{
ios::sync_with_stdio(false);
cin.tie(nullptr);
int n;
cin>>n;
for(int i=1;i<=n;i++)
{
for(int j=1;j<=n+1;j++)
{
cin>>a[i][j];
}
}
int ans=gauss(n);
if(ans==-1){cout<<-1<<'\n';return 0;}
if(ans>0){cout<<0<<'\n';return 0;}
for(int i=1;i<=n;i++)
{
cout<<fixed<<setprecision(2)<<'x'<<i<<'='<<a[i][0]<<'\n';
}
return 0;
}
给定一个 阶行列式 ,求 。结果对 取模。Code (1)
#include <bits/stdc++.h>
using namespace std;
using ll=long long;
const int N=605;
ll a[N][N];
int main()
{
ios::sync_with_stdio(false);
cin.tie(nullptr);
int n,mod;
cin>>n>>mod;
for(int i=1;i<=n;i++)
{
for(int j=1;j<=n;j++)
{
cin>>a[i][j];
}
}
ll ans=1;
for(int i=1;i<=n;i++)
{
for(int j=i+1;j<=n;j++)
{
while(a[i][i])
{
int f=a[j][i]/a[i][i];
for(int k=i;k<=n;k++)
{
a[j][k]=(a[j][k]-a[i][k]*f%mod)%mod;
}
swap(a[i],a[j]);
ans=-ans;
}
swap(a[i],a[j]);
ans=-ans;
}
}
for(int i=1;i<=n;i++)ans=ans*a[i][i]%mod;
cout<<(ans+mod)%mod<<'\n';
return 0;
}
给定一个有 个点, 条边的无向图(保证连通),每条边上各有一个阻值不同的电阻,求点 到点 的总电阻,,注意是多测。 给定一张 个节点 条边的无向图,边权表示电阻阻值,求节点 到 的等效电阻。 设节点 的电势为 ,边 的电阻为 ,则电导为: 根据 欧姆定律,从 到 的电流为: 根据 基尔霍夫电流定律,对于任意一个普通节点,所有关联支路电流的代数和为 。 考虑节点 的电流: 展开可得: 这是一个关于各点电势的线性方程。 考虑添加一条边 ,设其电导为 。节点 的方程加入 ,节点 的方程加入 ,因此对应到系数矩阵就是: 把所有边的贡献都加到矩阵里,就得到了整个线性方程组。 我们可以在节点 通入 电流,并将节点 设为零电势点,求出节点 的电势。 利用 高斯消元 求解各点电势后,等效电阻为:Solution
题意简述
参考资料
解题思路
参考代码
#include <bits/stdc++.h>
using namespace std;
const double eps=1e-8;
const int N=105;
double a[N][N];
bool gauss(int n)
{
for(int i=1;i<=n;i++)
{
int t=i;
for(int j=i+1;j<=n;j++)
{
if(fabs(a[j][i])>fabs(a[t][i]))t=j;
}
if(fabs(a[t][i])<eps)return 0;
swap(a[i],a[t]);
for(int j=n+1;j>=i;j--)a[i][j]/=a[i][i];
for(int j=1;j<=n;j++)
{
if(i==j)continue;
for(int k=n+1;k>i;k--)a[j][k]-=a[i][k]*a[j][i];
}
}
return 1;
}
int main()
{
ios::sync_with_stdio(false);
cin.tie(nullptr);
int n,m;
while(cin>>n>>m)
{
while(m--)
{
int u,v;
double r;
cin>>u>>v>>r;
double g=1/r;
a[u][u]+=g;
a[u][v]-=g;
a[v][v]+=g;
a[v][u]-=g;
}
a[1][n+1]=1;
for(int i=1;i<n;i++)a[n][i]=0;
a[n][n]=1;
a[n][n+1]=0;
gauss(n);
cout<<fixed<<setprecision(2)<<a[1][n+1]<<'\n';
memset(a,0,sizeof a);
}
return 0;
}
给定一张图,表示一个电阻电路,求等效电阻。Code (1)
#include <bits/stdc++.h>
using namespace std;
const double eps=1e-8;
const int N=105;
double a[N][N];
bool gauss(int n)
{
for(int i=1;i<=n;i++)
{
int t=i;
for(int j=i+1;j<=n;j++)
{
if(fabs(a[j][i])>fabs(a[t][i]))t=j;
}
if(fabs(a[t][i])<eps)return 0;
swap(a[i],a[t]);
for(int j=n+1;j>=i;j--)a[i][j]/=a[i][i];
for(int j=1;j<=n;j++)
{
if(i==j)continue;
for(int k=n+1;k>i;k--)a[j][k]-=a[i][k]*a[j][i];
}
}
return 1;
}
int main()
{
ios::sync_with_stdio(false);
cin.tie(nullptr);
int n,m;
cin>>n>>m;
while(m--)
{
int u,v;
double r;
cin>>u>>v>>r;
double g=1/r;
a[u][u]+=g;
a[u][v]-=g;
a[v][v]+=g;
a[v][u]-=g;
}
a[1][n+1]=1;
for(int i=1;i<n;i++)a[n][i]=0;
a[n][n]=1;
a[n][n+1]=0;
gauss(n);
cout<<fixed<<setprecision(2)<<a[1][n+1]<<'\n';
return 0;
}