大熊猫:DataFrame行上的复杂过滤器


85

我想通过每行的功能来过滤行,例如

def f(row):
  return sin(row['velocity'])/np.prod(['masses']) > 5

df = pandas.DataFrame(...)
filtered = df[apply_to_all_rows(df, f)]

或者再举一个更复杂,人为的例子,

def g(row):
  if row['col1'].method1() == 1:
    val = row['col1'].method2() / row['col1'].method3(row['col3'], row['col4'])
  else:
    val = row['col2'].method5(row['col6'])
  return np.sin(val)

df = pandas.DataFrame(...)
filtered = df[apply_to_all_rows(df, g)]

我该怎么办?

Answers:


121

您可以使用来执行此操作DataFrame.apply,该功能会沿给定轴应用功能,

In [3]: df = pandas.DataFrame(np.random.randn(5, 3), columns=['a', 'b', 'c'])

In [4]: df
Out[4]: 
          a         b         c
0 -0.001968 -1.877945 -1.515674
1 -0.540628  0.793913 -0.983315
2 -1.313574  1.946410  0.826350
3  0.015763 -0.267860 -2.228350
4  0.563111  1.195459  0.343168

In [6]: df[df.apply(lambda x: x['b'] > x['c'], axis=1)]
Out[6]: 
          a         b         c
1 -0.540628  0.793913 -0.983315
2 -1.313574  1.946410  0.826350
3  0.015763 -0.267860 -2.228350
4  0.563111  1.195459  0.343168

15
apply在这种情况下没有必要。常规布尔索引可以正常工作。df[df['b] > df['c']]。实际需要的情况apply很少,甚至很少需要axis=1
Ted Petrou

@TedPetrou如果不确定数据框中的每个元素的类型正确,该怎么办。常规布尔索引是否支持异常处理?
D. Ror。

13

假设我有一个DataFrame如下:

In [39]: df
Out[39]: 
      mass1     mass2  velocity
0  1.461711 -0.404452  0.722502
1 -2.169377  1.131037  0.232047
2  0.009450 -0.868753  0.598470
3  0.602463  0.299249  0.474564
4 -0.675339 -0.816702  0.799289

我可以使用sin和DataFrame.prod创建一个布尔掩码:

In [40]: mask = (np.sin(df.velocity) / df.ix[:, 0:2].prod(axis=1)) > 0

In [41]: mask
Out[41]: 
0    False
1    False
2    False
3     True
4     True

然后使用掩码从DataFrame中选择:

In [42]: df[mask]
Out[42]: 
      mass1     mass2  velocity
3  0.602463  0.299249  0.474564
4 -0.675339 -0.816702  0.799289

2
实际上,这可能是一个不好的例子:np.sin自动广播到所有元素。如果我用一个不太智能的功能代替它,一次只能处理一个输入怎么办?
Duckworth


5

我无法评论duckworthd的答案,但是它并不完美。当数据框为空时崩溃:

df = pandas.DataFrame(columns=['a', 'b', 'c'])
df[df.apply(lambda x: x['b'] > x['c'], axis=1)]

输出:

ValueError: Must pass DataFrame with boolean values only

在我看来,这似乎是熊猫中的错误,因为{}绝对是一组有效的布尔值。有关解决方案,请参阅Roy Hyunjin Han的答案


3

我发现的最佳方法是,而不是reduce=True为了避免出现空df的错误(因为无论如何此arg都已被弃用),只需在应用过滤器之前检查df size> 0:

def my_filter(row):
    if row.columnA == something:
        return True

    return False

if len(df.index) > 0:
    df[df.apply(my_filter, axis=1)]

0

您可以将该loc属性用于切片数据帧。

根据 文档loc可以有一个callable functionas参数。

In [3]: df = pandas.DataFrame(np.random.randn(5, 3), columns=['a', 'b', 'c'])

In [4]: df
Out[4]: 
          a         b         c
0 -0.001968 -1.877945 -1.515674
1 -0.540628  0.793913 -0.983315
2 -1.313574  1.946410  0.826350
3  0.015763 -0.267860 -2.228350
4  0.563111  1.195459  0.343168

# define lambda function
In [5]: myfilter = lambda x: x['b'] > x['c']

# use my lambda in loc
In [6]: df1 = df.loc[fif]

如果要将过滤器功能fif与其他过滤器标准结合使用

df1 = df.loc[fif].loc[(df.b >= 0.5)]
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.