-
Notifications
You must be signed in to change notification settings - Fork 44
/
InterpSpline11_test.go
234 lines (217 loc) · 7.27 KB
/
InterpSpline11_test.go
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
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
// InterpSpline11_test
/*
------------------------------------------------------
作者 : Black Ghost
日期 : 2018-12-9
版本 : 0.0.0
------------------------------------------------------
用节点处的一阶导数表示的三次样条插值函数,
一阶导数边界条件
n+1个点, n个区间
理论:
区间[x(i-1), xi]上的三次样条函数表达为:
(x-xi)^2 * [hi+2(x-x(i-1))]
Si(x) = -----------------------------y(i-1) +
hi^3
(x-x(i-1))^2 * [hi+2(xi-x)]
-----------------------------yi +
hi^3
(x-xi)^2 * (x-x(i-1))
-----------------------m(i-1) +
hi^2
(x-x(i-1))^2 * (x-xi)
-----------------------mi
hi^2
令 lambdai = h(i+1)/(hi+h(i+1))
Mi = 1-lambdai = hi/(hi+h(i+1))
y(i+1)-yi yi-y(i-1)
fi = 3(Mi---------- + lambdai-----------)
h(i+1) hi
(i = 1,...,n-1)
则mi可由n+1阶线性方程组求得(利用LEs_Chasing):
|2 M1 || m1 | | f1-l1*m0 |
| l2 2 M2 || m2 | = | f2 |
| ........ || ... | | ... |
| l(n-2) 2 M(n-2)||m(n-2)| | f(n-2) |
| l(n-1) 2 ||m(n-1)| |f(n-1)-M(n-1)mn|
参考 李信真, 车刚明, 欧阳洁, 等. 计算方法. 西北工业大学
出版社, 2000, pp 116-123.
------------------------------------------------------
输入 :
A 数据点矩阵,(n+1)x3,第一列xi;第二列yi;
第三列y'i,且y'i只需给出y'0和y'n
输出 :
B 插值方程系数结果矩阵,从前到后对应从0到3阶,4xn
err 解出标志:false-未解出或达到步数上限;
true-全部解出
------------------------------------------------------
*/
package goNum_test
import (
"math"
"testing"
"github.com/chfenger/goNum"
)
// InterpSpline11 用节点处的一阶导数表示的三次样条插值函数, 一阶导数边界条件
func InterpSpline11(A goNum.Matrix) (goNum.Matrix, bool) {
/*
用节点处的一阶导数表示的三次样条插值函数, 一阶导数边界条件
输入 :
A 数据点矩阵,(n+1)x3,第一列xi;第二列yi;
第三列y'i,且y'i只需给出y'0和y'n
输出 :
B 插值方程系数结果矩阵,从前到后对应从0到3阶,4xn
err 解出标志:false-未解出或达到步数上限;
true-全部解出
*/
var err bool = false
n := A.Rows - 1
sol := goNum.ZeroMatrix(4, n)
BA := goNum.ZeroMatrix(n-1, n-1) //对角占优的三对角矩阵
BB := goNum.ZeroMatrix(n-1, 1) //解向量
BC := goNum.ZeroMatrix(n-1, 1) //值向量
//1解插值函数的一阶导数mi
//1.0.1第一行
if true { //限制变量使用范围
h1 := A.GetFromMatrix(1, 0) - A.GetFromMatrix(0, 0)
h2 := A.GetFromMatrix(2, 0) - A.GetFromMatrix(1, 0)
y0 := A.GetFromMatrix(0, 1)
y1 := A.GetFromMatrix(1, 1)
y2 := A.GetFromMatrix(2, 1)
l1 := h2 / (h1 + h2)
M1 := 1.0 - l1
f1 := 3.0 * (M1*(y2-y1)/h2 + l1*(y1-y0)/h1)
BA.SetMatrix(0, 0, 2.0)
BA.SetMatrix(0, 1, M1)
BC.Data[0] = f1 - l1*A.GetFromMatrix(0, 2)
}
//1.0.2其它行
for i := 2; i < n-1; i++ {
yi_1 := A.GetFromMatrix(i-1, 0)
yi := A.GetFromMatrix(i, 0)
yi1 := A.GetFromMatrix(i+1, 0)
hi := A.GetFromMatrix(i, 0) - A.GetFromMatrix(i-1, 0)
hi1 := A.GetFromMatrix(i+1, 0) - A.GetFromMatrix(i, 0)
lambdai := hi1 / (hi + hi1)
Mi := 1.0 - lambdai
fi := 3.0 * (Mi*(yi1-yi)/hi1 + lambdai*(yi-yi_1)/hi)
//赋予BA
BA.SetMatrix(i-1, i-2, lambdai)
BA.SetMatrix(i-1, i-1, 2.0)
BA.SetMatrix(i-1, i, Mi)
BC.Data[i-1] = fi
}
//1.0.3最后一行
if true { //i=n-1
hn_1 := A.GetFromMatrix(n-1, 0) - A.GetFromMatrix(n-2, 0)
hn := A.GetFromMatrix(n, 0) - A.GetFromMatrix(n-1, 0)
yn_2 := A.GetFromMatrix(n-2, 1)
yn_1 := A.GetFromMatrix(n-1, 1)
yn := A.GetFromMatrix(n, 1)
lambdan_1 := hn / (hn_1 + hn)
Mn_1 := 1.0 - lambdan_1
fn_1 := 3.0 * (Mn_1*(yn-yn_1)/hn + lambdan_1*(yn_1-yn_2)/hn_1)
BA.SetMatrix(n-2, n-3, lambdan_1)
BA.SetMatrix(n-2, n-2, 2.0)
BC.Data[n-2] = fn_1 - Mn_1*A.GetFromMatrix(n, 2)
}
//1.1求解
soltemp, errtemp := goNum.LEs_Chasing(BA, BC)
if errtemp != true {
panic("Error in goNum.InterpSpline11: Solve Error with goNum.LEs_Chasing")
}
for i := 0; i < n-1; i++ {
BB.Data[i] = soltemp.Data[i]
}
//2求解Si(x)
S0 := goNum.ZeroMatrix(4, 1)
S1 := goNum.ZeroMatrix(4, 1)
S2 := goNum.ZeroMatrix(4, 1)
S3 := goNum.ZeroMatrix(4, 1)
for i := 1; i < n+1; i++ {
xi_1 := A.GetFromMatrix(i-1, 0)
xi := A.GetFromMatrix(i, 0)
yi_1 := A.GetFromMatrix(i-1, 1)
yi := A.GetFromMatrix(i, 1)
mi_1 := 0.0
mi := 0.0
if i == 1 {
mi_1 = A.GetFromMatrix(0, 2)
mi = BB.Data[i-1]
} else if i == n {
mi_1 = BB.Data[i-2]
mi = A.GetFromMatrix(n, 2)
} else {
mi_1 = BB.Data[i-2]
mi = BB.Data[i-1]
}
hi := xi - xi_1
temp0 := goNum.ZeroMatrix(4, 1)
temp1 := goNum.ZeroMatrix(4, 1)
//2.1 S0
temp0.Data[2] = 1.0
temp0.Data[1] = -2.0 * xi
temp0.Data[0] = xi * xi
for j := 3; j > 0; j-- {
temp0.Data[j] = 2.0 * temp0.Data[j-1]
temp1.Data[j-1] = (hi - 2.0*xi_1) * temp0.Data[j-1]
S0.Data[j] = (temp0.Data[j] + temp1.Data[j]) * yi_1 / math.Pow(hi, 3.0)
}
S0.Data[0] = temp1.Data[0] * yi_1 / math.Pow(hi, 3.0)
//2.1 S1
temp0 = goNum.ZeroMatrix(4, 1)
temp1 = goNum.ZeroMatrix(4, 1)
temp0.Data[2] = 1.0
temp0.Data[1] = -2.0 * xi_1
temp0.Data[0] = xi_1 * xi_1
for j := 3; j > 0; j-- {
temp0.Data[j] = -2.0 * temp0.Data[j-1]
temp1.Data[j-1] = (hi + 2.0*xi) * temp0.Data[j-1]
S1.Data[j] = (temp0.Data[j] + temp1.Data[j]) * yi / math.Pow(hi, 3.0)
}
S1.Data[0] = temp1.Data[0] * yi / math.Pow(hi, 3.0)
//2.2 S2
temp0 = goNum.ZeroMatrix(4, 1)
temp1 = goNum.ZeroMatrix(4, 1)
temp0.Data[2] = 1.0
temp0.Data[1] = -2.0 * xi
temp0.Data[0] = xi * xi
for j := 3; j > 0; j-- {
temp0.Data[j] = temp0.Data[j-1]
temp1.Data[j-1] = -1.0 * xi_1 * temp0.Data[j-1]
S2.Data[j] = (temp0.Data[j] + temp1.Data[j]) * mi_1 / math.Pow(hi, 2.0)
}
S2.Data[0] = temp1.Data[0] * mi_1 / math.Pow(hi, 2.0)
//2.3 S3
temp0 = goNum.ZeroMatrix(4, 1)
temp1 = goNum.ZeroMatrix(4, 1)
temp0.Data[2] = 1.0
temp0.Data[1] = -2.0 * xi_1
temp0.Data[0] = xi_1 * xi_1
for j := 3; j > 0; j-- {
temp0.Data[j] = temp0.Data[j-1]
temp1.Data[j-1] = -1.0 * xi * temp0.Data[j-1]
S3.Data[j] = (temp0.Data[j] + temp1.Data[j]) * mi / math.Pow(hi, 2.0)
}
S3.Data[0] = temp1.Data[0] * mi / math.Pow(hi, 2.0)
//2.4 Si(x)
for j := 0; j < 4; j++ {
sol.SetMatrix(j, i-1, S0.Data[j]+S1.Data[j]+S2.Data[j]+S3.Data[j])
}
}
err = true
return sol, err
}
func BenchmarkInterpSpline11(b *testing.B) {
A30 := goNum.NewMatrix(4, 3, []float64{0.0, 0.0, 1.0,
1.0, 3.0, 0.0,
2.0, 4.0, 0.0,
3.0, 6.0, 0.0})
for i := 0; i < b.N; i++ {
goNum.InterpSpline11(A30)
// [0 -14/3 30 ]
// [1 15 -37 ]
// [14/3 -28/3 50/3]
// [-8/3 2 -7/3]
}
}