LU分解法求逆矩阵 C语言实现
最近在网上找了下,没有找到我想要的C语言版本,找到的也是错误的。故自己写了一个,并进行了相关测试,贴出来分享。
具体的LU分解算法就不细说了,随便找本书就知道了,关键是分解的处理流程,细节特别容易出错,一切都在代码里面。
#include <stdio.h>
#include <memory.h>
#include <stdlib.h> #define N 4
#define DEBUG 1 //debug label,0即不打印相关结果,非0打印相关输出结果 void matrix_inverse_LU(float a[][N])
{
float l[N][N], u[N][N];
float l_inverse[N][N], u_inverse[N][N];
float a_inverse[N][N];
int i, j, k;
float s, t; memset(l, 0, sizeof(l));
memset(u, 0, sizeof(u));
memset(l_inverse, 0, sizeof(l_inverse));
memset(u_inverse, 0, sizeof(u_inverse));
memset(a_inverse, 0, sizeof(u_inverse)); for (i = 0; i < N;i++) //计算l矩阵对角线
{
l[i][i] = 1;
} for (i = 0;i < N;i++)
{
for (j = i;j < N;j++)
{
s = 0;
for (k = 0;k < i;k++)
{
s += l[i][k] * u[k][j];
}
u[i][j] = a[i][j] - s; //按行计算u值
} for (j = i + 1;j < N;j++)
{
s = 0;
for (k = 0; k < i; k++)
{
s += l[j][k] * u[k][i];
}
l[j][i] = (a[j][i] - s) / u[i][i]; //按列计算l值
}
} for (i = 0;i < N;i++) //按行序,行内从高到低,计算l的逆矩阵
{
l_inverse[i][i] = 1;
}
for (i= 1;i < N;i++)
{
for (j = 0;j < i;j++)
{
s = 0;
for (k = 0;k < i;k++)
{
s += l[i][k] * l_inverse[k][j];
}
l_inverse[i][j] = -s;
}
} #if DEBUG
printf("test l_inverse:\n");
for (i = 0; i < N; i++)
{
for (j = 0; j < N; j++)
{
s = 0;
for (k = 0; k < N; k++)
{
s += l[i][k] * l_inverse[k][j];
} printf("%f ", s);
}
putchar('\n');
}
#endif for (i = 0;i < N;i++) //按列序,列内按照从下到上,计算u的逆矩阵
{
u_inverse[i][i] = 1 / u[i][i];
}
for (i = 1;i < N;i++)
{
for (j = i - 1;j >=0;j--)
{
s = 0;
for (k = j + 1;k <= i;k++)
{
s += u[j][k] * u_inverse[k][i];
}
u_inverse[j][i] = -s / u[j][j];
}
} #if DEBUG
printf("test u_inverse:\n");
for (i = 0;i < N;i++)
{
for (j = 0;j < N;j++)
{
s = 0;
for (k = 0;k < N;k++)
{
s += u[i][k] * u_inverse[k][j];
} printf("%f ",s);
}
putchar('\n');
}
#endif for (i = 0;i < N;i++) //计算矩阵a的逆矩阵
{
for (j = 0;j < N;j++)
{
for (k = 0;k < N;k++)
{
a_inverse[i][j] += u_inverse[i][k] * l_inverse[k][j];
}
}
} #if DEBUG
printf("test a:\n");
for (i = 0; i < N; i++)
{
for (j = 0; j < N; j++)
{
s = 0;
for (k = 0; k < N; k++)
{
s += a[i][k] * a_inverse[k][j];
} printf("%f ", s);
}
putchar('\n');
}
#endif
} void main()
{
int i, j, k;
float a[N][N]; for (i = 0;i < N;i++)
{
for (j = 0;j < N;j++)
{
a[i][j] = rand() % 10;
}
} matrix_inverse_LU(a);
}
提醒一下,打印出来的验证结果,可能跟单位矩阵E有稍许不同,如下图所示:

主要是因为相关浮点数计算误差所致,系统原因,不是算法问题。
解决这个问题的方法,就是用更精确的double类型或者改用各适合进行科学计算的工具/语言。
LU分解法求逆矩阵 C语言实现的更多相关文章
- Matlab数值计算示例: 牛顿插值法、LU分解法、拉格朗日插值法、牛顿插值法
本文源于一次课题作业,部分自己写的,部分借用了网上的demo 牛顿迭代法(1) x=1:0.01:2; y=x.^3-x.^2+sin(x)-1; plot(x,y,'linewidth',2);gr ...
- Guass列选主元消去法和三角分解法
最近数值计算学了Guass列主消元法和三角分解法解线性方程组,具体原理如下: 1.Guass列选主元消去法对于AX =B 1).消元过程:将(A|B)进行变换为,其中是上三角矩阵.即: k从1到n-1 ...
- 寒假答辩作品——掘地求升C语言版
寒假答辩—掘地求升(C语言版) 前言 这个是作为寒假答辩作品写的. 之前考虑过用Unity写个游戏,但毕竟不熟悉C#,感觉几乎都是在套模板,而且写着不顺手,有想法却只能 看着C#发呆,很是无奈,所以决 ...
- [Architecture] 系统架构正交分解法
[Architecture] 系统架构正交分解法 前言 随着企业成长,支持企业业务的软件,也会越来越庞大与复杂.当系统复杂到一定程度,开发人员会发现很多系统架构的设计细节,很难有条理.有组织的用一张大 ...
- 利用栈实现算术表达式求值(Java语言描述)
利用栈实现算术表达式求值(Java语言描述) 算术表达式求值是栈的典型应用,自己写栈,实现Java栈算术表达式求值,涉及栈,编译原理方面的知识.声明:部分代码参考自茫茫大海的专栏. 链栈的实现: pa ...
- 分治法求一个N个元素数组的逆序数
背景 逆序数:也就是说,对于n个不同的元素,先规定各元素之间有一个标准次序(例如n个 不同的自然数,可规定从小到大为标准次序),于是在这n个元素的任一排列中,当某两个元素的先后次序与标准次序不同时, ...
- 《github一天一道算法题》:分治法求数组最大连续子序列和
看书.思考.写代码. /*************************************** * copyright@hustyangju * blog: http://blog.csdn. ...
- 时间序列分解-STL分解法
时间序列分解-STL分解法 [转载时请注明来源]:http://www.cnblogs.com/runner-ljt/ Ljt 作为一个初学者,水平有限,欢迎交流指正. STL(’Seasonal a ...
- 项目管理——WBS工作分解法
首先我们要了解什么是WBS工作分解法 工作分解结构(Work Breakdown Structure,简称WBS)跟因数分解是一个原理,就是把一个项目,按一定的原则分解,项目分解成任务,任务再分解成一 ...
随机推荐
- Oracle数据块
最小单位的输入\输出 数据块由操作系统中的一个或多个块组成 数据库是表空间的基本单位 DB_BLOCK_SIZE 查看 Oracle 块的大小语句: SQL> show parameter db ...
- P1828香甜的黄油
这是一道关于最短路的绿题. 题目给出一些农场,每个农场有奶牛,农场与农场之间存在边,要使所有奶牛到达其中一个农场的总距离最短,输出他们到达这个农场的距离.首先我想到了最小生成树,但我发现其实并不是,因 ...
- python3.7 安装Scrapy 失败问题
python的Scrapy框架,需要Twisted依赖以及VC++ 14 以上的环境,这些就不再赘述.讲讲今天安装Twisted和Scrapy遇到的其他问题. 首先就是直接安装Twisted成功后,安 ...
- SVN简单流程总结
1 创建仓库 2 启动svn服务器 svnserve -d -r 仓库地址(如:D:\SVN\repoDemo1) 3 新的用户第一次与服务器交互时,需要使用checkout将仓库检出到本 ...
- Linux服务器Java进程突然消失排查办法
出处:JAVA进程突然消失的原因? 问题描述 在实际生产环境下,如果我们遇见Java进程突然消失,该如何去排查问题? 思路 可能有几种原因: ①.Java应用程序的问题:发生OOM导致进程Crash ...
- 如何用纯 CSS 绘制一个世界上不存在的彭罗斯三角形
效果预览 在线演示 按下右侧的"点击预览"按钮可以在当前页面预览,点击链接可以全屏预览. https://codepen.io/comehope/pen/RyvgMZ 可交互视频教 ...
- 用data的方法获取值的时候,要注意的问题一定要在先封装好
var art=$(".add-more").prev().find(".content").data("list1"); 我们一定要在aj ...
- luogu P4437 [HNOI/AHOI2018]排列
luogu 问题本质是把\(a_i\)作为\(i\)的父亲,然后如果有环就不合法,否则每次要取数,要满足取之前他的父亲都被取过(父亲为0可以直接取),求最大价值 贪心想法显然是要把权值大的尽量放在后面 ...
- react 在新窗口 打开页面
遇到这个需求 首先通过 Link a去尝试直接跳转.发现2个问题 1.Link跳转 会自动进行登录校验,我设想是路由没有匹配到,去验证后大致排除了. 因为这个链接 直接粘贴到浏览器 是可以访问到的. ...
- linux创建定时任务发送钉钉通知
一.现在钉钉里面添加机器人 添加成功后,复制出Webhook链接. 注意,自定义关键字时你的发送信息中一定要完整包含关键字 二.找到自己的服务器 1. sudo su 切换到root用户 2.cron ...