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:
		 David Kleiven
					David Kleiven
				
			
				
					committed by
					
						 Vladimír Chalupecký
						Vladimír Chalupecký
					
				
			
			
				
	
			
			
			 Vladimír Chalupecký
						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