Линейная интерполяция с использованием numpy.interp
У меня есть одномерный массив A с плавающей точкой, который в основном хорош, но некоторые значения отсутствуют. Отсутствующие данные заменить на nan(не число). Я должен заменить отсутствующие значения в массиве линейной интерполяцией из близких хороших значений. Так, например:
F7(np.array([10.,20.,nan,40.,50.,nan,30.]))
должен вернуться
np.array([10.,20.,30.,40.,50.,40.,30.]).
Каков наилучший способ сделать это с помощью Python?
Любая помощь приветствуется
Спасибо
3 ответа
Вы могли бы использовать scipy.interpolate.interp1d
:
>>> from scipy.interpolate import interp1d
>>> import numpy as np
>>> x = np.array([10., 20., np.nan, 40., 50., np.nan, 30.])
>>> not_nan = np.logical_not(np.isnan(x))
>>> indices = np.arange(len(x))
>>> interp = interp1d(indices[not_nan], x[not_nan])
>>> interp(indices)
array([ 10., 20., 30., 40., 50., 40., 30.])
РЕДАКТИРОВАТЬ: мне потребовалось время, чтобы выяснить, как np.interp
работает, но это может сделать работу также:
>>> np.interp(indices, indices[not_nan], x[not_nan])
array([ 10., 20., 30., 40., 50., 40., 30.])
Я бы пошел с pandas
, Минималистичный подход с oneliner:
from pandas import *
a=np.array([10.,20.,nan,40.,50.,nan,30.])
Series(a).interpolate()
Out[219]:
0 10
1 20
2 30
3 40
4 50
5 40
6 30
Или, если вы хотите сохранить его в виде массива:
Series(a).interpolate().values
Out[221]:
array([ 10., 20., 30., 40., 50., 40., 30.])
Чтобы не создавать новый объект Series или новые элементы в Series каждый раз, когда вы хотите интерполировать данные, используйте RedBlackPy. Смотрите пример кода ниже:
import redblackpy as rb
# we do not include missing data
index = [0,1,3,4,6]
data = [10,20,40,50,30]
# create Series object
series = rb.Series(index=index, values=data, dtype='float32',
interpolate='linear')
# Now you have access at any key using linear interpolation
# Interpolation does not creates new items in Series
print(series[2]) # prints 30
print(series[5]) # prints 40
# print Series and see that keys 2 and 5 do not exist in series
print(series)
Последний вывод следующий:
Series object Untitled
0: 10.0
1: 20.0
3: 40.0
4: 50.0
6: 30.0