-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathInverse_matrix.f95
89 lines (82 loc) · 3.12 KB
/
Inverse_matrix.f95
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
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
! 通过求解矩阵方程得到矩阵的逆矩阵。
MODULE NUMERICAL
IMPLICIT NONE
PRIVATE UPPER
PRIVATE LOWER
PUBLIC SHOWMAT
PUBLIC INVERSE_MAT
CONTAINS
! 打印矩阵
SUBROUTINE SHOWMAT(MAT)
IMPLICIT NONE
REAL, INTENT(IN) :: MAT(:, :)
INTEGER :: I
DO I=1, SIZE(MAT, 2)
WRITE(*, "(3F8.4)") MAT(:, I)
END DO
RETURN
END SUBROUTINE
! 上三角化系数矩阵,同时对等号右边列向量做同等行变换
SUBROUTINE UPPER(MAT, E)
IMPLICIT NONE
REAL, INTENT(INOUT) :: MAT(:, :)
REAL, INTENT(INOUT) :: E(:, :) ! 单位矩阵
INTEGER :: I, J
DO I=1, SIZE(MAT, 2)-1
DO J=I+1, SIZE(MAT, 2)
! 注意,100语句与200语句的顺序不能调换!
100 E(:, J) = E(:, J) - E(:, I) * MAT(I, J) / MAT(I, I)
200 MAT(:, J) = MAT(:, J) - MAT(I, J) * MAT(:, I) / MAT(I, I)
END DO
END DO
RETURN
END SUBROUTINE
! 下三角化系数矩阵,同时对等号右边列向量做同等行变换
SUBROUTINE LOWER(MAT, E)
IMPLICIT NONE
REAL, INTENT(INOUT) :: MAT(:, :)
REAL, INTENT(INOUT) :: E(:, :)
INTEGER :: I, J
! 进行矩阵的行变换
DO I=SIZE(MAT, 2), 2, -1
DO J=I-1, 1, -1
! 注意,300语句与400语句的顺序不能调换!
300 E(:, J) = E(:, J) - E(:, I) * MAT(I, J) / MAT(I, I)
400 MAT(:, J) = MAT(:, J) - MAT(I, J) * MAT(:, I) / MAT(I, I)
END DO
END DO
RETURN
END SUBROUTINE
! 求逆矩阵
SUBROUTINE INVERSE_MAT(MAT, E)
IMPLICIT NONE
REAL, INTENT(INOUT) :: MAT(:, :) ! 系数矩阵
REAL, INTENT(INOUT) :: E(:, :)
INTEGER :: I, J
CALL SHOWMAT(MAT)
WRITE(*, *) "------------------"
CALL UPPER(MAT, E)
CALL SHOWMAT(MAT)
WRITE(*, *) "------------------"
CALL LOWER(MAT, E)
CALL SHOWMAT(MAT)
WRITE(*, *) "------------------"
! 得出MAT的逆矩阵
FORALL(I=1:SIZE(MAT, 2)) E(:, I) = E(:, I) / MAT(I, I)
RETURN
END SUBROUTINE
END MODULE
PROGRAM MAIN
USE NUMERICAL
IMPLICIT NONE
REAL :: MAT(3, 3) = (/ 1, 1, 3, 3, 2, 2, 1, 3, 1 /)
REAL :: BACKUP(3, 3) = (/ 1, 1, 3, 3, 2, 2, 1, 3, 1 /)
INTEGER :: I, J
REAL :: E(3, 3)
! 生成单位矩阵
FORALL(I=1:3, J=1:3, I==J) E(I, J) = 1.0
FORALL(I=1:3, J=1:3, I/=J) E(I, J) = 0.0
CALL INVERSE_MAT(MAT, E)
CALL SHOWMAT(E)
WRITE(*, *) MATMUL(BACKUP, E) ! 使用原矩阵与求得的逆矩阵进行乘积,看看是否是单位矩阵
END PROGRAM