mirror of
https://github.com/gonum/gonum.git
synced 2025-10-27 01:00:26 +08:00
integrate: Add Romberg integration
This commit is contained in:
committed by
Vladimír Chalupecký
parent
94c839842d
commit
a07f9102d0
65
integrate/romberg.go
Normal file
65
integrate/romberg.go
Normal file
@@ -0,0 +1,65 @@
|
||||
// Copyright ©2019 The Gonum Authors. All rights reserved.
|
||||
// Use of this source code is governed by a BSD-style
|
||||
// license that can be found in the LICENSE file.
|
||||
|
||||
package integrate
|
||||
|
||||
import (
|
||||
"math"
|
||||
"math/bits"
|
||||
)
|
||||
|
||||
// Romberg returns an approximate value of the integral
|
||||
// \int_a^b f(x)dx
|
||||
// computed using the Romberg's method. The function f is given
|
||||
// as a slice of equally-spaced samples, that is,
|
||||
// f[i] = f(a + i*dx)
|
||||
// and dx is the spacing between the samples.
|
||||
//
|
||||
// The length of f must be 2^k + 1, where k is a positive integer,
|
||||
// and dx must be positive.
|
||||
//
|
||||
// See https://en.wikipedia.org/wiki/Romberg%27s_method for a description of
|
||||
// the algorithm
|
||||
func Romberg(f []float64, dx float64) float64 {
|
||||
if len(f) < 3 {
|
||||
panic("integral: invalid slice length: must be at least 3")
|
||||
}
|
||||
|
||||
if dx <= 0 {
|
||||
panic("integral: invalid spacing: must be larger than 0")
|
||||
}
|
||||
|
||||
n := len(f) - 1
|
||||
k := bits.Len(uint(n - 1))
|
||||
|
||||
if len(f) != 1<<uint(k)+1 {
|
||||
panic("integral: invalid slice length: must be 2^k + 1")
|
||||
}
|
||||
|
||||
work := make([]float64, 2*(k+1))
|
||||
prev := work[:k+1]
|
||||
curr := work[k+1:]
|
||||
|
||||
h := dx * float64(n)
|
||||
prev[0] = (f[0] + f[len(f)-1]) * 0.5 * h
|
||||
|
||||
step := n
|
||||
for i := 1; i <= k; i++ {
|
||||
h /= 2
|
||||
step /= 2
|
||||
var estimate float64
|
||||
for j := 0; j < n/2; j += step {
|
||||
estimate += f[2*j+step]
|
||||
}
|
||||
|
||||
curr[0] = estimate*h + 0.5*prev[0]
|
||||
for j := 1; j <= i; j++ {
|
||||
factor := math.Pow(4, float64(j))
|
||||
curr[j] = (factor*curr[j-1] - prev[j-1]) / (factor - 1)
|
||||
}
|
||||
|
||||
prev, curr = curr, prev
|
||||
}
|
||||
return prev[k]
|
||||
}
|
||||
86
integrate/romberg_test.go
Normal file
86
integrate/romberg_test.go
Normal file
@@ -0,0 +1,86 @@
|
||||
// Copyright ©2019 The Gonum Authors. All rights reserved.
|
||||
// Use of this source code is governed by a BSD-style
|
||||
// license that can be found in the LICENSE file.
|
||||
|
||||
package integrate
|
||||
|
||||
import (
|
||||
"math"
|
||||
"testing"
|
||||
|
||||
"gonum.org/v1/gonum/floats"
|
||||
)
|
||||
|
||||
func TestRomberg(t *testing.T) {
|
||||
const n = 1<<8 + 1
|
||||
x := floats.Span(make([]float64, n), 0, 1)
|
||||
|
||||
for i, test := range []struct {
|
||||
x []float64
|
||||
f func(x float64) float64
|
||||
want float64
|
||||
tol float64
|
||||
}{
|
||||
{
|
||||
x: x,
|
||||
f: func(x float64) float64 { return x },
|
||||
want: 0.5,
|
||||
tol: 1e-12,
|
||||
},
|
||||
{
|
||||
x: x,
|
||||
f: func(x float64) float64 { return x * x },
|
||||
want: 1.0 / 3.0,
|
||||
tol: 1e-12,
|
||||
},
|
||||
{
|
||||
x: x,
|
||||
f: func(x float64) float64 { return x * x * x },
|
||||
want: 1.0 / 4.0,
|
||||
tol: 1e-12,
|
||||
},
|
||||
{
|
||||
x: x,
|
||||
f: func(x float64) float64 { return math.Sqrt(x) },
|
||||
want: 2.0 / 3.0,
|
||||
tol: 1e-4,
|
||||
},
|
||||
{
|
||||
x: x,
|
||||
f: func(x float64) float64 { return math.Sin(math.Pi * x) },
|
||||
want: 2.0 / math.Pi,
|
||||
tol: 1e-12,
|
||||
},
|
||||
{
|
||||
x: floats.Span(make([]float64, 3), 0, 1),
|
||||
f: func(x float64) float64 { return x * x },
|
||||
want: 1.0 / 3.0,
|
||||
tol: 1e-12,
|
||||
},
|
||||
{
|
||||
x: floats.Span(make([]float64, 3), 0, 1),
|
||||
f: func(x float64) float64 { return x * x * x },
|
||||
want: 1.0 / 4.0,
|
||||
tol: 1e-12,
|
||||
},
|
||||
{
|
||||
x: x,
|
||||
f: func(x float64) float64 { return x * math.Exp(-x) },
|
||||
want: (math.Exp(1) - 2) / math.Exp(1),
|
||||
tol: 1e-12,
|
||||
},
|
||||
} {
|
||||
n := len(test.x)
|
||||
y := make([]float64, n)
|
||||
for i, v := range test.x {
|
||||
y[i] = test.f(v)
|
||||
}
|
||||
|
||||
dx := (test.x[n-1] - test.x[0]) / float64(n-1)
|
||||
v := Romberg(y, dx)
|
||||
diff := math.Abs(v - test.want)
|
||||
if diff > test.tol {
|
||||
t.Errorf("test #%d: got=%v want=%v diff=%v\n", i, v, test.want, diff)
|
||||
}
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user