146 lines
3.6 KiB
Go
146 lines
3.6 KiB
Go
// IntegralNewtonCotes
|
||
/*
|
||
------------------------------------------------------
|
||
作者 : 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
|
||
|
||
// 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 := 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
|
||
}
|