数值计算方法 Chapter6. 解线性方程组的迭代法

0. 问题描述

这一章节要解的问题和上一章是一样的,依然还是 n n n元线性方程组的求解问题。

{ a 11 x 1 + a 12 x 2 + . . . + a 1 n x n = y 1 a 21 x 1 + a 22 x 2 + . . . + a 2 n x n = y 2 . . . a n 1 x 1 + a n 2 x 2 + . . . + a n n x n = y n \left\{ \begin{aligned} a_{11}x_1 + a_{12}x_2 + ... + a_{1n}x_n &= y_1 \\ a_{21}x_1 + a_{22}x_2 + ... + a_{2n}x_n &= y_2 \\ ... \\ a_{n1}x_1 + a_{n2}x_2 + ... + a_{nn}x_n &= y_n \end{aligned} \right. a11x1+a12x2+...+a1nxna21x1+a22x2+...+a2nxn...an1x1+an2x2+...+annxn=y1=y2=yn

但是,上一章主要是通过矩阵的线性变换转换成可以快速求解的三角阵或者对角阵的方式进行求解,其计算结果是精确的结果。

而本章则是的思路则是将问题 A x = y Ax=y Ax=y转换成 x = A ′ x x = A'x x=Ax的迭代形式,从而,我们就可以给出迭代数组 x k + 1 = A ′ x k x_{k+1} = A'x_{k} xk+1=Axk

此时,如果 x k x_k xk满足收敛条件,那么 x k x_{k} xk就会收敛到 x = A ′ x x = A'x x=Ax的一组解当中,上述问题同样可以得到解答。

1. Jacobi迭代

1. Jacobi迭代方法

对原始方程进行线性变换,我们显然有:

{ x 1 = − a 12 a 11 x 2 − . . . − a 1 n a 11 x n + y 1 a 11 x 2 = − a 21 a 22 x 2 − . . . − a 2 n a 22 x n + y 2 a 22 . . . x n = − a n 1 a n n x 1 − a n 2 a n n x 2 − . . . + y n a n n \left\{ \begin{aligned} x_{1} = & &-\frac{a_{12}}{a_{11}}x_2 &-... &-\frac{a_{1n}}{a_{11}}x_n &+ \frac{y_1}{a_{11}} \\ x_{2} = &-\frac{a_{21}}{a_{22}}x_2 & &- ... &-\frac{a_{2n}}{a_{22}}x_n &+ \frac{y_2}{a_{22}} \\ ... \\ x_{n} = &-\frac{a_{n1}}{a_{nn}}x_1 &-\frac{a_{n2}}{a_{nn}}x_2 &- ... & &+ \frac{y_n}{a_{nn}} \end{aligned} \right. x1=x2=...xn=a22a21x2annan1x1a11a12x2annan2x2.........a11a1nxna22a2nxn+a11y1+a22y2+annyn

我们将其写成矩阵形式即为:

( x 1 x 2 . . . x n ) = ( 0 − a 12 a 11 . . . − a 1 n a 11 − a 21 a 22 0 . . . − a 2 n a 22 . . . . . . . . . . . . − a n 1 a n n − a n 2 a n n . . . 0 ) ( x 1 x 2 . . . x n ) + ( y 1 a 11 y 2 a 22 . . . y n a n n ) \begin{pmatrix} x_1 \\ x_2 \\ ... \\ x_n \end{pmatrix} = \begin{pmatrix} 0 & -\frac{a_{12}}{a_{11}} & ... & -\frac{a_{1n}}{a_{11}} \\ -\frac{a_{21}}{a_{22}} & 0 & ... & -\frac{a_{2n}}{a_{22}} \\ ... & ... & ... & ... \\ -\frac{a_{n1}}{a_{nn}} & -\frac{a_{n2}}{a_{nn}} & ... & 0 \end{pmatrix} \begin{pmatrix} x_1 \\ x_2 \\ ... \\ x_n \end{pmatrix} + \begin{pmatrix} \frac{y_1}{a_{11}} \\ \frac{y_2}{a_{22}} \\ ... \\ \frac{y_n}{a_{nn}} \end{pmatrix} x1x2...xn=0a22a21...annan1a11a120...annan2............a11a1na22a2n...0x1x2...xn+a11y1a22y2...annyn

我们将其简化为符号表达即为:

x = B x + g x = Bx + g x=Bx+g

基于此,我们就可以给出Jacobi迭代公式如下:

x k + 1 = B x k + g x_{k+1} = Bx_{k} + g xk+1=Bxk+g

2. Jacobi迭代矩阵

我们给出更加严格的数据表达如下,令:

D = ( a 11 a 22 . . . a n n ) D = \begin{pmatrix} a_{11} \\ & a_{22} \\ & & ... \\ & & & a_{nn} \end{pmatrix} D=a11a22...ann

则有:

A x = ( D + A − D ) x = y Ax = (D + A - D)x = y Ax=(D+AD)x=y

进而可以导出:

D x = ( D − A ) x + y Dx = (D-A)x + y Dx=(DA)x+y

两边左乘 D − 1 D^{-1} D1即有:

x = D − 1 ( D − A ) x + D − 1 y x = D^{-1}(D-A)x + D^{-1}y x=D1(DA)x+D1y

B = D − 1 ( D − A ) B=D^{-1}(D-A) B=D1(DA) g = D − 1 y g=D^{-1}y g=D1y,即可得到上一小节中一模一样的内容。

迭代矩阵 B B B即为Jacobi迭代矩阵。

3. Jacobi迭代收敛条件

Jacobi迭代收敛的充要条件为:

迭代矩阵 B B B的谱半径 ρ ( B ) < 1 \rho(B) < 1 ρ(B)<1

亦即:

  • 对于迭代矩阵 B B B的所有本征值 λ i \lambda_i λi,均有 ∣ λ i ∣ < 1 |\lambda_i| < 1 λi<1,或者等价的 m a x ( ∣ λ i ∣ ) < 1 max(|\lambda_i|) < 1 max(λi)<1

不过,在一些特定的情况下,上述充要条件可以有适当的放宽。

具体而言,有如下定理:

定理 6.1
若方程组 A x = y Ax=y Ax=y的系数矩阵 A A A,满足下列条件之一,则其Jacobi迭代收敛:
(1) A为行对角优阵,即 ∣ a i i ∣ > ∑ j ≠ i ∣ a i j ∣ |a_{ii}| > \sum_{j \neq i} |a_{ij}| aii>j=iaij i = 1 , 2 , . . . , n i=1,2,...,n i=1,2,...,n
(2) A为列对角优阵,即 ∣ a j j ∣ > ∑ i ≠ j ∣ a i j ∣ |a_{jj}| > \sum_{i \neq j} |a_{ij}| ajj>i=jaij j = 1 , 2 , . . . , n j=1,2,...,n j=1,2,...,n

4. python伪代码实现

最后,我们给出Jacobi迭代的python伪代码如下:

def jacobi_iter(A, y, epsilon=1e-6):
    n = len(A)
    B = [[0 for _ in range(n)] for _ in range(n)]
    g = [0 for _ in range(n)]
    for i in range(n):
        for j in range(n):
            if j == i:
                continue
            B[i][j] = -A[i][j] / A[i][i]
        g[i] = y[i] / A[i][i]
    
    x = [0 for _ in range(n)]
    for _ in range(10**6):
        z = [0 for _ in range(n)]
        for i in range(n):
            z[i] += g[i]
            for j in range(n):
                z[i] += B[i][j] * x[j]
        if max(abs(z[i] - x[i]) for i in range(n)) < epsilon:
            break
        x = z
    return z

2. Gauss-Seidel迭代

1. Gauss-Seidel迭代方法

Gauss-Seidel迭代方程和上述Jacobi迭代事实上是非常相似的,唯一的区别在于说Jacobi迭代是以 x ( k ) x^{(k)} x(k)为整体每次一起进行迭代更新的,而Guass-Seidel迭代则是在计算每一个 x i ( k ) x^{(k)}_{i} xi(k)的时候就是用当前已经迭代计算完成的所有的 x j ( k ) x^{(k)}_{j} xj(k)的结果。

即是说,以每一个元素为单位不断进行迭代更新。

用公式表达即为:

{ x 1 ( k + 1 ) = − a 12 a 11 x 2 ( k ) − . . . − a 1 n a 11 x n ( k ) + y 1 a 11 x 2 ( k + 1 ) = − a 21 a 22 x 1 ( k + 1 ) − . . . − a 2 n a 22 x n ( k ) + y 2 a 22 . . . x n ( k + 1 ) = − a n 1 a n n x 1 ( k + 1 ) − a n 2 a n n x 2 ( k + 1 ) − . . . + y n a n n \left\{ \begin{aligned} x^{(k+1)}_{1} = & &-\frac{a_{12}}{a_{11}}x^{(k)}_2 &-... &-\frac{a_{1n}}{a_{11}}x^{(k)}_n &+ \frac{y_1}{a_{11}} \\ x^{(k+1)}_{2} = &-\frac{a_{21}}{a_{22}}x^{(k+1)}_1 & &- ... &-\frac{a_{2n}}{a_{22}}x^{(k)}_n &+ \frac{y_2}{a_{22}} \\ ... \\ x^{(k+1)}_{n} = &-\frac{a_{n1}}{a_{nn}}x^{(k+1)}_1 &-\frac{a_{n2}}{a_{nn}}x^{(k+1)}_2 &- ... & &+ \frac{y_n}{a_{nn}} \end{aligned} \right. x1(k+1)=x2(k+1)=...xn(k+1)=a22a21x1(k+1)annan1x1(k+1)a11a12x2(k)annan2x2(k+1).........a11a1nxn(k)a22a2nxn(k)+a11y1+a22y2+annyn

2. Gauss-Seidel迭代矩阵

同样的,我们给出更加严格的数学推导如下:

A = D + L + U = ( a 11 a 22 . . . a n n ) + ( 0 a 21 0 . . . a n 1 a n 2 . . . 0 ) ( 0 a 12 . . . a 1 n 0 . . . a 2 n . . . 0 ) \begin{aligned} A &= D + L + U \\ &= \begin{pmatrix} a_{11} \\ & a_{22} \\ & & ... \\ & & & a_{nn} \end{pmatrix} + \begin{pmatrix} 0 \\ a_{21} & 0 \\ ... \\ a_{n1} & a_{n2} & ... & 0 \end{pmatrix} \begin{pmatrix} 0 & a_{12} & ... & a_{1n} \\ & 0 & ... & a_{2n} \\ & & ... \\ & & & 0 \end{pmatrix} \end{aligned} A=D+L+U=a11a22...ann+0a21...an10an2...00a120.........a1na2n0

从而,我们有:

A x = ( D + L + U ) x = y Ax = (D+L+U)x = y Ax=(D+L+U)x=y

亦即:

( D + L ) x = − U x + y (D+L)x = -Ux + y (D+L)x=Ux+y

两边同乘以 ( D + l ) − 1 (D+l)^{-1} (D+l)1即有:

x = − ( D + L ) − 1 U x + ( D + L ) − 1 y x = -(D+L)^{-1}Ux + (D+L)^{-1}y x=(D+L)1Ux+(D+L)1y

S = − ( D + L ) − 1 U S = -(D+L)^{-1}U S=(D+L)1U f = ( D + L ) − 1 y f = (D+L)^{-1}y f=(D+L)1y,我们即可写出Gauss-Seidel迭代公式:

x ( k + 1 ) = S x ( k ) + f x^{(k+1)} = Sx^{(k)} + f x(k+1)=Sx(k)+f

称迭代矩阵 S S S即为Gauss-Seidel迭代矩阵。

当然,如上一小节所述,实际在算法实现当中并没有必要计算出 S S S f f f,只需要根据前面 J a c o b i Jacobi Jacobi迭代矩阵进行实时参数更新即可。

3. Gauss-Seidel迭代收敛条件

同样的,我们给出书中关于Gauss-Seidel迭代的收敛条件如下:

定理6.2
若方程组系数矩阵为行或列对角优时,则Gauss-Seidel迭代收敛。
定理6.3
若方程组系数矩阵 A A A为对称正定阵,则Gauss-Seidel迭代收敛。

4. 伪代码实现

同样的,我们用python给出伪代码如下:

def gauss_seidel_iter(A, y, epsilon=1e-6):
    n = len(A)
    B = [[0 for _ in range(n)] for _ in range(n)]
    g = [0 for _ in range(n)]
    for i in range(n):
        for j in range(n):
            if j == i:
                continue
            B[i][j] = -A[i][j] / A[i][i]
        g[i] = y[i] / A[i][i]
    
    x = [0 for _ in range(n)]
    cnt = 0
    for _ in range(10**6):
        for i in range(n):
            z = g[i]
            for j in range(n):
                z += B[i][j] * x[j]
            if abs(z-x[i]) < epsilon:
                cnt += 1
            else:
                cnt = 0
            x[i] = z

            if cnt >= n:
                break
        if cnt >= n:
            break
    return x

3. 松弛迭代

1. 松弛迭代方法

松弛迭代的原型依然还是之前的Jacobi迭代,不过,和Gauss-Seidel迭代的实时参数更新不同,松弛迭代在这里是对Jacobi迭代式的批次更新以及Gauss-Seidel迭代式的实时更新取了一个折中,通过一个超参 w w w来进行参数更新速度的控制。

具体而言,即为:

{ x 1 ( k + 1 ) = ( 1 − w ) x 1 ( k ) + w ( b 12 x 2 ( k ) + b 13 x 3 ( k ) + . . . + b 1 n x n ( k ) + g 1 ) x 2 ( k + 1 ) = ( 1 − w ) x 2 ( k ) + w ( b 21 x 1 ( k + 1 ) + b 23 x 3 ( k ) + . . . + b 2 n x n ( k ) + g 2 ) . . . x n ( k + 1 ) = ( 1 − w ) x n ( k ) + w ( b n 1 x 1 ( k + 1 ) + b n 2 x 2 ( k + 1 ) + . . . + b n , n − 1 x n − 1 ( k + 1 ) + g n ) \left\{ \begin{aligned} x^{(k+1)}_1 &= (1-w)x^{(k)}_1 + w(b_{12}x^{(k)}_2 + b_{13}x^{(k)}_3 + ... + b_{1n}x^{(k)}_n + g_1) \\ x^{(k+1)}_2 &= (1-w)x^{(k)}_2 + w(b_{21}x^{(k+1)}_1 + b_{23}x^{(k)}_3 + ... + b_{2n}x^{(k)}_n + g_2) \\ ... \\ x^{(k+1)}_n &= (1-w)x^{(k)}_n + w(b_{n1}x^{(k+1)}_1 + b_{n2}x^{(k+1)}_2 + ... + b_{n,n-1}x^{(k+1)}_{n-1} + g_n) \end{aligned} \right. x1(k+1)x2(k+1)...xn(k+1)=(1w)x1(k)+w(b12x2(k)+b13x3(k)+...+b1nxn(k)+g1)=(1w)x2(k)+w(b21x1(k+1)+b23x3(k)+...+b2nxn(k)+g2)=(1w)xn(k)+w(bn1x1(k+1)+bn2x2(k+1)+...+bn,n1xn1(k+1)+gn)

2. 松弛迭代矩阵

同样的,我们给出松弛迭代的数学表达如下:

x ( k + 1 ) = ( 1 − w ) x ( k ) + w ( L ~ x ( k + 1 ) + U ~ x ( k ) + g ) x^{(k+1)} = (1-w)x^{(k)} + w(\tilde{L}x^{(k+1)} + \tilde{U}x^{(k)} + g) x(k+1)=(1w)x(k)+w(L~x(k+1)+U~x(k)+g)

仿照Gauss-Seidel迭代,我们同样可以给出其迭代矩阵格式如下:

x ( k + 1 ) = S w x ( k ) + f x^{(k+1)} = S_w x^{(k)} + f x(k+1)=Swx(k)+f

其中,

{ S w = ( I + w D − 1 L ) − 1 [ ( 1 − w ) I − w D − 1 U ] f = w ( I + w D − 1 L ) − 1 g \left\{ \begin{aligned} S_w &= (I + wD^{-1}L)^{-1}[(1-w)I - wD^{-1}U] \\ f &= w(I + wD^{-1}L)^{-1}g \end{aligned} \right. {Swf=(I+wD1L)1[(1w)IwD1U]=w(I+wD1L)1g

3. 松弛迭代的收敛条件

而松弛迭代的收敛判断存在如下两个定理:

定理 6.4
松弛迭代收敛的必要条件为 0 < w < 2 0 < w < 2 0<w<2
定理 6.5
A A A为对称正定矩阵,则当 0 < w < 2 0 < w < 2 0<w<2时,松弛迭代恒收敛;

特别的,当 0 < w < 1 0 < w < 1 0<w<1时,称其为亚松弛迭代;当 1 < w < 2 1 < w < 2 1<w<2时,称其为超松弛迭代;当 w = 1 w=1 w=1时,迭代退化为Gauss-Seidel迭代。

4. 伪代码实现

最后,我们同样给出松弛迭代的伪代码实现如下:

def loose_iter(A, y, w=0.5, epsilon=1e-6):
    n = len(A)
    B = [[0 for _ in range(n)] for _ in range(n)]
    g = [0 for _ in range(n)]
    for i in range(n):
        for j in range(n):
            if j == i:
                continue
            B[i][j] = -A[i][j] / A[i][i]
        g[i] = y[i] / A[i][i]
    
    x = [0 for _ in range(n)]
    cnt = 0
    for _ in range(10**6):
        for i in range(n):
            z = (1-w) * x[i] + w * g[i]
            for j in range(n):
                z += w * B[i][j] * x[j]
            if abs(z-x[i]) < epsilon:
                cnt += 1
            else:
                cnt = 0
            x[i] = z

            if cnt >= n:
                break
        if cnt >= n:
            break
    return x

4. 逆矩阵计算

最后,我们来考察一下逆矩阵计算。

逆矩阵的计算原则上来说其实算是上述解线性方程组的一个特殊应用,事实上解 n n n个单元向量然后将其解拼接一下就能得到我们的逆矩阵了。

我们这里就不进行详述了,只给出python伪代码实现如下:

def cal_inverse_matrix(A):
    n = len(A)
    res = [[0 for _ in range(n)] for _ in range(n)]
    for j in range(n):
        y = [0 for _ in range(n)]
        y[j] = 1
        x = gauss_seidel_iter(A, y)
        for i in range(n):
            res[i][j] = x[i]
    return res
  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值