python - 将函数应用于多索引多列数据帧的 Pythonic 方法是什么?

标签 python numpy pandas scikit-learn linear-regression

给定下面的多索引多列数据框,我想将 LinearRegression 应用于此数据框的每个 block ,例如,“索引(X,1),列 A”。并将预测数据帧计算为 df_result。

                           A    B
X   1   1997-01-31  -0.061332   0.630682
        1997-02-28  -2.671818   0.377036
        1997-03-31  0.861159    0.303689
        ...
        1998-01-31  0.535192    -0.076420
        ...
        1998-12-31  1.430995    -0.763758
Y   1   1997-01-31  -0.061332   0.630682
        1997-02-28  -2.671818   0.377036
        1997-03-31  0.861159    0.303689
        ...
        1998-01-31  0.535192    -0.076420
        ...
        1998-12-31  1.430995    -0.763758

这是我尝试过的:

import pandas as pd
import numpy as np 
from sklearn.linear_model import LinearRegression

N = 24
dates = pd.date_range('19970101', periods=N, freq='M')
df=pd.DataFrame(np.random.randn(len(dates),2),index=dates,columns=list('AB')) 
df2=pd.concat([df,df],keys=[('X','1'),('Y','1')])

regr = LinearRegression()

# df_result will be reassined, copy the index and metadata from df2
df_result=df2.copy()

# I know the double loop below is not a clever idea. What is the right way?
for row in df2.index.to_series().unique():
    for col in df2.columns:
        #df2 can contain missing values
        lenX=np.count_nonzero(df2.ix[row[:1],col].notnull().values.ravel())
        X=np.array(range(lenX)).reshape(lenX,1)
        y=df2.ix[row[:1],col]
        y=y[y.notnull()]

        # train the model
        regr.fit(X,y)

        df_result.ix[row[:1],col][:lenX] = regr.predict(X)

问题是上面的双循环使得计算速度很慢,100kb的数据集需要十多分钟。执行此操作的 pythonic 方法是什么?

编辑:

上面代码最后一行的第二个问题是我正在处理数据帧切片的副本。此操作未更新“df_result”的某些列。

编辑 2:

原始数据的某些列可能包含缺失值,我们不能直接对其应用回归。例如,

df2.ix[('X','1','1997-12-31')]['A']=np.nan
df2.ix[('Y','1','1998-12-31')]['A']=np.nan

最佳答案

我不太明白行循环。

无论如何,为了保持数字的一致性,我将 np.random.seed(1) 放在了顶部

简而言之,我认为您可以通过函数、groupby 和调用 .transform() 来实现您想要的。

def do_regression(y):
    X=np.array(range(len(y))).reshape(len(y),1)
    regr.fit(X,y)
    return regr.predict(X)

df_regressed = df2.groupby(level=[0,1]).transform(do_regression)

print df_regressed.head()

                       A         B
X 1 1997-01-31  0.779476 -1.222119
    1997-02-28  0.727184 -1.138630
    1997-03-31  0.674892 -1.055142
    1997-04-30  0.622601 -0.971653
    1997-05-31  0.570309 -0.888164

与您的 df_result 输出匹配。

print df_result.head()

                       A         B
X 1 1997-01-31  0.779476 -1.222119
    1997-02-28  0.727184 -1.138630
    1997-03-31  0.674892 -1.055142
    1997-04-30  0.622601 -0.971653
    1997-05-31  0.570309 -0.888164

哦,还有几个替代方案:

X=np.array(range(len(y))).reshape(len(y),1)

1.) X = np.expand_dims(range(len(y)), axis=1)
2.) X = np.arange(len(y))[:,np.newaxis]

为空数据编辑

确定 2 个建议:

使用 interpolate 方法填充空值是否合法?

df2 = df2.interpolate()

对非空值进行回归,然后将空值弹出到适当的索引位置

   def do_regression(y):

        x_s =np.arange(len(y))
        x_s_non_nulls =  x_s[y.notnull().values]
        x_s_non_nulls = np.expand_dims(x_s_non_nulls, axis=1)

        y_non_nulls = y[y.notnull()]  # get the non nulls

        regr.fit(x_s_non_nulls,y_non_nulls)  # regression
        results = regr.predict(x_s_non_nulls)

        #pop back in then nulls.
        for idx in np.where(y.isnull().values ==True):
            results = np.insert(results,idx,np.NaN)

        return results

关于python - 将函数应用于多索引多列数据帧的 Pythonic 方法是什么?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32730251/

相关文章:

python - 如何使用 Sikuli 键入希腊字母

python - 我可以使用 Linux LiveCD 安装任何东西(比如 Python)吗?

Python - 提高代码 Pandas.append 的速度

python-3.x - 选择作为 MultiIndex 一部分的两个 DatetimeIndex 之间的行

python - 创建重复行并更改特定列中的值

python - 调用 pandas 分析后,Matplotlib pyplot 绘图看起来有所不同。我怎样才能解决这个问题?

Python 类型提示 : how to tell X is a subclass for Foo?

python - Pandas 按每周日期分组

python - 需要在python中逐行合并2个大的csv文件

python - 在非英语 Ubuntu 上编译 Cython .pyx 文件(unicode 错误)