4

我有一些代码使用 scipy.integration.cumtrapz 来计算采样信号的反导数。我想使用辛普森规则而不是梯形。然而 scipy.integration.simps 似乎没有累积对应物......我错过了什么吗?有没有一种简单的方法可以与“scipy.integration.simps”进行累积集成?

4

3 回答 3

4

您可以随时编写自己的:

def cumsimp(func,a,b,num):
    #Integrate func from a to b using num intervals.

    num*=2
    a=float(a)
    b=float(b)
    h=(b-a)/num

    output=4*func(a+h*np.arange(1,num,2))
    tmp=func(a+h*np.arange(2,num-1,2))
    output[1:]+=tmp
    output[:-1]+=tmp 
    output[0]+=func(a)
    output[-1]+=func(b)
    return np.cumsum(output*h/3)

def integ1(x):
    return x

def integ2(x):
    return x**2

def integ0(x):
    return np.ones(np.asarray(x).shape)*5

首先看一个常数函数的求和和导数。

print cumsimp(integ0,0,10,5)
[ 10.  20.  30.  40.  50.]

print np.diff(cumsimp(integ0,0,10,5))
[ 10.  10.  10.  10.]

现在检查一些简单的例子:

print cumsimp(integ1,0,10,5)
[  2.   8.  18.  32.  50.]

print cumsimp(integ2,0,10,5)
[   2.66666667   21.33333333   72.          170.66666667  333.33333333]

在这里显式地编写你的被积函数比在这种情况下重现辛普森的 scipy 规则函数要容易得多。当提供单个数组时,选择间隔将很难做到,您是否:

  • 使用所有其他值作为辛普森规则的边缘,其余值作为中心?
  • 使用数组作为边缘并插入中心值?

对于您希望如何对间隔求和,还有一些选项。这些并发症可能是它没有用 scipy 编码的原因。

于 2013-08-13T17:31:42.397 回答
0

我会选择丹尼尔的解决方案。但是,如果您正在积分的功能本身会受到波动的影响,您需要小心。辛普森要求函数表现良好(在这种情况下,意思是连续的)。

有一些技术可以使表现中等的函数看起来比实际表现更好(实际上是函数的近似形式),但在这种情况下,您必须确保该函数“充分”近似于您的函数。在这种情况下,您可能会使间隔可能不均匀以处理问题。

一个例子可能是考虑一个场的流动,在较长的时间尺度上,它被一个表现良好的函数逼近,但在较短的时间内,它的密度受到有限的随机波动的影响。

于 2021-09-28T10:27:44.810 回答
0

你的问题很久以前就已经回答了,但我最近遇到了同样的问题。我写了一些函数来计算等距点的累积积分;代码可以在GitHub上找到。插值多项式的顺序范围从 1(梯形规则)到 7。正如 Daniel 在上一个答案中指出的那样,必须对如何对区间求和做出一些选择,尤其是在边界处;因此,根据您使用的软件包,结果可能会略有不同。另请注意,对于高阶多项式,数值积分可能会受到龙格现象(意外振荡)的影响。

这是一个例子:

import numpy as np
from scipy import integrate as sp_integrate
from gradiompy import integrate as gp_integrate

# Definition of the function (polynomial of degree 7)
x = np.linspace(-3,3,num=15)
dx = x[1]-x[0]
y = 8*x + 3*x**2 + x**3 - 2*x**5 + x**6 - 1/5*x**7
y_int = 4*x**2 + x**3 + 1/4*x**4 - 1/3*x**6 + 1/7*x**7 - 1/40*x**8

# Cumulative integral using scipy
y_int_trapz = y_int [0] + sp_integrate.cumulative_trapezoid(y,dx=dx,initial=0)
print('Integration error using scipy.integrate:')
print('  trapezoid = %9.5f' % np.linalg.norm(y_int_trapz-y_int))

# Cumulative integral using gradiompy
y_int_trapz = gp_integrate.cumulative_trapezoid(y,dx=dx,initial=y_int[0])
y_int_simps = gp_integrate.cumulative_simpson(y,dx=dx,initial=y_int[0])
print('\nIntegration error using gradiompy.integrate:')
print('  trapezoid = %9.5f' % np.linalg.norm(y_int_trapz-y_int))
print('  simpson   = %9.5f' % np.linalg.norm(y_int_simps-y_int))

# Higher order cumulative integrals
for order in range(5,8,2):
    y_int_composite = gp_integrate.cumulative_composite(y,dx,order=order,initial=y_int[0])
    print('  order %i   = %9.5f' % (order,np.linalg.norm(y_int_composite-y_int)))
    
# Display the values of the cumulative integral
print('\nCumulative integral (with initial offset):\n',y_int_composite)

您应该得到以下结果:

'''
Integration error using scipy.integrate:
  trapezoid = 176.10502

Integration error using gradiompy.integrate:
  trapezoid = 176.10502
  simpson   =   2.52551
  order 5   =   0.48758
  order 7   =   0.00000

Cumulative integral (with initial offset):
 [-6.90203571e+02 -2.29979407e+02 -5.92267425e+01 -7.66415188e+00
  2.64794452e+00  2.25594840e+00  6.61937372e-01  1.14797061e-13
  8.20130517e-01  3.61254267e+00  8.55804341e+00  1.48428883e+01
  1.97293221e+01  1.64257877e+01 -1.13464286e+01]
'''
于 2021-05-20T11:32:32.893 回答