LAPACK(Linear Algebra PACKage)库,是用Fortran语言编写的线性代数计算库,包含线性方程组求解(\(AX=B\))、矩阵分解、矩阵求逆、求矩阵特征值、奇异值等。该库用BLAS库做底层运算。

本示例将使用MKL中的LAPACK库计算线性方程组\(AX=B\)的解,并扩展使用此思路求逆矩阵的过程。首先介绍原理部分:

LU分解

引用自 LU分解 - 维基百科

对于方阵\(A\),其\(LU\)分解是将它分解成一个下三角矩阵\(L\)与上三角矩阵\(U\)的乘积,即\(A=LU\),

如一个\(3 \times 3\)的矩阵\(A\) ,其\(LU\)分解会写成下面的形式:

\[{\displaystyle A={\begin{bmatrix}a_{11}&a_{12}&a_{13}\\a_{21}&a_{22}&a_{23}\\a_{31}&a_{32}&a_{33}\\\end{bmatrix}}={\begin{bmatrix}l_{11}&0&0\\l_{21}&l_{22}&0\\l_{31}&l_{32}&l_{33}\\\end{bmatrix}}{\begin{bmatrix}u_{11}&u_{12}&u_{13}\\0&u_{22}&u_{23}\\0&0&u_{33}\\\end{bmatrix}}}{\displaystyle }\\
A\vec x = \vec b \Leftrightarrow (LU)\vec x = \vec b \Leftrightarrow \left\{ {\begin{array}{*{20}{c}}
{L\vec y = \vec b}\\
{U\vec x = \vec y}
\end{array}} \right.
\]

分解之后,由于\(L\)与\(U\)分别为下、上三角矩阵,再去求解\(X\)将变得更加简单。

然而,\(LU\)分解只适用于能用消去法处理的矩阵(比如左上角第一个元素为0时就无法消去)。

而\(PLU\)分解在加入置换矩阵\(P\)进行换行后,便可对任意实矩阵进行\(LU\)分解,此时\(A=P*L*U\)。

LAPACKE_sgesv计算线性方程组\(A*X = B\) 的解,其中 \(A\) 是$ N×N$ 矩阵,\(X\) 和 \(B\) 是 \(N×NRHS\) 矩阵。 将 \(A\) 分解为 \(A = P * L * U\),其中 \(P\) 是置换矩阵,\(L\)是单位下三角矩阵,\(U\)是上三角矩阵。 然后使用\(A\)的分解式来求解方程组\(A * X = B\)。

1 参数详解

lapack_int LAPACKE_sgesv( matrix_layout, 	// (input) 行优先(LAPACK_ROW_MAJOR)或列优先(LAPACK_COL_MAJOR)
n, // (input) 线性方程的个数,n>=0
nrhs, // (input) 矩阵B的列数,即线性方程组右端的项个数,nrhs>=0
a, // (input/output)系数矩阵A,维度为nxn
lda, // (input) A矩阵的第一维
ipiv, // (output) 置换矩阵,ipiv[i]表示矩阵A的第i行与第ipiv[i]行进行了交换
b, // (input/output)B矩阵
ldb // (input) B矩阵的第一维
);

2 定义线性方程组

Intel给出的官方示例为:

\[A = \left[ {\begin{array}{*{20}{r}}
{6.80}&{ - 6.05}&{ - 0.45}&{8.32}&{ - 9.67}\\
{ - 2.11}&{ - 3.30}&{2.58}&{2.71}&{ - 5.14}\\
{5.66}&{5.36}&{ - 2.70}&{4.35}&{ - 7.26}\\
{5.97}&{ - 4.44}&{0.27}&{ - 7.17}&{6.08}\\
{8.23}&{1.08}&{9.04}&{2.14}&{ - 6.87}
\end{array}} \right]~~~~~B = \left[ {\begin{array}{*{20}{r}}
{4.02}&{ - 1.56}&{9.81}\\
{6.19}&{4.00}&{ - 4.09}\\
{ - 8.22}&{ - 8.67}&{ - 4.57}\\
{ - 7.57}&{1.75}&{ - 8.61}\\
{ - 3.03}&{2.86}&{8.99}
\end{array}} \right]
\]

去求解\(AX=B\)的解\(X\)。

#include <stdlib.h>
#include <stdio.h>
#include "mkl_lapacke.h" // 参数
#define N 5
#define NRHS 3
#define LDA N
#define LDB NRHS
MKL_INT n = N, nrhs = NRHS, lda = LDA, ldb = LDB, info;

MKL_INT ipiv[N];
float a[LDA*N] = {
6.80f, -6.05f, -0.45f, 8.32f, -9.67f,
-2.11f, -3.30f, 2.58f, 2.71f, -5.14f,
5.66f, 5.36f, -2.70f, 4.35f, -7.26f,
5.97f, -4.44f, 0.27f, -7.17f, 6.08f,
8.23f, 1.08f, 9.04f, 2.14f, -6.87f
};
float b[LDB*N] = {
4.02f, -1.56f, 9.81f,
6.19f, 4.00f, -4.09f,
-8.22f, -8.67f, -4.57f,
-7.57f, 1.75f, -8.61f,
-3.03f, 2.86f, 8.99f
};

3 执行求解过程

LAPACKE_sgesv( LAPACK_ROW_MAJOR, n, nrhs, a, lda, ipiv, b, ldb );

输出结果为:

完整代码

#include <stdlib.h>
#include <stdio.h>
#include "mkl_lapacke.h" extern void print_matrix(const char* desc, MKL_INT m, MKL_INT n, float* a, MKL_INT lda);
extern void print_int_vector(const char* desc, MKL_INT n, MKL_INT* a); #define N 5
#define NRHS 3
#define LDA N
#define LDB NRHS int main() { MKL_INT n = N, nrhs = NRHS, lda = LDA, ldb = LDB, info; MKL_INT ipiv[N];
float a[LDA * N] = {
6.80f, -6.05f, -0.45f, 8.32f, -9.67f,
-2.11f, -3.30f, 2.58f, 2.71f, -5.14f,
5.66f, 5.36f, -2.70f, 4.35f, -7.26f,
5.97f, -4.44f, 0.27f, -7.17f, 6.08f,
8.23f, 1.08f, 9.04f, 2.14f, -6.87f
};
float b[LDB * N] = {
4.02f, -1.56f, 9.81f,
6.19f, 4.00f, -4.09f,
-8.22f, -8.67f, -4.57f,
-7.57f, 1.75f, -8.61f,
-3.03f, 2.86f, 8.99f
}; printf("LAPACKE_sgesv (row-major, high-level) Example Program Results\n"); info = LAPACKE_sgesv(LAPACK_ROW_MAJOR, n, nrhs, a, lda, ipiv,
b, ldb); if (info > 0) {
printf("The diagonal element of the triangular factor of A,\n");
printf("U(%i,%i) is zero, so that A is singular;\n", info, info);
printf("the solution could not be computed.\n");
exit(1);
} print_matrix("Solution", n, nrhs, b, ldb); print_matrix("Details of LU factorization", n, n, a, lda); print_int_vector("Pivot indices", n, ipiv);
exit(0);
} void print_matrix(const char* desc, MKL_INT m, MKL_INT n, float* a, MKL_INT lda) {
MKL_INT i, j;
printf("\n %s\n", desc);
for (i = 0; i < m; i++) {
for (j = 0; j < n; j++) printf(" %6.2f", a[i * lda + j]);
printf("\n");
}
} void print_int_vector(const char* desc, MKL_INT n, MKL_INT* a) {
MKL_INT j;
printf("\n %s\n", desc);
for (j = 0; j < n; j++) printf(" %6i", a[j]);
printf("\n");
}

补充:矩阵求逆

简单来说,在使用以上API计算\(AX=B\),当\(B\)为单位矩阵时,\(X\)即为\(A^{-1}\)。

将上述案例中的

float b[LDB * N] = {
4.02f, -1.56f, 9.81f,
6.19f, 4.00f, -4.09f,
-8.22f, -8.67f, -4.57f,
-7.57f, 1.75f, -8.61f,
-3.03f, 2.86f, 8.99f
};
/**********改为**********/ #define NRHS 5
float b[LDB * N] = {
1.0f, 0.0f, 0.0f, 0.0f, 0.0f,
0.0f, 1.0f, 0.0f, 0.0f, 0.0f,
0.0f, 0.0f, 1.0f, 0.0f, 0.0f,
0.0f, 0.0f, 0.0f, 1.0f, 0.0f,
0.0f, 0.0f, 0.0f, 0.0f, 1.0f,
};

即可求解矩阵\(A\)的逆矩阵,输出为:

对比在Matlab中使用inv()函数求逆:

A = [ 6.80, -6.05, -0.45,  8.32, -9.67;
-2.11, -3.30, 2.58, 2.71, -5.14;
5.66, 5.36, -2.70, 4.35, -7.26;
5.97, -4.44, 0.27, -7.17, 6.08;
8.23, 1.08, 9.04, 2.14, -6.87];
A_inv=inv(A)

结果相同。

MKL库线性方程组求解(LAPACKE_?gesv)的更多相关文章

  1. MKL库奇异值分解(LAPACKE_dgesvd)

    对任意一个\(m\times n\)的实矩阵,总可以按照SVD算法对其进行分解.即: \[A = U\Sigma V^T \] 其中\(U.V\)分别为\(m\times m.n\times n\)的 ...

  2. [转]Numpy使用MKL库提升计算性能

    from:http://unifius.wordpress.com.cn/archives/5 系统:Gentoo Linux (64bit, Kernel 3.7.1)配置:Intel(R) Cor ...

  3. 如何在 code blocks中使用 mkl库

    为了安装caffe, 所以安装了mkl, 现在想在codeblock的项目中使用mkl. 设置mkl环境变量: mkl安装好后默认是在/opt/intel/mkl中,其中/opt/intel/mkl/ ...

  4. Matlab - 线性方程组求解

    常用函数:det 计算矩阵的行列式的值inv 求矩阵的逆阵rank 求矩阵的秩[V D]=eig(A) 求矩阵A的特征值和特征向量poly 求矩阵的特征多项式rref 用初等变换将矩阵化成行阶梯形nu ...

  5. poj 2891 模线性方程组求解

    Strange Way to Express Integers Time Limit: 1000MS   Memory Limit: 131072K Total Submissions: 8005   ...

  6. MKL库矩阵乘法

    此示例是利用Intel 的MKL库函数计算矩阵的乘法,目标为:\(C=\alpha*A*B+\beta*C\),由函数cblas_dgemm实现: 其中\(A\)为\(m\times k\)维矩阵,\ ...

  7. 【原创】开源Math.NET基础数学类库使用(06)直接求解线性方程组

                   本博客所有文章分类的总目录:[总目录]本博客博文总目录-实时更新  开源Math.NET基础数学类库使用总目录:[目录]开源Math.NET基础数学类库使用总目录 前言 ...

  8. 开源Math.NET基础数学类库使用(06)数值分析之线性方程组直接求解

    原文:[原创]开源Math.NET基础数学类库使用(06)数值分析之线性方程组直接求解 开源Math.NET基础数学类库使用系列文章总目录:   1.开源.NET基础数学计算组件Math.NET(一) ...

  9. python 求解线性方程组

    Python线性方程组求解 求解线性方程组比较简单,只需要用到一个函数(scipy.linalg.solve)就可以了.比如我们要求以下方程的解,这是一个非齐次线性方程组: 3x_1 + x_2 - ...

  10. 科学计算库(BLAS,LAPACK,MKL,EIGEN)

    函数库接口标准:BLAS (Basic Linear Algebra Subprograms)和LAPACK (Linear Algebra PACKage) 1979年,Netlib首先用Fortr ...

随机推荐

  1. 创建的容器都是没有指定 volume的,为什么docker volume ls命令会看到很多volume列表?

    有那么一天,停止了世界所有的容器(在你电脑上),执行以下命令 docker volume ls docker volume ls | wc -l 哎呦,我去,怎么这么多啥玩意,再执行以下命令看占了我多 ...

  2. 如何优化和提高MaxKB回答的质量和准确性?

    目前 ChatGPT.GLM等生成式人工智能在文本生成.文本到图像生成等在各行各业的都有着广泛的应用,但是由于大模型训练集基本都是构建于网络公开的数据,对于一些实时性的.非公开的或离线的数据是无法获取 ...

  3. nodejs读写yaml

    nodejs读写yaml npm install -g js-yaml // read.js const fs = require('fs'); const yaml = require('js-ya ...

  4. Web前端入门第 33 问:CSS 元素外观常用属性(边框、阴影、轮廓、透明度)

    background 作为元素外观里的重点功臣介绍完毕,本文再一览其他常用的外观属性. 本文示例中,盒子基础样式: .box { font-size: 20px; margin: 20px; padd ...

  5. mybatis——分页插件PageHelper的使用

    项目开发中涉及列表查询时,经常会需要对查询结果进行分页处理:常用的一个插件--PageHelper,是国内非常优秀的一款开源的mybatis分页插件,它支持基本主流与常用的数据库,一致支持mysql. ...

  6. MCP协议的相关知识总结

    一.基本概念与核心价值 定义与定位 MCP(Model Context Protocol,模型上下文协议)是由Anthropic推出的开放标准协议,旨在通过标准化接口实现大语言模型(LLM)与外部数据 ...

  7. jetbrains这是作啥妖呢,用了好久都没有问题,现在这是咋了?

    rider今天更新了下2025.1 数据库驱动下载不来: 插件不能正常访问: 清理下用户的配置目录: %UserProfile%\AppData\Local\JetBrains %UserProfil ...

  8. Linux TCP网关的线程结构方案

    如果所示: 无论客户端还是服务端链接网关的socket都拆分为读EPoll.写EPoll分别独立. 有两个线程:线程A(左).线程B(右): 线程A负责服务端Socket的读和客户端socket的写, ...

  9. Python 类不要再写 __init__ 方法了

    花下猫语:我们周刊第 98 期分享过一篇文章,它指出了 __init__ 方法存在的问题和新的最佳实践,第 99 期也分享了一篇文章佐证了第一篇文章的观点.我认为它们提出的是一个值得注意和思考的问题, ...

  10. servlet 作用域

    作用域总结 HttpServletRequest:一次请求,请求响应之前有效 HttpSession:一次会话开始,浏览器不关闭或不超时之前有效 ServletContext:服务器启动开始,服务器停 ...