编程中减少代码重复的两个工具,一是循环,一是函数。

循环,用来处理对多个同类输入做相同事情(即迭代),如对不同列做相同操作、对不同数据集做相同操作。

R语言有三种方式实现循环:

(1)for循环、while循环

(2)apply函数族

(3)泛型函数map

一. for循环、while循环

首先作两点说明:

(1)关于“for循环运行速度慢”的说法,实际上已经过时了,现在的R、Matlab等软件经过多年的内部优化已经不慢了,之所以表现出来慢,是因为你没有注意两个关键点:

  • 提前为保存循环结果分配存储空间;
  • 为循环体中涉及到的数据选择合适的数据结构。

(2)apply函数族和泛型函数map能够更加高效简洁地实现一般的for循环、while循环,但这不代表for循环、while循环就没用了,它们可以在更高的层次使用(相对于在逐元素级别使用)

1. 基本for循环

例如,有如下的tibble数据:

library(tidyverse)df <- tibble(a = rnorm(10),b = rnorm(10),c = rnorm(10),d = rnorm(10)
)

用复制-粘贴法,计算每一列的中位数:

median(df$a)
median(df$b)
median(df$c)
median(df$d) 

为了避免“粘贴-复制多于两次”,改用for循环实现:

output <- vector("double", ncol(df))     #1.输出
for (i in seq_along(df)) {               #2.迭代器output[[i]] <- median(df[[i]])         #3.循环体
}
output  #输出结果略

for循环有三个组件:

(1) 输出:output <- vector("double", length(x))

在循环开始之前,最好为输出分配足够的存储空间,这样效率更高:若每循环一次,就用c()合并一次,效率会很低下。

通常是用vector()函数创建一个给定长度的空向量,它有两个参数:向量类型(logical, integer, double, character等)、向量长度。

(2)迭代器:i in seq_along(df)

确定怎么循环:每次for循环将对i赋一个seq_along(df)中的值,可将i理解为代词it. 其中seq_along()是“1:length(df)”的安全版本,它能保证遇到长度为0的向量时,仍能正确工作:

y <- vector("double", 0)
seq_along(y) 

1:length(y)

你可能不会故意创建长度为0的向量,但容易不小心创建,则会导致报错。

(3) 循环体:output[[i]] <- median(df[[i]])

即执行具体操作的代码,它将重复执行,每次对不同的i值。

第1次迭代将执行:output[[1]] <- median(df[[1]]),

第2次迭代将执行:output[[2]] <- median(df[[2]]),

……

2. for循环变种

基本的for循环有4个变种:

(1) 修改已存在的对象,创建的新对象

有时需要用for循环修改一个已存在的对象,例如,对数据框 df 的每一列做归一化:

rescale01 <- function(x) {rng <- range(x, na.rm = TRUE)(x - rng[1]) / (rng[2] - rng[1])
}
df$a <- rescale01(df$a)
df$b <- rescale01(df$b)
df$c <- rescale01(df$c)
df$d <- rescale01(df$d)

用for循环来做,先考虑其3个组件:

输出:已经存在,与输入相同。

迭代器:可以将数据框看作是多个列构成的列表,故可以用seq_along(df)来迭代每一列。

循环体:应用函数rescale01().

于是写出如下的for循环:

for (i in seq_along(df)) {df[[i]] <- rescale01(df[[i]])
}

通常来说,你可以用这种循环来修改一个列表或数据框,注意这里是用[[ ]], 而不是[ ]. 因为原子向量最好用[[ ]], 这样能清楚地表明你处理的是一个单独的元(而不是子集)。

(2) 循环模式

· 根据数值索引:for(i in seq_along(xs), 用x[[i]] 提取值。

· 根据元素值:for(x in xs). 若你只关心附带作用,这特别有用。例如绘图、保存文件等,因为很难高效率地保存这种结果。

· 根据名字:for(nm in names(xs)). 对每个名字,访问其对应的值 x[[nm]], 若你需要使用图形标题或文件的名称,这就很有用。当创建命名的输出时,确保按如下方式命名结果向量:

results <- vector("list", length(x))
names(results) <- names(x)

注:用数值索引迭代是最常用的形式,因为只要给定位置,名字和元素值都可以提取:

for (i in seq_along(x)) {name <- names(x)[[i]]value <- x[[i]]
}

(3) 结果长度未知

有时候,你可能不知道输出结果有多长。例如,你想要模拟一些长度随机的随机向量。你可能优先想到通过逐步增加长度的方法解决该问题:

means <- c(0, 1, 2)
output <- double()
for (i in seq_along(means)) {n <- sample(100, 1)output <- c(output, rnorm(n, means[[i]]))
}
str(output)

但这种做法很低效,因为每次迭代,R都要复制上一次迭代的全部数据,其复杂度为

.

一个好的方法是,先将结果保存为列表,等循环结束再将列表重组为一个单独的向量:

out <- vector("list", length(means))
for (i in seq_along(means)) {n <- sample(100, 1)out[[i]] <- rnorm(n, means[[i]])
}
str(out)

str(unlist(out))

这里是用unlist()函数将一个向量的列表摊平为一个单独的向量。更严格的方法是用purrr包中的flatten_dbl(), 若输入不是double型的列表,将会报错。

还有两种结果长度未知的情形:

· 生成一个长字符串。不是用paste()函数将上一次的迭代结果拼接到一起,而是将结果保存为字符向量,再用函数paste(output, collapse= " ")合并为一个单独的字符串;

· 生成一个大的数据框。不是依次用rbind()函数合并每次迭代的结果,而是将结果保存为列表,再用dplyr包中的bind_rows(output)函数合并成一个单独的数据框。

所以,遇到上述模式时,要先转化为更复杂的结果对象,最后再做一步合并。

(4) 迭代次数未知(while循环)

有时候你甚至不知道输入序列有多长,这通常出现在做模拟的时候。例如,你可能想要在一行中循环直到连续出现3个“Head”,此时不适合用for循环,而是适合用while循环。

while循环更简单些,因为它只包含两个组件:条件、循环体:

while (condition) {#body
}

While循环是比for循环更一般的循环,因为for循环总可以改写为while循环,但while循环不一定能改写为for循环:

for (i in seq_along(x)) {#循环体
}
#等价于
i <- 1
while (i <= length(x)) {#循环体i <- i + 1
}

下面用while循环实现:抛一枚硬币直到连续出现3次“正面”,需要的次数:

flip <- function() sample(c("Tail", "Head"), 1)flips <- 0
nheads <- 0while (nheads < 3) {if (flip() == "Head") {nheads <- nheads + 1} else {nheads <- 0}flips <- flips + 1
}
flips

while循环并不常用,但在模拟时常用,特别是在预先不知道迭代次数的情形。

二. apply函数族

apply函数族可以代替大部分的for循环、while循环,其大意是“应用(apply)”某函数(fun)到一系列的对象上。根据应用到的对象的不同,是一族apply函数。

常用的有:

  • 分组计算:apply()和tapply()
  • 循环迭代:lapply()和sapply()
  • 多变量计算:mapply()
  1. 函数apply()

apply()函数是最常用的代替for循环的函数。apply函数可以对矩阵、数据框、数组(二维、多维),按行或列进行循环计算,对子元素进行迭代,并把子元素以参数传递的形式给自定义的FUN函数中,并以返回计算结果。基本格式为:

apply(x, MARGIN=..., fun, ...)

其中,x为数据对象(矩阵、多维数组、数据框);

MARGIN=1表示按行,2表示按列;

fun表示要作用的函数。

x<-matrix(1:6, ncol=3)
x

apply(x,1,mean) #按行求均值

apply(x,2,mean) #按列求均值

2. 函数tapply()

按一组因子INDEX对数据列 x 分组,再分别对每组作用上函数fun。基本格式为:

tapply(x, INDEX, fun, ..., simplify=TRUE)

其中,x通常为向量;

INDEX为与x长度相同的因子列表(若不是因子,R会强制转化为因子);

simplify=TRUE且fun计算结果为标量值,则返回值为数组,若为FALSE,则返回值为list对象

dat <- data.frame(height=c(174,165,180,171,160), sex=c("F","F","M","M","F"))
tapply(dat$height,dat$sex, mean)   #计算分组均值: 不同sex对应的height的均值

3. 函数lapply()

该函数是一个最基础循环操作函数,用来对vector、list、data.frame逐元、逐成分、逐列分别应用函数fun,并返回和 x 长度同样的 list 作为结果。

基本格式为:

lapply(x, fun, ...)

其中,x为数据对象(列表、数据框、向量)。

x<-list(a=1:5, b=exp(0:3))
x

lapply(x, mean)

4. 函数sapply()

sapply() 是 lapply() 的简化版本,多了一个参数simplify,若simplify=FALSE,则同lapply(),若为TRUE,则将输出的list简化为向量或矩阵。基本格式为:

sapply(x, fun, ..., simplify=TRUE, USE.NAMES=...)

5. 函数mapply()

是函数sapply()的多变量版本,将对多个变量的每个参数作用某函数。基本格式为:

mapply(fun, ..., MoreArgs=NULL, SIMPLIFY=TRUE, USE.NAMES=TRUE)

其中,

MoreArgs为fun函数的其它参数列表;

SIMPLIFY为逻辑值或字符串,取值为TRUR时,将结果转化为一个向量、矩阵或高维阵列(但不是所有结果都可转化);

... 可以接收多个数据,mapply将fun应用于这些数据的第一个元素组成的数组,然后是第二个元素组成的数组,以此类推。

返回值是vector或matrix,取决于fun返回值是一个还是多个。

#重复生成列表list(x=1:2), 重复次数times=1:3,结果为列表
mapply(rep, times=1:3, MoreArgs = list(x=1:2))

mapply(function(x,y) x^y, c(1:3), c(1:3))

mapply(function(x,y) c(x+y, x^y), c(1:3), c(1:3))  

Alco <- data.frame(AlcoholDrunk=c("YES","YES","NO","YES","YES","YES",NA,"YES","YES","YES","YES","YES","YES","NO","NO","NO","NO","YES"),
AmountDrunk=c(3.0, 1.0, NA ,3.0,  NA, 0.0,  NA, 0.0, NA, 1.7,  NA,  NA, 0.0,  NA,  NA,  NA,  NA, 2.0))

其中,变量AlcoholDrunk有三种取值,“YES”表示有饮酒史;“NO”表示无饮酒史;NA表示数据不可获取。

定义alcohol()函数实现功能:若AlcoholDrunk是NA,直接返回NA;若是NO,返回NO;否则返回变量AmountDrunk的值。因为需要传递两个变量的值,就需要用mapply()函数:

alcohol <- function(texVal, numVal){if(is.na(texVal)) {return("NA")}else if(texVal=="NO") {return("NO")}else if(is.na(numVal)) {return("amount Unknown")}else {return(numVal)}
}
mapply(alcohol, Alco$AlcoholDrunk, Alco$AmountDrunk)

三. 泛型函数map

泛型函数,相当于数学中的“泛函”,即函数的函数。

“传递一个函数给另一个函数”是非常强大的思想,这也是R作为泛函型编程语言的表现之一。

注:apply函数族也属于泛型函数。

purrr包,提供的函数足以代替许多通常的for循环。虽然apply函数族也能解决类似的问题,但purrr包更具有一致性,从而也更容易学习。另外,purrr包还支持一些快捷用法,且所有函数都是用C语言写的,速度更快。

用purrr包的解决问题的逻辑是:

(1)针对列表每个单独的元,你怎么解决某问题?一旦你解决了该问题,purrr包就可以将你的求解推广到列表中的每一个元。

(2)若你正在解决一个复杂问题,你怎么把它分解成若干小问题,使得你能够逐步完成求解?用purrr包,你就可以将这些小问题的求解步骤用管道组合到一起。

  1. map函数族

“遍历一个向量,对每个元做相同的操作,并保存结果”,这种循环模式是如此常见,所以purrr包提供了一族map函数来做这件事。一个函数针对一种类型的输出:

  • map()—映射列表,基本等同于lapply()
  • map_lgl()—映射逻辑向量
  • map_int()—映射整数型向量
  • map_dbl()—映射浮点数向量
  • map_chr()—映射字符型向量

每个函数都接受一个输入向量,应用一个函数到每一个元,再返回与输入向量同名同长度的新向量;向量的类型由map函数的后缀所确定。

注:map_*()函数必须接受原子向量,可以是行、列向量。

例如,对前文的 数据框 df,

map_dbl(df, mean)        

map_dbl(df, median)        

map_dbl(df, sd)

与用for循环相比,map函数是聚焦在所执行的操作(mean(), median(), sd()),而不是循环遍历每个元并存储结果。若改用管道操作更明显:

df %>% map_dbl(mean)
df %>% map_dbl(median)
df %>% map_dbl(sd)

2. 多变量迭代的map函数

前面map函数族实现的都是对一个向量(单变量的数据)进行迭代操作。实际中,经常会用到针对多个变量进行并行迭代,这就需要用map2()或pmap()函数。

(1)两个变量的迭代:map2()

例如,根据

不同的参数组合,生成正态分布随机数:
mu = c(5,10,-3)
sigma = c(1,5,10)
map2(mu, sigma, rnorm, n = 5)

前文的mapply例子,也可以用map2来实现:

unlist(map2(Alco$AlcoholDrunk, Alco$AmountDrunk,alcohol))

(2)更多个变量迭代:pmap()

前面都是随机生成5个数,让个数也变起来:

n <- c(1,3,5)
args <- list(mean = mu, sd = sigma, n = n)
pmap(args, rnorm)

主要参考文献:

  1. R for Data Science. Hadley Wickham, Garrett Grolemund,O'Reilly, 2017.
  2. 张良均,谢佳标,杨坦,肖刚. R语言与数据挖掘. 机械工业出版社,2016.

mapply函数​www.jianshu.com

原创作品,转载请注明。

pycharm中配置r语言_【R语言】R语言中的循环相关推荐

  1. r语言中矩阵QR分解_从零开始学R语言Day4|向量、矩阵和数组

    从零开始学R语言Day4|向量.矩阵和数组 1.1向量 1.1.1向量 在Day2中我们提及过用和c()函数来构建向量,具体实例如下. 我们还可以采用vector("类型",长度) ...

  2. rstudio r语言_如何在R中接受用户输入?

    rstudio r语言 Taking a user input is very simple in R using readline() function. In this tutorial, we ...

  3. python 运行r语言_如何在R中运行Python

    python 运行r语言 尽管我很喜欢R,但很显然Python还是一种很棒的语言-既适用于数据科学又适用于通用计算. R用户想要在Python中做一些事情可能有充分的理由. 也许这是一个很棒的库,还没 ...

  4. python预处理c语言_详解C语言编程中预处理器的用法

    预处理最大的标志便是大写,虽然这不是标准,但请你在使用的时候大写,为了自己,也为了后人. 预处理器在一般看来,用得最多的还是宏,这里总结一下预处理器的用法. #include #define MACR ...

  5. visio中公式太小_串联管道/并联管道中调节阀可调比R的计算

    调节阀的可调节比R就是调节阀所能控制的最大流量与最小流量之比,也称之为可调节范围.下面我们继续探讨调节阀可调节比R在串联管路和并联管路中的应用. 串联管道中调节阀的可调比计算 如上图所示的串联管道中, ...

  6. 取两个数较小值c语言_如何提升C语言安全性,达到RUST的安全性

    可信C语言:让C语言达到和RUST一样的安全性 1. 所有权 所有运行的程序都必须管理其使用计算机内存的方式.一些语言中具有GC(Garbage Collection)机制,在程序运行时不断地寻找不再 ...

  7. 互联网最新开发语言_互联网是多语言的,但您需要学习普通话

    互联网最新开发语言 The internet is becoming the town square for the global village of tomorrow. - Bill Gates. ...

  8. 浅谈python语言_浅谈Python语言基础知识点!

    一.Python 语言的简述 Python语言是一种解释型.面向对象的编程语言,是一种开源语言. Python属于动态类定义语言,也是一种强调类型语言. 二.Python 语言的特点 1.简单.高级 ...

  9. java中二进制怎么说_面试:说说Java中的 volatile 关键词?

    volatile 这个关键字可能很多朋友都听说过,或许也都用过.在 Java 5 之前,它是一个备受争议的关键字,因为在程序中使用它往往会导致出人意料的结果.在 Java 5之后,volatile 关 ...

  10. python中continue用法案例_记录今天学习python中for与while循环针对break和continue的用法...

    python中有两个主要的循环for与while,其中针对这两个循环有两种不同的中断用法break与continue. 首先先看下面的循环代码: 1: for i in range(10):#变量i带 ...

最新文章

  1. html 右边框变短,HTML / CSS:使边框右侧高度动态化
  2. c语言resource files的作用,c – resource.h中的宏用于什么?
  3. IO知识点整理(序列化,管道流,数据流,字节数组流,与编码)
  4. docker网络原理
  5. list集合去重复元素
  6. 如何让CentOS8虚拟机与主机相互Ping通
  7. WP7的Push Notifications
  8. CMMI认证办理需要什么条件
  9. java 16进制_JAVA 十六进制与字符串的转换
  10. SSD硬盘无法格式化怎么办
  11. QCLOUD APIGATEWAY HTTP header字段整理
  12. Vue 的最大的优势是什么?
  13. C++不重起Windows直接更改IP地址
  14. 【菜菜的sklearn课堂笔记】逻辑回归与评分卡-用逻辑回归制作评分卡-分箱
  15. 配置测试,你了解吗?
  16. Gradle 学习 ----Gradle 进阶说明
  17. studiolibrary安装_初学者daz studio中文基础安装布局教程
  18. D - Squirrel and chestnut(二分)
  19. python-微信自动发送信息2
  20. H5在微信里只能调用相机,不能调用相册

热门文章

  1. Java @SafeVarargs注解
  2. SpringBoot获取配置文件常量值
  3. 车辆工程用得到python吗_如今车辆工程真的不如以前了吗?
  4. 开红数显示服务器为空,网维大师常见问题:图标空白或红号问号
  5. 山东大学 2020级数据库系统 实验三
  6. c语言音像店程序,音像店管理程序_C 课程设计.pdf
  7. 【youcans 的 OpenCV 例程 200 篇】107. 退化图像的维纳滤波
  8. 详细分析如何利用python批量爬取百度图片
  9. 用HTML编写教学评估系统,在线教学质量评价系统的设计与实现
  10. oracle 表访问,向oracle导入访问表