IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> C++知识库 -> 线性代数代码实现(四)行列式与可逆矩阵的判定(c++) -> 正文阅读

[C++知识库]线性代数代码实现(四)行列式与可逆矩阵的判定(c++)

前言:

? ? ? ? 上一篇文章中,我们探讨了矩阵的秩与可逆矩阵的判定方法,我提到将说出下一种判断矩阵是否可逆的方法,这篇文章就跟大家探讨一下。

一、线性代数知识回顾:

1.? 排列:由1,2,···,n 这n个数组成的有序的数组就是一个 n?阶排列

2.? 逆序数:在一个排列中,如果有两个数 a 和 b,若满足 a 在 b 的前面,且 a 大于 b ,那么它称为一个逆序,一个排列的所有逆序的个数称为逆序数。

i_{1}i_{2}\cdot \cdot \cdot i_{n}?为一个 n 阶排列,记\tau (i_{1}i_{2}\cdot \cdot \cdot i_{n})?为该排列的逆序数

3.? 行列式:对于一个 n 阶方阵?A=(a_{ij})_{n\times n} ,记?\left | A \right |?或 det(A)?为?A?的行列式,它的定义如下:

\left | A \right |?等于所有取自不同行不同列的 n 个元素的乘积

a_{1j_{1}}a_{2j_{2}}\cdot \cdot \cdot a_{nj_{n}}

的代数和,记?j_{1}j_{2}\cdot \cdot \cdot j_{n}?是 1,2,···,n 的一个排列,行列式的定义可写为:

?\left | A \right |=\begin{vmatrix } a_{11}&a_{12} &\cdot \cdot \cdot & a_{1n}\\ a_{21}&a_{22} &\cdot \cdot \cdot &a_{2n} \\ \cdot \cdot \cdot & \cdot \cdot \cdot & \cdot \cdot \cdot &\cdot \cdot \cdot \\ a_{n1}&a_{n2} &\cdot \cdot \cdot &a_{nn} \end{vmatrix} =\sum_{j_{1}j_{2}\cdot \cdot \cdot j_{n}}^{} (-1)^{\tau (j_{1}j_{2}\cdot \cdot \cdot j_{n})} a_{1j_{1}}a_{2j_{2}}\cdot \cdot \cdot a_{nj_{n}}

?有了定义后,我们固然是可以写出代码计算行列式的,我们可以看到行列式的计算需要找到所有的取自不同行不同列的元素乘积,如果需要代码实现,那么我们需要找到 n 阶排列的所有情况(n? ?阶全排列),而 n 阶全排列总共有 n ! 个,当矩阵比较大时,我们的程序就需要执行很多次,相当耗费时间。因此,我们不这样求解行列式。我们给出另外一种方法求解行列式,这需要以下线性代数知识:

4.? 定理

交换行列式两行,行列式反号

把一行的倍数加到另一行,行列式不变

5.? 上三角行列式:主对角线以下元素为 0 的行列式,上三角行列式的值等于主对角线元素的成绩。设??A=(a_{ij})_{n\times n}?为上三角行列式,则:

\left | A \right |=a_{11}a_{22}\cdot \cdot \cdot a_{nn}

也就是说,将行列式通过上面第 4 条定理所说的两条化为上三角行列式,记录交换行列式两行的次数,然后通过第 5 条就可以求解出行列式的值了,这样做可以处理较大阶的行列式!下面再给出一个定理说明可逆与行列式的关系:

6.? 定理

矩阵可逆的充要条件是它的行列式不等于零

有了这个定理,我们就可以给出第二个判断矩阵是否可逆的方法了!

二、算法设计:

1.? 求解行列式:(与我写的第二篇文章将矩阵化为上三角矩阵十分相似,没看过第二篇文章的伙伴们可以点击这个链接:

线性代数代码实现(二)上三角矩阵(C++)_AI研究者的博客-CSDN博客

在将矩阵化为上三角矩阵的算法中增加三条:

1)令 result?= 1,每次交换不同的两行时,执行 result?= -result

2)在外层循环最后执行?result*=mat[i][i],计算上三角矩阵主对角线元素的乘积,并判断result是否为零,若果为零,则说明行列式为零,则退出循环,若不为零,则继续循环

3)循环结束后,执行 result*=mat[m][n] ,乘上主对角线最后一个元素(这里 m = n),返回result

2.? 判断矩阵是否可逆:

如果矩阵不是方阵,则不可逆,否则计算出行列式的值, 如果不为零,则矩阵可逆(非奇异),否则,矩阵不可逆(奇异)

三、代码实现:

类的定义:

class Mat
{
public:
	int m = 1, n = 1; //行数和列数
	double mat[N][N] = { 0 };  //矩阵开始的元素

	Mat() {}
	Mat(int mm, int nn)
	{
		m = mm; n = nn;
	}
	void create();//创建矩阵
	void copy(Mat& a);//复制

	double det();//计算行列式
	bool judgeinv();//判断矩阵是否可逆
};

?det()函数求解矩阵的行列式:

double Mat::det()
{
	if (m != n)
	{
		cout << "不是方阵,无行列式" << endl;
		exit(1);
	}
	Mat a; copy(a);
	double result = 1;
	for (int i = 1; i < m; i++)
	{
		//寻找第 i 列不为零的元素
		int k;
		for (k = i; k <= m; k++)
		{
			if (fabs(mat[k][i]) > 1e-10) //满足这个条件时,认为这个元素不为0
				break;
		}
		if (k <= m)//说明第 i 列有不为0的元素
		{
			if (k != i)//说明需要交换不同的两行
			{
				//交换第 i 行和第 k 行所有元素
				for (int j = i; j <= n; j++)//从第 i 个元素交换即可,因为前面的元素都为0
				{//使用mat[0][j]作为中间变量交换元素
					mat[0][j] = mat[i][j]; mat[i][j] = mat[k][j]; mat[k][j] = mat[0][j];
				}
				result = -result;
			}
			double a;//倍数
			for (int j = i + 1; j <= m; j++)
			{
				a = -mat[j][i] / mat[i][i];
				for (k = i; k <= n; k++)
				{
					mat[j][k] += a * mat[i][k];//第 i 行 a 倍加到第 j 行
				}
			}
		}
		result *= mat[i][i];
		if (fabs(result) < 1e-10) //如果为零,则已经可以说明行列式为零了,没必要继续算下去了
			break;
	}
	result *= mat[m][n];
	return result;
}

judgeinv()函数判断矩阵是否可逆:

bool Mat::judgeinv()
{
	if (m != n)//不是方阵,不可逆
		return false;
	if (fabs(det()) > 1e-10)//行列式不等于 0
		return true;
	else
		return false;
}

有兴趣的同学也可以自己利用定义做一做求解行列式的函数,主要难点在于如何知道全排列,有兴趣的同学可以想一想。计算行列式当然也有其他方法,这篇文章中介绍的方法算是比较好的方法,大家也可以研究其他方法,同样,判断矩阵是否可逆也有其他方法,大家也同样可以研究研究。包括线性代数的知识,大家都可以仔细思考,实际上线性代数的很多知识都有很好的几何意义,例如我们讲过的矩阵乘法,行列式计算等都是有十分好的几何意义,大家可以去理解理解。(悄悄告诉大家:n 阶矩阵的行列式的绝对值实际上就是 n 维空间中平行多面体的体积)这里附上完整代码:

#include<iostream>
#include <cmath>
#define N 10
using namespace std;
class Mat
{
public:
	int m = 1, n = 1; //行数和列数
	double mat[N][N] = { 0 };  //矩阵开始的元素

	Mat() {}
	Mat(int mm, int nn)
	{
		m = mm; n = nn;
	}
	void create();//创建矩阵
	void copy(Mat& a);//复制

	double det();//计算行列式
	bool judgeinv();//判断矩阵是否可逆
};

void Mat::create()
{
	for (int i = 1; i <= m; i++)
	{
		for (int j = 1; j <= n; j++)
		{
			cin >> mat[i][j];
		}
	}
}
void Mat::copy(Mat& a)//复制函数
{
	a.m = m; a.n = n;
	for (int i = 1; i <= m; i++)
	{
		for (int j = 1; j <= n; j++)
		{
			a.mat[i][j] = mat[i][j];
		}
	}
}
double Mat::det()
{
	if (m != n)
	{
		cout << "不是方阵,无行列式" << endl;
		exit(1);
	}
	Mat a; copy(a);
	double result = 1;
	for (int i = 1; i < m; i++)
	{
		//寻找第 i 列不为零的元素
		int k;
		for (k = i; k <= m; k++)
		{
			if (fabs(mat[k][i]) > 1e-10) //满足这个条件时,认为这个元素不为0
				break;
		}
		if (k <= m)//说明第 i 列有不为0的元素
		{
			if (k != i)//说明需要交换不同的两行
			{
				//交换第 i 行和第 k 行所有元素
				for (int j = i; j <= n; j++)//从第 i 个元素交换即可,因为前面的元素都为0
				{//使用mat[0][j]作为中间变量交换元素
					mat[0][j] = mat[i][j]; mat[i][j] = mat[k][j]; mat[k][j] = mat[0][j];
				}
				result = -result;
			}
			double a;//倍数
			for (int j = i + 1; j <= m; j++)
			{
				a = -mat[j][i] / mat[i][i];
				for (k = i; k <= n; k++)
				{
					mat[j][k] += a * mat[i][k];//第 i 行 a 倍加到第 j 行
				}
			}
		}
		result *= mat[i][i];
		if (fabs(result) < 1e-10) //如果为零,则已经可以说明行列式为零了,没必要继续算下去了
			break;
	}
	result *= mat[m][n];
	return result;
}
bool Mat::judgeinv()
{
	if (m != n)//不是方阵,不可逆
		return false;
	if (fabs(det()) > 1e-10)//行列式不等于 0
		return true;
	else
		return false;
}

int main()
{
	Mat a(3, 3);
	cout << "请输入 " << a.m << "*" << a.n << " 的矩阵:" << endl;
	a.create();
	cout << "矩阵的行列式为:  " << a.det() << endl;
	if (a.judgeinv())
		cout << "可逆!" << endl;
	else
		cout << "不可逆!" << endl;
	return 0;
}

若有不足之处,欢迎大家指正!

  C++知识库 最新文章
【C++】友元、嵌套类、异常、RTTI、类型转换
通讯录的思路与实现(C语言)
C++PrimerPlus 第七章 函数-C++的编程模块(
Problem C: 算法9-9~9-12:平衡二叉树的基本
MSVC C++ UTF-8编程
C++进阶 多态原理
简单string类c++实现
我的年度总结
【C语言】以深厚地基筑伟岸高楼-基础篇(六
c语言常见错误合集
上一篇文章      下一篇文章      查看所有文章
加:2021-10-20 12:19:11  更:2021-10-20 12:19:41 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/1 13:47:30-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码