【pandas】去重、填充、排序、变换



2017年10月16日    Author:Guofei

文章归类: 0x12_Pandas与numpy    文章编号: 103

版权声明:本文作者是郭飞。转载随意,标明原文链接即可。本人邮箱
原文链接:https://www.guofei.site/2017/10/16/pandascleandata3.html


去重

data.drop_duplicates(inplace=True)
data.drop_duplicates(subset='column1') # 找第一列重复者
data.duplicated(keep='last') # 'first','last',返回布尔类型
data.duplicated(keep='last')  # 返回Series,bool类型,存放是否是重复行/列
# keep='first','last'

删除空数据

dropna(how='any') # how='all'

删除整行&整列

data.drop('animal2', axis='columns', inplace=True)

参数可以是list,以删除多行/多列

替换数据

data.replace([4,5],np.nan,inplace=True)
data.replace({4:np.nan,5:999},inplace=True)

填充空数据

可以向上填充/向下填充

a=data.fillna(method='bfill',inplace=True)
# method :bfill,ffill,

也可以用值填充

a=data.fillna(data.mean(),inplace=True)

值填充时,可以每列不一样

df.fillna({'a':999,'b':888,'c':777,'d':666})

线性插值填充

df1.interpolate()

线性插值填充:把index作为间隔

df1.interpolate(method='index')

计算新列

import pandas as pd
import numpy as np
df=pd.DataFrame(np.arange(16).reshape(-1,4),index=list('abcd'),columns=list('wxyz'))
df.loc[:,'ww']=df.loc[:,'w']*2+df.loc[:,'x']

改变数据类型

df.loc[:,['ww']].astype('float') # int

排序

sort

  • sort_values按值排序
  • sort_index按index排序
df.sort_values(by=['w','z'],ascending=[False,True],inplace=True)
df.sort_index(axis=1, ascending=False)
df.sort_index(ascending=True,inplace=True)

sorted

sorted(iterable,key,reverse)可以对任何iterable的对象进行排序

a=['111','26','76','3']
sorted(a,key=int)
sorted(a,key=lambda x:int(x[0]))

b=[{'a':4},{'b':2},{'c':3}]
sorted(b,key=lambda x:x[1])

rank

返回排序的序号

import pandas as pd
import numpy as np
df=pd.DataFrame(np.random.rand(16).reshape(-1,4),columns=list('wxyz'))
df.loc[:,'w']=[0,1,1,2]
df
w x y z
0 0 0.098404 0.099138 0.381158
1 1 0.776177 0.478243 0.523116
2 1 0.397995 0.040227 0.362902
3 2 0.997362 0.072824 0.709957
df.rank(ascending=True,method='average')
w x y z
0 1.0 1.0 3.0 2.0
1 2.5 3.0 4.0 3.0
2 2.5 2.0 1.0 1.0
3 4.0 4.0 2.0 4.0

method说明:

  • ‘average’(default): 相等分组中,按平均值
  • ‘min’: 取最小排名
  • ‘max’: 取最大排名
  • ‘first’: 按照原始数据中出现的顺序分配排名

apply:最好用的转化工具

生成样例

import pandas as pd
import numpy as np
from scipy import stats
rv = stats.uniform()
df = pd.DataFrame(rv.rvs(size=(100, 5)), columns=list('abcde'))

apply 后接 func:接受1行/列

这种用法比较清晰,推荐。

# 生成新的两列
df_new = df.apply(lambda x:pd.Series({'new_a':x['a']+x['b'],'new_b':x['a']+x['b']}),axis=1)

详细解析:

  • axis=1时,每次给func一行,df_new 是一个相同高度的DataFrame。apply 后面的函数(记为func)是这样的:
    • func接受df的一行。进入到函数后,数据是Series格式(记为s),s.index是df的列名,s.name是df的index
    • func返回一个Series(记为s_new),最后组成df_new。数据情况同上,s_new.index是df_new的列名,s_new.name是df_new的index。额外的,如果不指定s_new.name,那么df_new.index 会与 df.index 保持一致。
  • axis=0 (默认) 时,是按列计算,每次给func一列,df_new 是一个相同宽度的DataFrame。apply函数对应关系与上面类似
  • 如果func的return的是数字、列表等,那么返回的 df_new 是一个 Series

如果想保留原本的列,添加新列:

df_new = df.apply(lambda x: x.append(pd.Series({
    'new_a': x.a + x.b
    , 'new_b': x.b + x.c
})), axis=1)

applymap:对每个单元格独立操作

pd_1.applymap(lambda x:x+1)

apply 后接 dict

df = pd.DataFrame([['about', 'a|b|cc'], ['cool', 'x|yy|z']], columns=['col1', 'col2'], index=['a', 'b'])

def func(x):
    return x.upper()
df_new = df.apply({'col1':func, 'col2':func})

详细解析:

  1. col1 是 df 某列的列名
  2. 每次输入一个单元格
    • func 每次接受 df.col1 的1个单元格作为输入。
    • df_new 是一个DataFrame,列名是col1(所以这个方法的缺陷是,你不能对一列做两种操作,例如你想做一列大写和一列小写,就得多写几行了,但是用 apply后接func可以一行解决)
  3. 每次输入一列。如果 func 报错,就会尝试输入整个列作为 Series
    • 如果 func 返回 Series/list等,df_new 是一个对应的DataFrame(列不变,变成多个)
    • 如果 func 返回 一个元素,df_new 是一个对应的 Series (相当于agg操作)
    • 例子:
       df = pd.DataFrame([['about', 'a|b|cc'], ['cool', 'x|yy|z']], columns=['col1', 'col2'], index=['a', 'b'])
       def func(x):
      x.shape # 如果输入一个元素,令其报错
      return x.shape[0]
       df.apply({'col1':func, 'col2':func})
      

其它

用map修改index&columns

  • rename可以完全替代这个
  • 参数不能是dict

index有map()方法,但没有apply方法,案例:

import pandas as pd
import numpy as np

df = pd.DataFrame(np.arange(16).reshape(4, -1), index=list('abcd'), columns=list('gfjk'))
df.index = df.index.map(str.upper)

agg:汇总计算

import pandas as pd
import numpy as np
from scipy import stats
rv=stats.uniform()
df=pd.DataFrame(rv.rvs(size=(100,5)),columns=list('abcde'))

def func(data):
    return data.max()

df.agg([np.mean,func]) # 此例返回2行,一行mean,一行func
# func接受每一列作为Series,返回一个数字

df.agg({'a':func}) # 对指定列做func,与上一条有个区别,func可以返回多组数字,这种情况下,func返回的多组数字是这条语句返回的多行

transform:用途有限

def func(data):
    return data+1

df.transform({'a':func,'b':func}) # func需要接受df每一列作为Series,返回同样大小的Series

transfrom:按列变换

df = pd.DataFrame([['about', 'a|b|cc'], ['cool', 'x|yy|z']], columns=['col1', 'col2'], index=['a', 'b'])

def func(x):
    return x.upper()

df.transform({'col1': func})

# 1. transform返回一个DataFrame
# 2. func每次接收一个单元格。如果不能运行,尝试接收整列作为 <Series>
# 3_1. 如果 func 返回一个 Series,那么把这个Series展开为多列
# 3_2. 如果 func 返回其它对象(数字、list等),那么把这个对象塞到新 DataFrame 的单元格内

例子:

df = pd.DataFrame([['a|b|cc'], ['x|yy|z']], columns=['col1'])

# 用字符串方法做一切操作
df.transform({'col1': lambda x: x.upper()})
df.transform({'col1': lambda x: len(x)})
df.transform({'col1': lambda x: 'a' in x})

df.transform({'col2': lambda x: x.split('|')})
df.transform({'col2': lambda x: ','.join(x.split('|'))})

# 可以直接加减乘
df.col1 + df.col2 * 2

您的支持将鼓励我继续创作!