Sherman-Morrison公式

  Sherman-Morrison公式以 Jack Sherman 和 Winifred J. Morrison命名,在线性代数中,是求解逆矩阵的一种方法。本篇博客将介绍该公式及其应用,首先我们来看一下该公式的内容及其证明。

  (Sherman-Morrison公式)假设\(A\in\mathbb{R}^{n\times n}\)为可逆矩阵,\(u,v\in\mathbb{R}^{n}\)为列向量,则\(A+uv^{T}\)可逆当且仅当\(1+v^{T}A^{-1}u\neq 0\), 且当\(A+uv^{T}\)可逆时,该逆矩阵由以下公式给出:

\[(A+uv^{T})^{-1}=A^{-1}-{A^{-1}uv^{T}A^{-1} \over 1+v^{T}A^{-1}u}.
\]

证明:

\((\Leftarrow)\)当\(1+v^{T}A^{-1}u\neq 0\)时,令\(X=A+uv^{T}, Y=A^{-1}-{A^{-1}uv^{T}A^{-1} \over 1+v^{T}A^{-1}u}\),则只需证明\(XY=YX=I\)即可,其中\(I\)为n阶单位矩阵。

\[{\displaystyle
{\begin{aligned}
XY&=(A+uv^{T})\left(A^{-1}-{A^{-1}uv^{T}A^{-1} \over 1+v^{T}A^{-1}u}\right)\\
&=AA^{-1}+uv^{T}A^{-1}-{AA^{-1}uv^{T}A^{-1}+uv^{T}A^{-1}uv^{T}A^{-1} \over 1+v^{T}A^{-1}u}\\
&=I+uv^{T}A^{-1}-{uv^{T}A^{-1}+uv^{T}A^{-1}uv^{T}A^{-1} \over 1+v^{T}A^{-1}u}\\
&=I+uv^{T}A^{-1}-{u(1+v^{T}A^{-1}u)v^{T}A^{-1} \over 1+v^{T}A^{-1}u}\\
&=I+uv^{T}A^{-1}-uv^{T}A^{-1}\\
&=I\end{aligned}}}
\]

同理,有\(YX=I\).因此,当\(1+v^{T}A^{-1}u\neq 0\)时,\((A+uv^{T})^{-1}=A^{-1}-{A^{-1}uv^{T}A^{-1} \over 1+v^{T}A^{-1}u}.\)

\((\Rightarrow)\)当\(u=0\)时,显然有\(1+v^{T}A^{-1}u=1\neq 0.\)当\(u\neq0\)时,用反正法证明该命题成立。假设\(A+uv^{T}\)可逆,但\(1+v^{T}A^{-1}u = 0\),则有

\[(A+uv^{T})A^{-1}u=u+u(v^{T}A^{-1}u)=(1+v^{T}A^{-1}u)u=0.
\]

因为\(A+uv^{T}\)可逆,故\(A^{-1}\)u=0,又因为\(A^{-1}\)可逆,故\(u=0\),此与假设\(u\neq 0\)矛盾。因此,当\(A+uv^{T}\)可逆时,有\(1+v^{T}A^{-1}u \neq 0.\)

Sherman-Morrison公式的应用

应用1:\(A=I\)时的Sherman-Morrison公式

  在Sherman-Morrison公式中,令\(A=I\),则有:\(I+uv^{T}\)可逆当且仅当\(1+v^{T}u\neq 0\), 且当\(I+uv^{T}\)可逆时,该逆矩阵由以下公式给出:

\[(I+uv^{T})^{-1}=I-{uv^{T} \over 1+v^{T}u}.
\]

  再令\(v=u\),则\(1+u^{T}u > 0\), 因此,\(I+uu^{T}\)可逆,且

\[(I+uu^{T})^{-1}=I-{uu^{T} \over 1+u^{T}u}.
\]

应用2:BFGS算法

  Sherman-Morrison公式在BFGS算法中的应用,可用来求解BFGS算法中近似Hessian矩阵的逆。本篇博客并不打算给出Sherman-Morrison公式在BFGS算法中的应用,将会再写篇博客介绍BFGS算法,到时再给出该公式的应用,并会在之后补上该博客的链接(因为笔者还没写)。

应用3:循环三对角线性方程组的求解

  本篇博客将详细讲述Sherman-Morrison公式在循环三对角线性方程组的求解中的应用。

  首先给给出理论知识介绍部分。

  对于\(A\in\mathbb{R}^{n\times n}\)为可逆矩阵,\(u,v\in\mathbb{R}^{n}\)为列向量,\(1+v^{T}A^{-1}u\neq 0\),需要求解方程\((A+uv^{T})x=b.\)对此,我们可以先求解以下两个方程:

\[Ay=b,\qquad Az=u$$.
然后令$x=y-\frac{v^{T}y}{1+v^{T}z}z$,该解即为原方程的解,验证如下:
\]

{\displaystyle

{\begin{aligned}

(A+uv{T})x&=(A+uv{T})(y-\frac{v{T}y}{1+v{T}z}z)\

&=Ay+uv{T}y-\frac{v{T}y}{1+v{T}z}Az-\frac{v{T}y}{1+v{T}z}uv{T}z\

&=b+uv{T}y-\frac{v{T}yu+v{T}yuv{T}z}{1+v^{T}z}\

&=b+uv{T}y-\frac{(1+v{T}z)v{T}yu}{1+v{T}z}\

&=b+uv{T}y-uv{T}y\

&=b\end{aligned}}}

\[  这样将原方程$ (A+uv^{T})x=b$分成两个方程,可以在一定程度上简化原方程。接下来,我们将介绍循环三对角线性方程组的求解。
  所谓循环三对角线性方程组,指的是系数矩阵为如下形式:
\]

A=\begin{bmatrix}

b_1&c_1&0&\cdots&0&a_1\

a_2&b_2&c_2&0&\vdots&0\

0&\ddots&\ddots&\ddots&0&\vdots\

\vdots&\vdots&a_{n-2}&b_{n-2}&c_{n-2}&0\

0&\cdots&\cdots&a_{n-1}&b_{n-1}&c_{n-1}\

c_n&0&\cdots&0&a_n&b_n\end{bmatrix}

\[循环三对角线性方程组可写成$Ax=d$,其中$d=(d_{1},d_2,...,d_{n})^{T}.$
  对于此方程的求解,我们令$u=(\gamma, 0,0,...,c_{n})^{T}, v=(1,0,0,...,\frac{a_1}{\gamma})^{T}$, 且$A=A^{'}+uv^{T}$,其中$A^{'}$如下:
\]

A^{'}=\begin{bmatrix}

b_1-\gamma&c_1&0&\cdots&0&0\

a_2&b_2&c_2&0&\vdots&0\

0&\ddots&\ddots&\ddots&0&\vdots\

\vdots&\vdots&a_{n-2}&b_{n-2}&c_{n-2}&0\

0&\cdots&\cdots&a_{n-1}&b_{n-1}&c_{n-1}\

0&0&\cdots&0&a_n&b_n-\frac{a_1c_n}{\gamma}\end{bmatrix}

\[$A^{'}$为三对角矩阵。根据以上的理论知识,我们只需要求解以下两个方程
$$A^{'}y=d,\qquad A^{'}z=u,\]

然后,就能根据\(y,z\)求出\(x\).而以上两个方程为三对角线性方程组,可以用追赶法(或Thomas法)求解,具体算法可以参考博客:三对角线性方程组(tridiagonal systems of equations)的求解 。

  综上,我们利用Sherman-Morrison公式的思想,可以将循环三对角线性方程组转化为三对角线性方程组求解。我们将会在下面给出该算法的Python语言实现。

Python实现

  我们要解的循环三对角线性方程组如下:

\[{\begin{bmatrix}
4&1&{0}&{0}&{2}\\
{1}&{4}&{1}&{0}&{0}\\
{0}&{1}&{4}&{1}&{0}\\
{0}&{0}&{1}&{4}&{1}\\
{3}&{0}&{0}&{1}&{4}\\
\end{bmatrix}}
{\begin{bmatrix}{x_{1}}\\{x_{2}}\\{x_{3}}\\{x_{4}} \\{x_{5}}\\\end{bmatrix}}={\begin{bmatrix}{7\\6\\ 6\\6\\8}\\\end{bmatrix}}
\]

  用Python实现解该方程的Python完整代码如下:

# use Sherman-Morrison Formula and Thomas Method to solve cyclic tridiagonal linear equation

import numpy as np

# Thomas Method for soling tridiagonal linear equation Ax=d
# parameter: a,b,c,d are list-like of same length
# b: main diagonal of matrix A
# a: main diagonal below of matrix A
# c: main diagonal upper of matrix A
# d: Ax=d
# return: x(type=list), the solution of Ax=d
def TDMA(a,b,c,d): try:
n = len(d) # order of tridiagonal square matrix # use a,b,c to create matrix A, which is not necessary in the algorithm
A = np.array([[0]*n]*n, dtype='float64') for i in range(n):
A[i,i] = b[i]
if i > 0:
A[i, i-1] = a[i]
if i < n-1:
A[i, i+1] = c[i] # new list of modified coefficients
c_1 = [0]*n
d_1 = [0]*n for i in range(n):
if not i:
c_1[i] = c[i]/b[i]
d_1[i] = d[i] / b[i]
else:
c_1[i] = c[i]/(b[i]-c_1[i-1]*a[i])
d_1[i] = (d[i]-d_1[i-1]*a[i])/(b[i]-c_1[i-1] * a[i]) # x: solution of Ax=d
x = [0]*n for i in range(n-1, -1, -1):
if i == n-1:
x[i] = d_1[i]
else:
x[i] = d_1[i]-c_1[i]*x[i+1] x = [round(_, 4) for _ in x] return x except Exception as e:
return e # Sherman-Morrison Fomula for soling cyclic tridiagonal linear equation Ax=d
# parameter: a,b,c,d are list-like of same length
# b: main diagonal of matrix A
# a: main diagonal below of matrix A
# c: main diagonal upper of matrix A
# d: Ax=d
# return: x(type=list), the solution of Ax=d
def Cyclic_Tridiagnoal_Linear_Equation(a,b,c,d):
try:
# use a,b,c to create cyclic tridiagonal matrix A
n = len(d)
A = np.array([[0] * n] * n, dtype='float64') for i in range(n):
A[i, i] = b[i]
if i > 0:
A[i, i - 1] = a[i]
if i < n - 1:
A[i, i + 1] = c[i]
A[0, n - 1] = a[0]
A[n - 1, 0] = c[n - 1] gamma = 1 # gamma can be set freely
u = [gamma] + [0] * (n - 2) + [c[n - 1]]
v = [1] + [0] * (n - 2) + [a[0] / gamma] # modify the coefficient to form A'
b[0] -= gamma
b[n - 1] -= a[0] * c[n - 1] / gamma
a[0] = 0
c[n - 1] = 0 # solve A'y=d, A'z=u by using Thomas Method
y = np.array(TDMA(a, b, c, d))
z = np.array(TDMA(a, b, c, u)) # use y and z to calculate x
# x = y-(v·y)/(1+v·z) *z
# x is the solution of Ax=d
x = y - (np.dot(np.array(v), y)) / (1 + np.dot(np.array(v), z)) * z x = [round(_, 3) for _ in x] return x except Exception as e:
return e def main():
'''
equation:
A = [[4,1,0,0,2],
[1,4,1,0,0],
[0,1,4,1,0],
[0,0,1,4,1],
[3,0,0,1,4]]
d = [7,6,6,6,8]
solution x should be [1,1,1,1,1]
''' a = [2, 1, 1, 1, 1]
b = [4, 4, 4, 4, 4]
c = [1, 1, 1, 1, 3]
d = [7, 6, 6, 6, 8] x = Cyclic_Tridiagnoal_Linear_Equation(a,b,c,d)
print('The solution is %s'%x) main()

输出结果如下:

The solution is [1.0, 1.0, 1.0, 1.0, 1.0]

参考文献

  1. https://en.wikipedia.org/wiki/Sherman–Morrison_formula
  2. http://wwwmayr.in.tum.de/konferenzen/Jass09/courses/2/Soldatenko_paper.pdf
  3. https://scicomp.stackexchange.com/questions/10137/solving-system-of-linear-equations-with-cyclic-tridiagonal-matrix
  4. https://blog.csdn.net/jclian91/article/details/80251244

注意:本人现已开通两个微信公众号: 用Python做数学(微信号为:python_math)以及轻松学会Python爬虫(微信号为:easy_web_scrape), 欢迎大家关注哦~~

Sherman-Morrison公式及其应用的更多相关文章

  1. ZROI 暑期高端峰会 A班 Day2 线性代数

    高斯消元 很普及组,不讲了 当主元没有逆的时候可以辗转相除. 如果也没有带余数除法--没救了 逆矩阵 我们定义矩阵 \(A\) 的逆矩阵为 \(A^{-1}\),满足 \(AA^{-1}=A^{-1} ...

  2. RLS自适应滤波器中用矩阵求逆引理来避免求逆运算

    在RLS自适应滤波器的实现过程中,难免不涉及矩阵的求逆运算.而求逆操作双是非常耗时的,一个很自然的想法就是尽可能的避免直接对矩阵进行求逆运算.那么,在RLS自适应滤波器的实现中,有没有一种方法能避免直 ...

  3. 2019暑期金华集训 Day2 线性代数

    自闭集训 Day2 线性代数 高斯消元 做实数时,需要找绝对值最大的作为主元,以获取更高精度. 在欧几里得环(简单例子是模合数)意义下也是对的.比如模合数意义下可以使用辗转相除法消元. 欧几里得环:对 ...

  4. 伍德伯里矩阵恒等式(Woodbury matrix identity)

    宜言饮酒,与子偕老.琴瑟在御,莫不静好. 更多精彩内容请关注微信公众号 "优化与算法" 在数学(特别是线性代数)中,Woodbury矩阵恒等式是以Max A.Woodbury命名的 ...

  5. 为WLW开发Latex公式插件

    WLW是写博客的利器,支持离线.格式排版等,而且拥有众多的插件.博客园推荐了代码插入插件,但是没有提供WLW的公式编译插件.目前我的一般做法是:先在Word下使用MathType编辑好公式,然后将公式 ...

  6. 百度编辑器UEditor与UEditor 公式插件完整Demo

    1.下载UEditor(我的是.net项目) 2.下载UEditor公式插件 3.新建解决方案和项目 4.在浏览器中预览index.html页面 结果: 5.index.html源码 <!DOC ...

  7. poi读取excel模板,填充内容并导出,支持导出2007支持公式自动计算

    /** * 版权所有(C) 2016 * @author www.xiongge.club * @date 2016-12-7 上午10:03:29 */ package xlsx; /** * @C ...

  8. Oracle Sales Cloud:管理沙盒(定制化)小细节1——利用公式创建字段并显示在前端页面

    Oracle Sales Cloud(Oracle 销售云)是一套基于Oracle云端的CRM管理系统.由于 Oracle 销售云是基于 Oracle 云环境的,它与传统的管理系统相比,显著特点之一便 ...

  9. Excel公式 提取文件路径后缀

    我们在代码中获取一个文件路径的后缀,是一个很简单的事. 如C#中,可以通过new FileInfo(filePath).Extension,或者Path.GetExtension(filePath)获 ...

随机推荐

  1. 学习STM32,你不得不了解的五大嵌入式操作系统

    学习STM32,你不得不了解的五大嵌入式操作系统                                                  原标题:学习STM32,你不得不了解的五大嵌入式操作 ...

  2. android开发笔记(3)

    在android环境开发过程中,我最陌生的是虚拟机的创建以及使用,下面是虚拟机的创建以及使用过程. 创建虚拟机: 点击创建虚拟机的按钮,create->填写虚拟机相关的信息,确认之后创建. 创建 ...

  3. ReSharper 10.0.0.1 Ultimate 完美破解补丁使用方法

    转自:http://www.leavescn.com/Page/Content.aspx?id=94 ReSharper 10.0.0.1 Ultimate 完美破解补丁使用方法,本资源来自互联网,感 ...

  4. 几种简单的编码(为什么使用ASCII码)

    二-十进制码(BCD码) 在目前的数字系统中,一般是采用二进制数进行运算的,但是由于人们习惯采用十进制数,因此常需进行十进制数和二进制数之间的转换,其转换方法上面已讨论过了.为了便于数字系统处理十进制 ...

  5. [git] 本地仓库信息的查询

    本地仓库信息查询操作 1.1  git status 查看当前暂存区状态 git  status 显示当前分支信息: 提交的目的分支信息: git 管理的有修改的文件: 当前仓库未被 git 管理的文 ...

  6. HoloLens开发手记-开发概述Development overview

    开发HoloLens全息应用将使用UWP平台(Universal Windows Platform),所有的HoloLens应用都是Win10通用应用,所有UWP通用应用都可以在HoloLens上运行 ...

  7. iOS-实现后台长时间运行

    前言 一般APP在按下Home键被挂起后,这时APP的 backgroundTimeRemaining 也就是后台运行时间大约只有3分钟,如果在退出APP后,过十几二十二分钟或者更长时间再回到APP, ...

  8. vue中axios的安装和使用

    有很多时候你在构建应用时需要访问一个 API 并展示其数据.做这件事的方法有好几种,而使用基于 promise 的 HTTP 客户端 axios 则是其中非常流行的一种. 安装包:如果没有安装cnpm ...

  9. Python模块——configparser

    configparser模块 该模块适用于配置文件的格式与windows ini文件类似,可以包含一个或多个节(section),每个节可以有多个参数(键=值) 创建文件 import configp ...

  10. dotnet new 命令使用模板生成Angular应用

    dotnet new 命令使用模板快速生成单页应用,本文以Angular应用为例. 最新版.NET Core SDK RC4 最大改动是更新了 dotnet new 命令. dotnet new 默认 ...