我希望对包含4列的大约400k行的数据帧进行排序,其中大约一半用if语句取出:
for a in range (0, howmanytimestorunthrough):
if ('Primary' not in DataFrameexample[a]):
#take out row
到目前为止,我一直在测试以下4种中的任何一种:
newdf.append(emptyline,)
nefdf.at[b,'column1'] = DataFrameexample.at[a,'column1']
nefdf.at[b,'column2'] = DataFrameexample.at[a,'column2']
nefdf.at[b,'column3'] = DataFrameexample.at[a,'column3']
nefdf.at[b,'column4'] = DataFrameexample.at[a,'column4']
b = b + 1
或与.loc相同
newdf.append(emptyline,)
nefdf.loc[b,:] = DataFrameexample.loc[a,:]
b = b + 1
或将if(not in)更改为if(in)并使用:
DataFrameexample = DataFrameexample.drop([k])
或尝试将空行设置为具有值,然后追加它:
notemptyline = pd.Series(DataFrameexample.loc[a,:].values, index = ['column1', 'column2', ...)
newdf.append(notemptyline, ignore_index=True)
所以从我到目前为止测试的结果来看,它们似乎都在少量行(2000)上工作正常,但是一旦我开始获得更多行,它们会花费指数更长的时间。 .at似乎比.loc快得多,即使我需要它运行4次,但仍然变慢(行的10倍,需要超过10次)。 .drop我想每次尝试复制数据帧,所以真的不起作用?我似乎无法让.append(notemptyline)正常工作,它只是一遍又一遍地替换索引0。
我知道必须有一种有效的方法,我似乎无法完全实现这一目标。有什么帮助吗?
答案 0 :(得分:2)
你的速度问题与.loc vs .at vs ...无关(对于.loc和.at之间的比较,看看这个question)但是来自显式循环每一行你的数据帧。熊猫的全部内容都是为了操作。
您希望根据比较过滤数据框。您可以将其转换为布尔索引器。
indexer = df!='Primary'
这将为您提供一个带有布尔值的4 x行数据帧。现在,您希望将维度减少到1 x n行,以便在行(轴1)中的所有值为真时值为true。
indexer = indexer.all(axis=1)
现在我们可以使用.loc来只获取索引器为True
df = df.loc[indexer]
这将比更多更快,然后迭代行。
编辑:
要检查df条目是否包含字符串,您可以替换第一行:
indexer = df.apply(lambda x: x.str.contains('Primary'))
请注意,您通常不想使用apply语句(在内部使用for循环来自定义函数)来迭代很多元素。在这种情况下,我们循环遍历列,如果你只有几个那么就可以了。