雅可比法

来自testwiki
跳转到导航 跳转到搜索

Template:NoteTA

在数值线性代数中,雅可比法Template:Lang)是一种解对角元素几乎都是各行和各列的绝对值最大的值的线性方程组的算法。求解出每个对角元素并插入近似值。不断迭代直至收敛[1]。这个算法是雅可比矩阵的精简版。方法的名字来源于德国数学家卡尔·雅可比

描述

给定一个n×n的线性方程组

A𝐱=𝐛

其中:

A=[a11a12a1na21a22a2nan1an2ann],𝐱=[x1x2xn],𝐛=[b1b2bn].

A 可以分解成对角部分D和剩余部分R:

A=D+R其 中D=[a11000a22000ann],R=[0a12a1na210a2nan1an20]

线性方程组可以重写为:

D𝐱=𝐛R𝐱

雅可比法是一种迭代方法。在每一次迭代中,上一次算出的x被用在右侧,用来算出左侧的新的x。这个过程可以如下表示:

𝐱(k+1)=D1(𝐛R𝐱(k)).


对每个元素可以用以下公式:

xi(k+1)=1aii(bijiaijxj(k)),i=1,2,,n.

注意计算 xi(k+1)需要x(k)中除了自己之外的每个元素。 不像高斯-赛德尔迭代,我们不能用 xi(k+1)覆盖xi(k),因为在接下来的计算中还要用到这些值。这是雅可比和高斯-塞德尔方法最显著的差别,也是为什么前者可以用并行算法而后者不能的原因。最小需要的存储空间是两个长度为n的向量。

算法

选择一个初始猜想值x0

while 未收敛
for i := 1 step until n do
σ=0
for j := 1 step until n do
if j != i then
σ=σ+aijxj(k1)
end if
end (j-loop)
xi(k)=(biσ)aii
end (i-loop)
检查是否收敛
end (未收敛时继续循环)

收敛

标准的收敛情况是当迭代矩阵的谱半径

ρ(D1R)<1.[1]

保证收敛的条件是矩阵A为严格或不可约地对角占优矩阵。严格的行对角占优矩阵指对于每一行,对角线上的元素之绝对值大于其余元素绝对值的和,即

|aii|>ji|aij|.

有时即使不满足此条件,雅可比法仍可收敛。

举例

一个形如 Ax=b 的线性方程,估计初始x(0)

A=[2157], b=[1113],x(0)=[11].

我们用上文描述的方程x(k+1)=D1(bRx(k))来估计x。首先,将等式写为D1(bRx(k))=Tx(k)+C以方便计算,其中T=D1RC=D1b。注意 R=L+U中的 LUA的严格 递增和递减部分。变成如下数值:

D1=[1/2001/7], L=[0050],U=[0100].

T=D1(L+U) as

T=[1/2001/7]{[0050]+[0100]}=[00.50.7140].

解出C为:

C=[1/2001/7][1113]=[5.51.857].

用计算出来的T和C,我们估计xx(1)=Tx(0)+C

x(1)=[00.50.7140][11]+[5.51.857]=[5.01.143].

继续迭代得:

x(2)=[00.50.7140][5.01.143]+[5.51.857]=[4.9291.713].

这个过程一直重复直到收敛(直到Ax(n)b足够小)。这个例子在25次迭代之后的解是

x=[7.1113.222].

一個用Fortran解的例子

subroutine solve(A,b,x,x0,n)
implicit real*8(a-z)
integer::n,imax=200
integer::i,j,k
real*8::tol=1d-15
real*8::A(n,n),b(n),x(n),x0(n),x1(n),x2(n)

write(102,501)
501 format('Jacobi iteration',/)

x1=x0

do k=1,imax
    do i=1,n
        s=0
        do j=1,n
            if (j==i) cycle           
            s=s+A(i,j)*x1(j)      
        enddo   
        x2(i)=(b(i)-s)/A(i,i)  
    enddo
    ! the following step check if convergence is reached
    dx2=0
    do i=1,n
        dx2=dx2+(x1(i)-x2(i))**2
    enddo
    dx2=dsqrt(dx2)
    if (dx2<tol) exit
    x1=x2
    write(102,502)k,x1 ! record the iteration process
    502 format(1x,i3,<n>(1x,1pd25.15))
enddo
x=x2
end subroutine solve

program  main
implicit real*8(a-z)
integer,parameter::n=3
real*8 ::A(n,n),b(n),x(n),x0(n)

open(unit=101,file='result.txt')
open(unit=102,file='it_result.txt')

x0=(/0d0,0d0,0d0/) ! initial guess
b=(/9d0,7d0,6d0/)
A=reshape((/10,-1,0,-1,10,-4,0,-2,10/),(/3,3/))

call solve(A,b,x,x0,n) ! solve Ax=b
  
write(101,501)'x(1)','x(2)','x(3)',x
501 format('jacobi result',//,<n>(1x,a25),/,<n>(1x,1pd25.15))

end program main

参考文献

  1. 1.0 1.1 [1] Template:Wayback,解线性方程组的迭代法

外部链接