-
Notifications
You must be signed in to change notification settings - Fork 44
/
IntegralNewtonCotes_test.go
163 lines (148 loc) · 3.98 KB
/
IntegralNewtonCotes_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
// IntegralNewtonCotes_test
/*
------------------------------------------------------
作者 : Black Ghost
日期 : 2018-12-11
版本 : 0.0.0
------------------------------------------------------
1-8级Newton-Cotes求积分公式
理论:
对于积分
b n
|f(x)dx ~= Sum Ak*f(xk)
a k=0
(n)
Ak = (b-a)C
k
(n) (-1)^(n-k) n
C = ------------ |t(t-1)(t-2)...(t-(k-1))(t-(k+1))...(t-n)dt
k k!(n-k)!n 0
特别的,n=1为梯形公式;
n=2为Simpson(辛浦生)公式;
n=4为Cotes(科特斯)公式
参考 李信真, 车刚明, 欧阳洁, 等. 计算方法. 西北工业大学
出版社, 2000, pp 145-153.
------------------------------------------------------
输入 :
fun 被积分函数
a, b 积分范围
n Newton-Cotes公式级数
输出 :
sol 解
err 解出标志:false-未解出或达到步数上限;
true-全部解出
------------------------------------------------------
注意 :
由于误差得不到有效控制,稳定性无法保证,故而并不是n值越
大越好,实际应用中很少使用n值较大的Newton-Cotes公式
------------------------------------------------------
*/
package goNum_test
import (
"math"
"testing"
"github.com/chfenger/goNum"
)
// IntegralNewtonCotes 1-8级Newton-Cotes求积分公式
func IntegralNewtonCotes(fun func(float64) float64, a, b float64, n int) (float64, bool) {
/*
1-8级Newton-Cotes求积分公式
输入 :
fun 被积分函数
a, b 积分范围
n Newton-Cotes公式级数
输出 :
sol 解
err 解出标志:false-未解出或达到步数上限;
true-全部解出
*/
//判断n
if (n < 1) || (n > 8) {
panic("Error in goNum.IntegralNewtonCotes: n is not correct")
}
//判断a, b
if a == b {
return 0.0, true
}
var sol float64
var err bool = false
//计算xi
xi := goNum.ZeroMatrix(n+1, 1)
for i := 0; i < n+1; i++ {
xi.Data[i] = a + (b-a)*float64(i)/float64(n)
}
//系数切片
coeff := [][]float64{
{1.0, 1.0},
{1.0, 4.0, 1.0},
{1.0, 3.0, 3.0, 1.0},
{7.0, 32.0, 12.0, 32.0, 7.0},
{19.0, 75.0, 50.0, 50.0, 75.0, 19.0},
{41.0, 216.0, 27.0, 272.0, 27.0, 216.0, 41.0},
{751.0, 3577.0, 1323.0, 2989.0, 2989.0, 1323.0, 3577.0, 751.0},
{989.0, 5888.0, -928.0, 10496.0, -4540.0, 10496.0, -928.0, 5888.0, 989.0},
}
//计算积分值
switch n {
case 1:
for i := 0; i < n+1; i++ {
sol += coeff[0][i] * fun(xi.Data[i])
}
sol = sol * (b - a) / 2.0
return sol, true
case 2:
for i := 0; i < n+1; i++ {
sol += coeff[1][i] * fun(xi.Data[i])
}
sol = sol * (b - a) / 6.0
return sol, true
case 3:
for i := 0; i < n+1; i++ {
sol += coeff[2][i] * fun(xi.Data[i])
}
sol = sol * (b - a) / 8.0
return sol, true
case 4:
for i := 0; i < n+1; i++ {
sol += coeff[3][i] * fun(xi.Data[i])
}
sol = sol * (b - a) / 90.0
return sol, true
case 5:
for i := 0; i < n+1; i++ {
sol += coeff[4][i] * fun(xi.Data[i])
}
sol = sol * (b - a) / 288.0
return sol, true
case 6:
for i := 0; i < n+1; i++ {
sol += coeff[5][i] * fun(xi.Data[i])
}
sol = sol * (b - a) / 840.0
return sol, true
case 7:
for i := 0; i < n+1; i++ {
sol += coeff[6][i] * fun(xi.Data[i])
}
sol = sol * (b - a) / 17280.0
return sol, true
case 8:
for i := 0; i < n+1; i++ {
sol += coeff[7][i] * fun(xi.Data[i])
}
sol = sol * (b - a) / 28350.0
return sol, true
default:
return 0.0, err
}
return sol, err
}
func fun34(x float64) float64 {
return math.Sqrt(x)
}
func BenchmarkIntegralNewtonCotes(b *testing.B) {
for i := 0; i < b.N; i++ {
// goNum.IntegralNewtonCotes(fun34, 0.5, 1.0, 1) //~0.43096
goNum.IntegralNewtonCotes(fun34, 0.5, 1.0, 3)
}
}