Python-Predicting /在给定数据集的情况下推断未来数据

前端之家收集整理的这篇文章主要介绍了Python-Predicting /在给定数据集的情况下推断未来数据前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。

我是Python的新手.我有一个数据集,我正在尝试使用numPy / sciPy来预测/推断未来的数据点.是否有一种简单的方法来提出适合我当前数据的数学函数(比如,正弦函数),然后我可以将新值传递给该函数以获得我的预测?

这就是我所拥有的,但我不认为它正在做我想要的:

import numpy as np
from scipy.optimize import curve_fit
import matplotlib.pyplot as plt

def main():

    y = [8.3,8.3,7.2,7.8,9.4,10.6,10.0,11.1,12.8,11.7,8.9,6.7,13.3,15.0,15.6,6.0,6.1,12.2,14.4,16.7,8.0,13.9,9.0,12.0,13.0,11.0,5.6,8.3]    
    x = np.array(np.arange(len(y)))        

    fitting_parameters,covariance = curve_fit(fit,x,y)
    a = fitting_parameters[0]
    b = fitting_parameters[1]
    c = fitting_parameters[2]
    d = fitting_parameters[3]

    for x_predict in range(len(y) + 1,len(y) + 24):
        next_x = x_predict
        next_y = fit(next_x,a,b,c,d)

        print("next_x: " + str(next_x))
        print("next_y: " + str(next_y))
        y.append(next_y)

    plt.plot(y)
    plt.show()

def fit(x,d):
    return a*np.sin(b*x + c) + d

我尝试使用curve_fit和univariatespline我的数据,但这只适合我当前的数据并分别平滑我的点数.我的观点是,这些工具只是“适合”我的数据,但实际上并没有给我一个我可以用来获得未来积分的功能.

我以为我可以使用离散傅里叶变换,因为我的数据是周期性的,看起来它可以被描述为正弦和余弦的总和.但是,一旦我从时域获得频域,我就会被困在如何“推断”以预测时域中的未来时段和点:

import numpy as np
import matplotlib.pyplot as plt

mydata = [8.3,8.3] 

sp = np.fft.rfft(mydata)
freq = np.fft.rfftfreq(len(mydata),d= 1.0)

plt.subplot(211)
plt.plot(mydata)
plt.subplot(212)
plt.plot(freq,sp,'r')
plt.show()

我知道推断可能是危险的和不可靠的,但是为了这个项目的目的,我只是想获得一个可以绘制的工作预测函数.

非常感谢您的帮助.

最佳答案
这是一种通过将周期性数据表示为傅立叶级数进行插值的方法.通过采用离散FFT获得傅立叶级数中使用的系数.

我不推荐这个 – 你可以在下面看到插值并不是人们直觉上认为非常好的 – 但是因为我在评论中已经提到它,所以我会跟进并显示一些代码:)

import numpy as np
import matplotlib.pyplot as plt
import scipy.fftpack as fftpack

def fft_inverse(Yhat,x):
    """Based on https://stackoverflow.com/a/4452499/190597 (mtrw)"""
    Yhat = np.asarray(Yhat)
    x = np.asarray(x).reshape(-1,1)
    N = len(Yhat)
    k = np.arange(N)
    total = Yhat * np.exp(1j * x * k * 2 * np.pi / N)
    return np.real(total.sum(axis=1))/N

mydata = [8.3,8.3] 

Yhat = fftpack.fft(mydata)

fig,ax = plt.subplots(nrows=2,sharex=True)
xs = np.arange(len(mydata))
ax[0].plot(xs,mydata)

new_xs = np.linspace(xs.min(),xs.max(),len(mydata)*1.5)
new_ys = fft_inverse(Yhat,new_xs)
ax[1].plot(new_xs,new_ys)

plt.xlim(xs.min(),xs.max())
plt.show()

以下是如何使用scipy.optimize查找适合模型函数的参数,然后可以使用该函数在任意x坐标处进行插值.使用单一罪恶的拟合仍然非常可怕,但我将发布代码只是为了展示如何使用scipy.optimize无论如何:

import numpy as np
import matplotlib.pyplot as plt
import scipy.optimize as optimize

mydata = np.array(
    [8.3,8.3]) 


def fit(x,d):
    return a*np.sin(b*x + c) + d

xs = np.linspace(0,2*np.pi,len(mydata))

guess = (mydata.ptp()/2,10,mydata.mean())
fitting_parameters,covariance = optimize.curve_fit(fit,xs,mydata,p0=guess)
a,d = fitting_parameters
print(a,d)

fig,sharex=True)
ax[0].plot(xs,len(mydata)*1.5)
new_ys = fit(new_xs,d)
ax[1].plot(new_xs,xs.max())
plt.show()

您可以通过选择更好的模型功能(代替合适)来改善贴合度.选择什么是由您的问题领域的先验知识引导的创造力和直觉问题.更好的方法不仅取决于拟合的好坏,还取决于您希望模型的简单性或复杂性,以及/或应用于新数据集时的预测能力.

猜你在找的Python相关文章