-
Notifications
You must be signed in to change notification settings - Fork 19
/
Copy pathGaussElimination.muse
70 lines (52 loc) · 3.76 KB
/
GaussElimination.muse
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
#title 线性方程组与Gauss消元法
<contents>
求解线性方程组与最小二乘问题是科学计算的中心问题之一.根据系数矩阵的性态,我们将采取不同的算法处理.
当要求解的线性方程组中系数矩阵是满秩方阵且无特殊结构 (如对称性质) 时,通常采用Gauss消元法求解.
* 三角方程组
当系数矩阵为下三角矩阵时,通过逐步向前代入可以得到方程组的解.可用MATLAB语法将此算法描述如下:
<algorithm name="向前消去法,行形式">
设$L\in \mathbb{R}^{n\times m}$是下三角矩阵,$b\in\mathbb{R}^n$,则此算法用$Lx=b$的解覆盖$b$.假定$L$是非奇异的.
1. 由$b(1)=b(1)/L(1,1)$求得$x_1$.
2. 对于$i=2:n$,由$b(i)=(b(i)-L(i,1:i-1)b(1:i-1))/L(1,i)$求得$x_2,\cdots,x_n$.
</algorithm>
还有另一种求解形式,可称为列形式的向前消去法:
<algorithm name="向前消去法,列形式">
1. 依次进行如下运算,求得$x_1,\cdots,x_{n-1}$:
<latex>
\begin{align*}
b(j)&=b(j)/L(j,j),\\
b(j+1:n)&=b(j+1:n)-b(j)L(j+1:n,j).
\end{align*}
</latex>
2. 由$b(n)=b(n)/L(n,n)$求得$x_n$.
</algorithm>
对上三角方程组,我们也有相应的方法,称为向后消去法,它也有行形式和列形式.以上各算法均需要$n^2$个flop.由于不同的数据调用模式 (行形式或列形式)对应不同的数据存储格式 (按行或按列存储)具有不同的效率,(这和计算机的内存调用过程有关,)因此,对于不同的数据结构,我们可以合理地选择不同的数据调用模式.Higham<cite>Hig89</cite>指出,三角方程组的求解精度可以达到"惊人"的好.
* 选主元的$LU$分解
对于一般的满秩线性方程组,我们采用Gauss变换将其化为三角方程组求解.这一过程的实质即将系数矩阵$A$分解为下三角阵$L$与上三角阵$U$之积$A=LU$,通过求解三角方程组$Ly=b$及$Ux=y$得到原方程组的解.
Gauss变换对应于矩阵的操作为初等行变换<cite>ZhaXu04</cite>,但仅这两种操作并不能保证算法的成功与稳定性,必须辅之以一定的选主元(pivoting)操作,即通过行(或列)置换将绝对值最大的元素调整到操作对象的位置上,保证行变换的有效性和精确性.主要有全选主元与列选主元两种方式.
全选主元即在第$i$步将主子阵$A(i:n,i:n)$中模最大的元素通过行列置换交换到$A(i,i)$位置,之后通过将第$i$行适当的倍数加到第$i+1:n$行上,将$A(i+1:n,i)$化为0,将$A$更新.对$i=1:n$执行后,得到分解形式:$PAQ=LU$,其中$P$,$Q$为置换阵.此算法需要$\frac{2}{3}n^3$个flop和$O(n^3)$次比较.由于选主元的过程会增加很大的搜索量,所以这种算法通常只用于确定矩阵的数值秩等精度要求很高的运算上,一般并不采用.
综合考虑稳定性与运算复杂性,通常采用如下描述的列选主元过程.所谓列选主元,即在第$i$步运算前通过比较得到第$i$列中模最大的元素并通过行置换将其交换到$A(i,i)$的位置,再采用Gauss变换将矩阵化为上三角形.最终得到分解$PA=LU$,其中$P$为行置换矩阵.这一算法也需要$\frac{2}{3}n^3$次flop,但仅需$O(n^2)$次比较操作.
需要指出,有些特殊情形是不必选主元的.比较明显的一个例子是系数矩阵为对角严格占优矩阵.
* 迭代法改进问题精度
设$Ax=b$通过列选主元$PA=LU$已求解,并且我们需要改进计算值$\hat{x}$的精度.如果我们执行如下的迭代改进
<latex>
\begin{align*}
&r=b-A\hat{x},\\
&\text{解}{}Ly=Pr,\\
&\text{解}{}Uz=y,\\
&x_\text{new}=x+z.
\end{align*}
</latex>
则在精确计算下,我们将得到原方程组的精确解.但由于有限精度计算导致$r$的精度只有很少几位有效数字,我们事实上得到的$x_\text{new}$不会比$\hat{x}$有更多精度.因此,我们有必要用扩充精度的浮点数运算来计算
余量$r=b-A\hat{x}$.典型地,如果$t$位数运算用来计算$PA=LU$,$x$,$y$和$z$,则用$2t$位数运算,即双精度来计算余量$r=b-A\hat{x}$,这一过程可以迭代,即一旦我们计算得到$PA=LU$和初始化$x=0$,我们可重复以下过程:
<latex>
\begin{align*}
&r=b-Ax\text{(双精度)},\\
&\text{解}Ly=Pr\text{得}y,\\
&\text{解}Uz=y\text{得}z,\\
&x_\text{new}=x+z.
\end{align*}
</latex>
我们称此过程为混合精度迭代改进.在双精度计算$r$时,必须用原始的$A$.其基本结论可粗略地叙述如下:
如果机器精度$u$和条件数满足$u\approx 10^{-d}$和$\kappa_\infty\approx 10^q$,则执行迭代改进$k$次之后,$x$有大约$\min{\{d,k(d-q)\}}$位正确的有效数字.
粗略地说,如果$u\kappa_\infty(A) \leqslant 1$,则迭代改进完全可给出一个全(单精度)的正确的解.注意到此过程是相对经济的,每次改进工作量$o(n^2)$,相比之下原始的分解$PA=LU$的工作量为$O(n^3)$.当然,若$A$相对于机器精度是足够坏的条件数,则得不到任何改进.混合精度的迭代改进的一个主要缺点是它的实现是与机器相关的,另一个不足之处是需要保留$A$的原始数据.