以下内容主要参照 Introducing dplyr 和 dplyr 包自带的简介 (Introduction to dplyr), 复制了原文对应代码, 并夹杂了个人理解和观点 (多附于括号内).
0 初始化0.1 安装
install.packages("dplyr")
0.2 示范数据
- library(Lahman): Lahman 包里的棒球比赛数据集 Batting
- library(hflights): hflights 包里的飞机航班数据
0.3 数据集类型
将过长过大的数据集转换为显示更友好的 tbl_df 类型:
hflights_df <- tbl_df(hflights)
可以 hflights_df 感受一下不再被刷屏的感觉.
1 基本操作
把常用的数据操作行为归纳为以下五种:
1.1 筛选: filter()
按给定的逻辑判断筛选出符合要求的子数据集, 类似于 base::subset() 函数
例如:
filter(hflights_df, Month == 1, DayofMonth == 1)
用R自带函数实现:
hflights[hflightsMonth == 1 & hflightsDayofMonth == 1, ]
除了代码简洁外, 还支持对同一对象的任意个条件组合, 如:
filter(hflights_df, Month == 1 | Month == 2)
注意: 表示 AND 时要使用 & 而避免 &&
1.2 排列: arrange()
按给定的列名依次对行进行排序.
例如:
arrange(hflights_df, DayofMonth, Month, Year)
对列名加 desc() 进行倒序:
arrange(hflights_df, desc(ArrDelay))
这个函数和 plyr::arrange() 是一样的, 类似于 order()
用R自带函数实现:
hflights[order(hflightsDayofMonth,hflightsMonth, hflightsYear),]hflights[order(desc(hflightsArrDelay)), ]
1.3 选择: select()
用列名作参数来选择子数据集:
select(hflights_df, Year, Month, DayOfWeek)
还可以用 : 来连接列名, 没错, 就是把列名当作数字一样使用:
select(hflights_df, Year:DayOfWeek)
用 - 来排除列名:
select(hflights_df, -(Year:DayOfWeek))
同样类似于R自带的 subset() 函数 (但不用再写一长串的 c("colname1", "colname2") 或者 which(colname(data) == "colname3"), 甚至还要去查找列号)
1.4 变形: mutate()
对已有列进行数据运算并添加为新列:
mutate(hflights_df, gain = ArrDelay - DepDelay, speed = Distance / AirTime * 60)
作用与 plyr::mutate() 相同, 与 base::transform() 相似, 优势在于可以在同一语句中对刚增加的列进行操作:
mutate(hflights_df, gain = ArrDelay - DepDelay, gain_per_hour = gain / (AirTime / 60))
而同样操作用R自带函数 transform() 的话就会报错:
transform(hflights, gain = ArrDelay - DepDelay, gain_per_hour = gain / (AirTime / 60))1.5 汇总: summarise()
对数据框调用其它函数进行汇总操作, 返回一维的结果:
summarise(hflights_df, delay = mean(DepDelay, na.rm = TRUE))
等同于 plyr::summarise(), 原文说该函数功能尚不是非常有用, 大概以后的更新会加强吧.
2 分组动作 group_by()
以上5个动词函数已经很方便了, 但是当它们跟分组操作这个概念结合起来时, 那才叫真正的强大! 当对数据集通过 group_by() 添加了分组信息后,mutate(), arrange() 和 summarise() 函数会自动对这些 tbl 类数据执行分组操作 (R语言泛型函数的优势).
例 如: 对飞机航班数据按飞机编号 (TailNum) 进行分组, 计算该飞机航班的次数 (count = n()), 平均飞行距离 (dist = mean(Distance, na.rm = TRUE)) 和 延时 (delay = mean(ArrDelay, na.rm = TRUE))
planes <- group_by(hflights_df, TailNum)delay <- summarise(planes, count = n(), dist = mean(Distance, na.rm = TRUE), delay = mean(ArrDelay, na.rm = TRUE))delay <- filter(delay, count > 20, dist < 2000)
用 ggplot2 包作个图观察一下, 发现飞机延时不延时跟飞行距离没太大相关性:
ggplot(delay, aes(dist, delay)) + geom_point(aes(size = count), alpha = 1/2) + geom_smooth() + scale_size_area()
(图就不上了, 右键复制来的链接太凶残了, 看着像是现算的)
更多例子见 vignette("introduction", package = "dplyr")
另: 一些汇总时的小函数
- n(): 计算个数
- n_distinct(): 计算 x 中唯一值的个数. (原文为 count_distinct(x), 测试无用)
- first(x), last(x) 和 nth(x, n): 返回对应秩的值, 类似于自带函数 x[1], x[length(x)], 和 x[n]
注意: 分组计算得到的统计量要清楚样本已经发生了变化, 此时的中位数是不可靠的
3 连接符 %>% 注意连接符为> 不是.
包里还新引进了一个操作符, 使用时把数据名作为开头, 然后依次对此数据进行多步操作.
比如:
Batting %.% group_by(playerID) %.% summarise(total = sum(G)) %.% arrange(desc(total)) %.% head(5)
这样可以按进行数据处理时的思路写代码, 一步步深入, 既易写又易读, 接近于从左到右的自然语言顺序, 对比一下用R自带函数实现的:
head(arrange(summarise(group_by(Batting, playerID), total = sum(G)) , desc(total)), 5)
或者像这篇文章所用的方法:
totals <- aggregate(. ~ playerID, data=Batting[,c("playerID","R")], sum)ranks <- sort.list(-totals$R)totals[ranks[1:5],]
文章里还表示: 用他的 MacBook Air 跑 %.% 那段代码用了 0.036 秒, 跑上面这段代码则用了 0.266 秒, 运算速度提升了近7倍. (当然这只是一例, 还有其它更大的数字.)
更多请 ?"%.%", 至于这个新鲜的概念会不会和 ggplot2 里的 + 连接号一样, 发挥出种种奇妙的功能呢? 还是在实际使用中多体验感受吧.
感想
可以看到, 用 dplyr 所含函数实现的代码都要简洁易读得多, 说到底, R语言只是一个工具, 作为工具, 就是要拿来用的, 越称手越便利越简洁越好, 可是, 正如 Hadley Wickham 在2013年的访谈中提到的那样:
如果你用了8小时进行数据清理和数据整理,而只用了2小时进行建模,那么很明显,你希望了解如何将数据清理和整理的时间尽可能缩短。
反思之下, 本人也是将大把的时间花在了对数据的反复调整上, 或许是手生, 当然R语言在这方面也确实有一定不足, 大神又说了:
数据分析有两个瓶颈,一是我们的目标是什么,二是我们如何用计算机去实现。我现有的很多作品,如 ggplot2,plyr 和 reshape2,更关注的是如何更简单地表达你的目标,而不是如何让计算机算得更快。
这种内在的理念正是要将工具工具化, 把无谓的时间减少, 让精力用在真正需要考虑的地方. 正如 Vim 一样, 在投入一定的学习成本后, 继续用继续学, 不知不觉地就能心手如一, 想做什么, 就已经按下去了, 从而更多地思考要编辑什么, 而不必纠结于光标移动选择等细节. 这其中的巧妙之处在于: 实现过程要以人脑的思维运作方式为标准, 让工具来适应人, 以实现目的为导向, ggplot2 的图形图层语法也是如此. 不管是软件也好, 编程语言也好, 高效的方法都是相通的, 这也正是许多人努力的方向, 另外平素语出惊人的王垠最近也表达了类似观点.
顺便肖凯老师在网易云课堂新开的R语言初级教程里提到了十大必学R包的说法, 并把 plyr 列为之一, 有趣的是居然还有人在问答平台上求详情, 好奇之下放狗一搜, 原来出处在此 (脱水版), 其中 ggplot2 和 reshape2 是平时都有在用的, 还有实用的 knitr 和 Slidify , 其它就没什么发言权了.
深入学习
暂时没有太多的相关资料, 如欲进一步学习, 可参阅:
- dplyr 包自带的60页详细文档
- 其余几个vignettes (网页) 或 vignette(package = "dplyr") , 包含了数据库相关, 混合编程, 运算性能比较, 以及新的 window-functions 等内容.
简单看了下vignette("window-functions", package = "dplyr"), 提供了一系列函数, 扩展了原来只能返回一个数值的聚焦类函数(如sum(), mean())至返回等长度的值, 变成 cumsum()和 cummean(), 以及 n(), lead() 和 lag()等便捷功能. - plyr 包的相关文档: 主页
- 还有 data.table 包也是很强大的哦, 空下来可以学一学
常见的数据处理包
dplyr——package
1.数据对象:tbl对象
使用dplyr包预处理时建议使用tbl_df()或tbl_cube()或tbl_sql()函数将原数据转换为tbl对象
2.观测筛选
将指定条件的观测筛选出来:filter()函数
filter(.data,…)
.data为tbl对象
…为观测筛选条件,类似于subset()函数,但不同的是filter()函数不能筛选某些关心的变量变量
library(dplyr)
df <- data.frame(x = c("a","b","c","a","b","e","d","f"),y = c(1,2,3,4,5,6,7,8))
dftbl <- tbl_df(df)
filter(dftbl,x %in% c("a","b"))
- 1
- 2
- 3
- 4
- 1
- 2
- 3
- 4
3.变量选取
select()函数可以筛选指定的变量,而且选择变量时也可以重新命名变量。如果要剔除某些变量,只需要在变量前加上负号”-“。
select()函数,传递的参数:
starts_with(x,ignor.case = TRUE) # 选择以字符x开头的变量
ends_with(x,ignore.case = TRUE) # 选择以字符x结尾的变量
contains(x,ignore.case = TRUE) #选择所有包含x的变量
matches(x,ignore.case = TRUE) #选择匹配正则表达式的变量
num_range(“x”,1:5,width = 2) #选择从x01到x05的数值型变量
one_of(“x”,”y”,”z”) #选择包含在声明变量中的变量
everything() #选择所有变量,一般调整数据集中变量顺序时使用
示例:
# 将dftbl数据集中的y变量放到x变量之前
select(dftbl,everything())
#筛选变量的同时,重新命名变量
select(dftbl,x1 = 1,y1 = y)
- 1
- 2
- 3
- 4
- 5
- 1
- 2
- 3
- 4
- 5
4.重命名变量
rename(tbl,newname = oldname,…)
rename(dftbl,x1 = x,y1 = y)
- 1
- 1
5.数据排序
数据预处理的过程中往往也需要按某些变量进行排序:arrange()函数实现语法:
arrange(.data,…)
arrange()函数默认以某个变量进行升序,如需降序则desc(var_name)
arrange(dftbl,y) # y变量升序
arrange(dftbl,desc(y)) #降序操作
- 1
- 2
- 1
- 2
6.数据扩展
通过mutae()函数可以在原始数据集的基础上扩展新变量,并保留原始变量
mutate(.data,…)
mutate(dftbl,z = y^2+y-10)
- 1
- 1
同样可以进行数据扩展的函数transmute(),与mutate()函数不同的是,该函数扩展新变量的同时将删除所有原始变量,结果中只有扩展变量。
transmute(dftbl,z =y^2)
- 1
- 1
7.数据聚合
在数据库操作中,往往需要进行聚合函数的应用,这里可以使用summarize()函数实现数据集聚合操作.个人理解的聚合函数就是对数据集中的变量进行统计计算。
语法如下:
summarize(.data,…)
可以用来聚合的函数有:
min(),max(),mean()…等统计量,以及IQR() #返回四分位极差
n() # 返回观测个数
n_distinct() #返回不同的观测个数
first() # 返回第一个观测
last() #返回最后一个观测
nth() #返回n个观测
summarize(dftbl,max(y))
summarize(dftbl,n())
- 1
- 2
- 1
- 2
还可以用group_by()函数实现分组聚合
group_by()语法如下:
group_by(.data,add = FALSE)
summarize(group_by(dftbl,x),sum(y))
- 1
- 1
8.数据关联
数据框中经常需要将多个表进行连接操作,如左连接、右连接、内连接等,这里dplyr包也提供了数据集的连接操作。如下:
inner_join #内连接
left_join #左连接
right_join #右连接
full_join #全连
semi_join # 返回能够与y表匹配的x表所有记录
anti_join # 返回无法与y表匹配的x表的所有记录
df2 <- data.frame(x = c("a","b","c"),z = c("A","B","C"))
df2tbl <- tbl_df(df2)
inner_join(x = dftbl,y = df2tbl,by = "x")
semi_join(x = dftbl,y = df2tbl,by = "x")
anti_join(x = dftbl,y = df2tbl,by = "x")
- 1
- 2
- 3
- 4
- 5
- 1
- 2
- 3
- 4
- 5
9.数据合并
R基础包cbind()和rbind()函数实现按列的方向进行数据合并和按行的方向进行合并。
dplyr包中也添加了类似功能的函数,分别是bind_cols()函数和bind_rows()函数
mydf1 <- data.frame(x = c(1,2,3,4),y = c(10,20,30,40))
mydf2 <- data.frame(x = c(5,6),y = c(50,60))
mydf3 <- data.frame(z = c(100,200,300,400))
bind_rows(mydf1,mydf2) #行变长
bind_cols(mydf1,mydf3)
- 1
- 2
- 3
- 4
- 5
- 1
- 2
- 3
- 4
- 5
10.管道函数
即通过%>%将上一个函数的输出作为下一个函数的输入
# 根据数据集dftbl和df2tbl,取出z变量对应的最大y值
inner_join(x = dftbl,y = df2tbl,by = "x") %>% group_by(z) %>% summarise(max(y))
- 1
- 2
- 1
- 2
11.连接数据库数据
如果需要获取MySQL数据库中的数据时,可以直接使用dplyr包中的src_mysql()函数。
src_mySQL 函数语法如下:
src_mysql(dbname,host = NULL,port = 0L,user = “root”,password = “”,…)
通过以上方式连接MySQL数据库后,使用tbl()函数获取数据集,tbl()函数语法如下:
tbl(src,from =”“)
src为src_mysql()函数对象
from为SQL语句
src <- src_mysql("test",host = "localhost",user = "root",password = "snake")
src
- 1
- 2
- 1
- 2
plyr-package
可以非常方便的实现数据结构之间的转换
其中的函数名有一定的规律,跟输入输出的数据结构相关。
1.函数介绍
a*ply函数形式
aaply(.data = ,.margins = ,.fun = ,...,.progress = "none",.inform = FALSE)
adply(.data = ,.margins = ,.fun = ,...,.progress = "none",.inform = FALSE)
alply(.data = ,.margins = ,.fun = ,...,.progress = "none",.inform = FALSE)
a_ply(.data = ,.margins = ,.fun = ,.progress = "none",.inform = FALSE) #输入结构:array,无输出结果
- 1
- 2
- 3
- 4
- 5
- 1
- 2
- 3
- 4
- 5
.data可以是数组也可以是矩阵;
.margins指定要分析的数组或矩阵的维度,即行维(margins = 1),列维(margins = 2)
.fun为行或列维指定需要处理的函数,可以是R自带的函数,如sum(),mean()等,也可以是自定义函数;
…为指定函数的其他参数;.progress指定以什么样的方式展示程序运行的进度,默认为不显示进度,还可以选择text(文本进度条)、tk(tk进度条)和win(windows系统自带的进度条)
.inform是否指定报错信息,默认不指定,因为设为TRUE,将会降低程序的执行效率,但该参数对bug的处理是有帮助的
示例;
library(plyr)
a <- array(data = 1:500000,dim = c(100000,5))
# 对每一行求均值,不显示进度条
test1 <- aaply(.data = a,.margins = 1,.fun = mean,.progress = "none")
head(test1)
# 对每一行求标准差,以文本的形式显示进度条
test2 <- adply(.data = a,.margins = 1,.fun = sd,.progress = "text")
head(test2)
# 对每一列求和,以tk形式显示进度条
a2 <- array(rnorm(100000),dim = c(100,1000))
test3 <- alply(.data = a2,.margins = 2,.fun = sum,.progress = "tk")
head(test3)
# 对每一列求最大值,以windows自带进度条显示进度
a3 <- array(rnorm(100000),dim = c(100,1000))
test4 <- a_ply(.data = a3,.margins = 2,.fun = max,.progress = "win")
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
# d*ply函数格式
daply(.data = ,.variables = ,.fun = ,…,.progress = “none”,.inform = FALSE)
ddply(.data = ,.variables = ,.fun = ,…,.progress = “none”,.inform = FALSE)
dlply(.data = ,.variables = ,.fun = ,…,.progress = “none”,.inform = FALSE)
d_ply(.data = ,.variables = ,.fun = ,…,.progress = “none”,.inform = FALSE)
.data 指定为数据框结构;
.variables指定数据框中的分组变量,需要用点号.引起来;
.fun 基于分组变量,可对数据框中的其余变量指定某种函数,可以是R自带的函数,如sum(),mean()等,也可以自定义函数,类似于聚合分析;
.progress和.inform与a*plyr函数参数一致。
示例:
# 构建自定义函数
fun <- function(data) apply(data,2,mean)
daply(.data = iris[,1:4],.variables = .(iris$Species),.fun = fun)
ddply(.data = iris[,1:4],.variables = .(iris$Species),.fun = fun)
dlply(.data = iris[,1:4],.variables = .(iris$Species),.fun = fun)
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
# l*ply函数格式
laply(.data = ,.fun = ,...,.progress = "none",.inform = FALSE)
ldply(.data = ,.fun = ,...,.progress = "none",.inform = FALSE)
llply(.data = ,.fun = ,...,.progress = "none",.inform = FALSE)
l_ply(.data = ,.fun = ,...,.progress = "none",.inform = FALSE)
- 1
- 2
- 3
- 4
- 1
- 2
- 3
- 4
.data可以指定为列表数据.
其余参数与a*ply()函数和d*ply()函数参数一致。
示例:
x1 <- 1:100
x2 <- seq(from = 100,to = 1000,by = 2)
x3 <- runif(150,min = 10,max = 100)
# 列表由向量构成
l1 <- list(x1 = x1,x2 = x2,x3 = x3)
laply(.data = l1,.fun = mean)
ldply(.data = l1,.fun = summary)
llply(.data = l1,.fun = quantile)
l_ply(.data = l1,.fun = summary)
# 构建数据框dll
y11 <- rnorm(n = 100,mean = 10,sd = 5)
y12 <- rt(n = 100,df = 3)
y13 <- rf(n = 100,df1 = 2,df2 = 3)
y14 <- factor(x = c("low","potential","high"),ordered = T)
y15 <- sample(y14,size = 100,replace = TRUE)
d11 <- data.frame(y1 = y11,y2 = y12,y3 = y13,y5 = y15)
head(dll)
# 构建数据框d21
y21 <- 1:100
y22 <- seq(from = 1,to = 2,length = 100)
y23 <- rchisq(n = 100,df = 8)
y24 <- factor(x = c("A","B","C","D"),order = T)
y25 <- sample(y24,size = 100,replace = TRUE)
d21 <- data.frame(y21 = y21,y22 = y22,y23 = y23,y25 = y25)
head(d21)
# 列表由数据框组成
l2 <- list(first = d11,second = d21)
library(psych)
fun <- function(data) describeBy(data[,1:3],group = data[,4])
llply(.data = l2,.fun = fun,.progress = "none")
llply(.data = l2,.fun = fun,.progress = "text")