程序猿成长之路之番外篇——矩阵算法

news2024/11/14 0:56:07

今天在复习线性代数知识的过程中,用java语言简单实现了一下矩阵算法。

数学知识回顾

1.什么是矩阵
在数学领域,矩阵就像一个表格,将数据排放进去,形成一个矩形。我们习惯用一个大括号把矩阵内的数据包括进来。

1.矩阵
在数学领域,矩阵就像一个表格,将数据排放进去,形成一个矩形。我们习惯用一个大括号把矩阵内的数据包括进来。

2. 矩阵的运算
矩阵可以进行加法、乘法运算,如果是个方形矩阵也可以转置或者求逆。此外,加法和乘法都有结合律、分配律等定律。

矩阵加法运算:
在这里插入图片描述
矩阵乘法运算:
运算规则相对较为繁琐:

  1. A(m,n) * B(n,k) = C(m,k) // 具有m行n列的矩阵A 乘以具有n行k列的矩阵B结果为m行k列的矩阵c
  2. 运算过程如下图:就拿C(1,1) – 结果矩阵的第一行第一列的元素来说,它等于A(1,1) * B(1,1) + A(1,2) * B(2,1) = 1 * 2 + 2 * 6 = 14
    在这里插入图片描述
  3. 矩阵的转置
    在这里插入图片描述
  4. 矩阵求逆
    我们知道伴随矩阵 = Det(矩阵A) * 矩阵的逆
    我们又知道 伴随矩阵 = Σ(Aij 的代数余子式), i,j∈(0,size))
    因此可以求出矩阵的逆 = |伴随矩阵| / Det(矩阵A)
    求逆的难点在于获取代数余子式。

算法实现

  1. 编写matrix类
package matrixUtils;

import java.util.Arrays;

class Matrix {
	@Override
	public String toString() {
		String str = "";
		for (int i = 0; i < arr.length;i++) {
			str += Arrays.toString(arr[i]);
			str +="\n";
		}
		return str;
	}
	//宽度
	private final int width;
	//高度
	private final int height;
	//数组
	private final double[][] arr;
	
	Matrix(int width, int height) {
		this.width = width;
		this.height = height;
		arr = new double[height][width];
	}
	
	Matrix(int width, int height, double[][] arr) {
		this.width = width;
		this.height = height;
		this.arr = arr;
	}
	
	public int getWidth() {
		return width;
	}
	public int getHeight() {
		return height;
	}
	public double[][] getArr() {
		return arr;
	}
	public void setArrVal(int x, int y, double val) {
		arr[x][y] = val;
	}
	public boolean compareTo(Matrix matrix) {
		//比较
		return this.width == matrix.width 
				& this.height == matrix.height;
	}
	public void setArrVals(int startIndex, int[][] val) {
		int size = val[0].length;
		size *= val.length;
		if (startIndex < 0 || size <= 0){
			return;
		}
		//count计数器
		for (int i = 0; i < size;i++) {
			arr[i / val[0].length][i % val[0].length] = val[i/val[0].length][i % val[0].length];
		}
	}
	
}
  1. 编写工厂类
public class MatrixFactory {
	private static final int MAX_SIZE = 1 << 30;
	/**
	 * 获取实例
	 */
	public static Matrix getInstance(int width, int height) {
		if (width < 0 || width > MAX_SIZE) {
			throw new IllegalArgumentException("宽度有误");
		}
		if (height < 0 || height > MAX_SIZE) {
			throw new IllegalArgumentException("高度有误");
		}
		return new Matrix(width, height);
	}
	
	public static Matrix getInstance(int width, int height, double[][] arr) {
		if (width < 0 || width > MAX_SIZE) {
			throw new IllegalArgumentException("宽度有误");
		}
		if (height < 0 || height > MAX_SIZE) {
			throw new IllegalArgumentException("高度有误");
		}
		return new Matrix(width, height, arr);
	}
}
  1. 矩阵乘法
    设计思路:设置一个计数器m,用于获取结果的列值,当m == matrix2的宽度时,也就是说当前已经完成对i行的处理,结果保存进matrix中,让i(行数)加1并且m重置为0,否则让结果列数自加。
	/**
	 * 矩阵相乘
	 * @param matrix
	 * @return
	 */
	public static Matrix multiply(Matrix matrix1,Matrix matrix2) {
		//生成一个新的矩阵
		if (matrix1 == null || matrix2 == null) {
			throw new IllegalArgumentException("矩阵不匹配,请检查");
		}
		if (matrix1.getWidth() != matrix2.getHeight()) {
			throw new IllegalArgumentException("矩阵不匹配,请检查");
		}
		//新建一个矩阵
		Matrix resMatrix = MatrixFactory.getInstance(
				matrix2.getWidth(),matrix1.getHeight()
		);
		int m = 0; //m - matrix2的列数
		for (int i = 0; i < matrix1.getHeight();) {
			int sum = 0; //sum - 求和
			for(int j = 0; j < matrix1.getWidth(); j++) {
				/**
				 * 按照matrix1 第i行 * matrix2 第m列 得到结果保存
				 */
				sum += matrix1.getArr()[i][j] * matrix2.getArr()[j][m];
			}
			resMatrix.setArrVal(i, m, sum);
			if (m == matrix2.getWidth() - 1) {
				i++;
				m=0;
			} else {
				m++;
			}
		}
		return resMatrix;
	}
  1. 矩阵转置
    设计思路: arr[i][j] == arr[j][i]。(i,j位置上的数据互换)
/**
	 * 矩阵反转
	 * @param matrix
	 * @return
	 */
	public static Matrix reverse(Matrix matrix) {
		//生成一个新的矩阵
		if (matrix == null) {
			throw new IllegalArgumentException("矩阵不匹配,请检查");
		}
		//生成一个新的矩阵
		Matrix resMatrix = MatrixFactory.getInstance(
				matrix.getHeight(), matrix.getWidth()
		);
		for(int i =0; i <matrix.getHeight();i++) {
			for (int j = 0; j < matrix.getWidth(); j++) {
				//如果行列值一样就跳过
				if (i == j) {
					resMatrix.setArrVal(i, j, matrix.getArr()[i][j]);
					continue;
				}
				resMatrix.setArrVal(j, i, matrix.getArr()[i][j]);
			}
		}
		return resMatrix;
	}
  1. 矩阵求det
    设计思路:利用递归,每次的余子式size-1,到了size为2时就直接使用余子式公式进行计算。
/**
	 * 获取矩阵det
	 * @param matrix
	 * @return
	 */
	public static double getMatrixDet(Matrix matrix) {
		if (matrix == null) {
			throw new IllegalArgumentException("矩阵不匹配,请检查");	
		}
		int height = matrix.getHeight();
		int width = matrix.getWidth();
		if (height != width) {
			throw new IllegalArgumentException("矩阵不匹配,请检查");	
		}
		return getMatrixDet(matrix,width);
	}
	
	/**
	 * 获取矩阵det
	 * @param matrix
	 * @param width
	 * @return
	 */
	private static double getMatrixDet(Matrix matrix, int size) {
		if (size <= 1) {
			return matrix.getArr()[0][0];
		} else if (size == 2) {
			return matrix.getArr()[0][0] * matrix.getArr()[1][1] - 
					matrix.getArr()[0][1] * matrix.getArr()[1][0];
 		}
		// 计算det,每次分解成大小为size-1的数组
		int det = 0;
		
		double[][] arr = matrix.getArr();
		for(int i = 0; i < arr[0].length; i++) {
			//获取余子式
			Matrix temp = getSubMatrix(0,i,size-1,arr);
			//统计det的值
			det += (Math.pow(-1, i)) * arr[0][i] * getMatrixDet(temp,size-1);
		}
		return det;
	}
	/**
	 * 获取余子式
	 * @param x - 要去除的第几行
	 * @param y - 要去除的第几列
	 * @param size 余子式size
	 * @param arr 原数组
	 * @return
	 */
	private static Matrix getSubMatrix(int x, int y,int size,double[][] arr) {
		//每次进行temp数组的填充
		Matrix temp = new Matrix(size,size);
		int addRow = 0; //填充计数
		for (int j = 0; j <= size; j++) {
			int addColumn = 0; //填充计数
			//跳过当前一行
			if (j == x) {
				addRow++;
				continue;
			}
			for (int m = 0; m < size + addColumn; m++) {
				//i列删除
				if (m == y) {
					addColumn++;
					continue;
				}
				//从第一行开始算,自动跳过第y列的数据
				temp.setArrVal(j-addRow,m - addColumn,arr[j][m]);
			}
		}
		return temp;
	}
  1. 矩阵求逆
    设计思路:利用公式 伴随矩阵 = det(矩阵) * 矩阵进行求解

	/**
	 * 求逆矩阵
	 * @param matrix
	 * @return
	 */
	public static Matrix inverse(Matrix matrix) {
		if (matrix.getWidth() != matrix.getHeight()) {
			throw new IllegalArgumentException("矩阵不匹配,请检查");
		}
		/**
		 * AX = B
		 * A-1AX=A-1A
		 * X=A-1B
		 */
		int size = matrix.getHeight(); //size
		double[][] arr = matrix.getArr(); //arr
		//结果矩阵
		Matrix resMatrix = MatrixFactory.getInstance(size,size);
		double det = getMatrixDet(matrix); //计算det
		for (int i = 0; i < Math.pow(size,2); i++) { //size * size = length
			//获取每一项的余子式
			Matrix temp = getSubMatrix(i/size, i%size, size-1, arr);
			resMatrix.setArrVal(i/size, i%size, Math.pow(-1, i) 
					* getMatrixDet(temp) / (det * 1.0)); //计算余子式的det / 总det
		}
		return reverse(resMatrix); //转置
	}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1536589.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

superset 二开增加 flink 数据源连接通过flink sql 查询数据

前言 superset 目前还不支持 flink 的数据源连接&#xff0c;目前我们公司在探索使用数据湖那一套东西&#xff1a; 使用 flink 作为计算引擎使用 paimon oss对象存储对接 flink 作为底层存储使用 superset 通过 flink gateway 查询 paimon 数据形成报表 增加flink数据源 …

Linux--进程(1)

目录 前言 1.冯诺依曼体系结构 2. 操作系统(Operator System)--第一个被加载的软件 3.进程 3.1基本概念 3.2Linux中的PCB 3.3通过系统调用创建子进程-fork初识 fork&#xff1a;创建一个子进程 为什么要创建子进程&#xff1f; fork的原理&#xff1a; 进一步了解fo…

黄仁勋对话Transformer七位作者:今天发生的一切都可以追溯到那一刻

关注文章底部的公众号,获取每日AI最新新闻 获取每日资讯 本周三,关于Transformer神经网络架构的开创性研究论文《Attention Is All You Need》作者们在GTC上齐聚一堂。 在GTC的900多个会议中,最受欢迎的是由NVIDIA创始人兼首席执行官黄仁勋主持的与传奇论文七位作者的交流。…

手上积累了一些企业目录,但是没有电话,在企XX天X查也没找到咋办?如何快速精准批量查询其他平台上查不到的企业电话?

在B端业务场景中&#xff0c;长期需要进行拓客。有时候是企业提供客户的联系方式&#xff0c;有时候是销售利用自己的人脉资源&#xff0c;对于资源不多的销售就需要查找到目标客户的联系方式。长期来说&#xff0c;销售都需要进行拓客&#xff0c;自己通过社交&#xff0c;网络…

ATA-5310前置微小信号放大器在红外线传感器中的应用

当涉及到红外线传感器时&#xff0c;前置微小信号放大器扮演着关键的角色。红外线传感器是一种用于探测和测量红外辐射的设备&#xff0c;它们通常用于热成像、物体检测、温度测量、动作检测等应用中。前置微小信号放大器在红外线传感器中的应用具有重要意义&#xff0c;下面将…

Vue js封装接口

天梦星服务平台 (tmxkj.top)https://tmxkj.top/#/ 1.安装axios npm install axios -g 2.在src下新建一个Api文件夹,再创建一个js文件 import axios from axios let configuration {url:"http://localhost:9090" } /*** 请求项目数据的请求体*/ async function h…

从错误中进行上下文学习

1、写作动机&#xff1a; 在上下文学习中&#xff0c;也称为少样本提示&#xff08;ICL&#xff09;&#xff0c;一直是调整LLM适应下游任务的标准方法&#xff0c;通过从少量输入-输出示例中学习。然而&#xff0c;所有基于ICL的方法都只从正确的输入-输出对中学习。 2、主要…

C++的内存管理

目录 1. C/C内存分布 2. C语言中动态内存管理方式 3. C内存管理方式 3.1 new/delete操作内置类型 4. operator new与operator delete函数 4.1 连续开辟空间(尽力了解) 5. new和delete的实现原理 5.1 内置类型 5.2 自定义类型 6. 深入理解 6.1malloc/free和new/delete的区…

探索多视角驱动的层次内容感知网络用于指静脉识别

文章目录 探索多视角驱动的层次内容感知网络用于指静脉识别总结摘要介绍相关工作多视角方法长短时记忆基于视角的目标表达 方法全局主干网络局部感知模块损失函数 实验和分析数据库实验设置和训练策略消融实验视角一致性的效果 参考文献 论文: Exploiting Multiperspective Dr…

小米相册提取表格选项消失解决方法

小米相册这次的提取表格选项消失 故障原因&#xff1a; 因为部分用户将小爱视觉&#xff08;原名扫一扫&#xff09;这个应用给卸载了导致 解决方法 应用商店下载 小爱视觉 安装后授权即可使用 注意&#xff1a;系统最好为最新的 Xiaomi HyperOS系统

抖音小店保证金是多少?怎么缴纳?最全的解答来了!

大家好&#xff0c;我是电商糖果 关于抖音小店的保证金&#xff0c;很多人并不太了解。 再加上保证金每年都会有变动&#xff0c;造成很多想开店的商家完全不懂&#xff0c;应该准备多少钱&#xff0c;以及如何缴纳&#xff1f; 这里糖果就给大家总结一下&#xff0c;想开店…

基于SSM+Jsp+Mysql的高校二手交易平台

基于SSMJspMysql的高校二手交易平台 基于SSMJspMysql的高校二手交易平台的设计与实现 开发语言&#xff1a;Java框架&#xff1a;ssm技术&#xff1a;JSPJDK版本&#xff1a;JDK1.8服务器&#xff1a;tomcat7数据库&#xff1a;mysql 5.7&#xff08;一定要5.7版本&#xff0…

JetBrains全家桶激活,分享 GoLand 2024 激活的方案

大家好&#xff0c;欢迎来到金榜探云手&#xff01; GoLand 公司简介 JetBrains 是一家专注于开发工具的软件公司&#xff0c;总部位于捷克。他们以提供强大的集成开发环境&#xff08;IDE&#xff09;而闻名&#xff0c;如 IntelliJ IDEA、PyCharm、和 GoLand等。这些工具被…

AI智能分析网关V4在非煤矿山安全生产视频智能监管场景中的应用

近年来&#xff0c;全国非煤矿山&#xff08;&#xff08;含金属非金属矿山、尾矿库&#xff0c;以及矿泉水等其他矿山&#xff09;安全生产工作取得明显成效&#xff0c;但安全基础仍然薄弱&#xff0c;事故总量仍然较大&#xff0c;重特大事故尚未得到根本遏制&#xff0c;安…

shell编程-jq命令详解

文章目录 前言一、jq简介1. 简介2. 语法3. 命令选项 二、用于处理json数据1. 过滤1.1 标识运算符1.2 基本过滤1.3 获取对象属性1.3 迭代数组元素1.4 获取数组元素1.5 使用运算符 2. 类型和值2.1 数组构造2.2 对象构造2.3 递归下降 3. 内置运算符和函数3.1 算术运算符3.2 函数3.…

外包干了20天,技术退步明显.......

先说一下自己的情况&#xff0c;大专生&#xff0c;21年通过校招进入杭州某软件公司&#xff0c;干了接近2年的功能测试&#xff0c;今年年初&#xff0c;感觉自己不能够在这样下去了&#xff0c;长时间呆在一个舒适的环境会让一个人堕落! 而我已经在一个企业干了2年的功能测试…

20240316-2-协同过滤(collaborative filtering)

协同过滤(collaborative filtering) 直观解释 协同过滤是推荐算法中最常用的算法之一&#xff0c;它根据user与item的交互&#xff0c;发现item之间的相关性&#xff0c;或者发现user之间的相关性&#xff0c;进行推荐。比如你有位朋友看电影的爱好跟你类似&#xff0c;然后最…

PublicCMS 后台模块 站点执行脚本RCE漏洞

PublicCMS是采用2023年主流技术开发的开源JAVACMS系统。由天津黑核科技有限公司开发&#xff0c;架构科学&#xff0c;轻松支撑上千万数据、千万PV&#xff1b;支持可视化编辑&#xff0c;多维扩展&#xff0c;全文搜索&#xff0c;全站静态化&#xff0c;SSI&#xff0c;动态页…

基于Spring Boot的拍卖管理系统设计与实现

摘 要 随着社会的发展&#xff0c;系统的管理形势越来越严峻。越来越多的用户利用互联网获得信息&#xff0c;但各种信息鱼龙混杂&#xff0c;信息真假难以辨别。为了方便用户更好的获得信息&#xff0c;因此&#xff0c;设计一种安全高效的拍卖管理系统极为重要。 为设计一个…

【Linux】Linux工具学习之gcc/g++

&#x1f525;博客主页&#xff1a; 小羊失眠啦. &#x1f3a5;系列专栏&#xff1a;《C语言》 《数据结构》 《C》 《Linux》 《Cpolar》 ❤️感谢大家点赞&#x1f44d;收藏⭐评论✍️ 接上文&#xff0c;我们已经学习了 Linux 中的编辑器 vim 的相关使用方法&#xff0c;现在…