gpt4 book ai didi

r - 当已知可能的输出时加速 `strsplit`

转载 作者:行者123 更新时间:2023-12-03 02:45:24 28 4
gpt4 key购买 nike

我有一个带有因子列的大型数据框,我需要通过用分隔符分割因子名称来将其分为三个因子列。这是我当前的方法,对于大数据帧(有时数百万行)来说非常慢:

data <- readRDS("data.rds")
data.df <- reshape2:::melt.array(data)
head(data.df)
## Time Location Class Replicate Population
##1 1 1 LIDE.1.S 1 0.03859605
##2 2 1 LIDE.1.S 1 0.03852957
##3 3 1 LIDE.1.S 1 0.03846853
##4 4 1 LIDE.1.S 1 0.03841260
##5 5 1 LIDE.1.S 1 0.03836147
##6 6 1 LIDE.1.S 1 0.03831485

Rprof("str.out")
cl <- which(names(data.df)=="Class")
Classes <- do.call(rbind, strsplit(as.character(data.df$Class), "\\."))
colnames(Classes) <- c("Species", "SizeClass", "Infected")
data.df <- cbind(data.df[,1:(cl-1)],Classes,data.df[(cl+1):(ncol(data.df))])
Rprof(NULL)

head(data.df)
## Time Location Species SizeClass Infected Replicate Population
##1 1 1 LIDE 1 S 1 0.03859605
##2 2 1 LIDE 1 S 1 0.03852957
##3 3 1 LIDE 1 S 1 0.03846853
##4 4 1 LIDE 1 S 1 0.03841260
##5 5 1 LIDE 1 S 1 0.03836147
##6 6 1 LIDE 1 S 1 0.03831485

summaryRprof("str.out")

$by.self
self.time self.pct total.time total.pct
"strsplit" 1.34 50.00 1.34 50.00
"<Anonymous>" 1.16 43.28 1.16 43.28
"do.call" 0.04 1.49 2.54 94.78
"unique.default" 0.04 1.49 0.04 1.49
"data.frame" 0.02 0.75 0.12 4.48
"is.factor" 0.02 0.75 0.02 0.75
"match" 0.02 0.75 0.02 0.75
"structure" 0.02 0.75 0.02 0.75
"unlist" 0.02 0.75 0.02 0.75

$by.total
total.time total.pct self.time self.pct
"do.call" 2.54 94.78 0.04 1.49
"strsplit" 1.34 50.00 1.34 50.00
"<Anonymous>" 1.16 43.28 1.16 43.28
"cbind" 0.14 5.22 0.00 0.00
"data.frame" 0.12 4.48 0.02 0.75
"as.data.frame.matrix" 0.08 2.99 0.00 0.00
"as.data.frame" 0.08 2.99 0.00 0.00
"as.factor" 0.08 2.99 0.00 0.00
"factor" 0.06 2.24 0.00 0.00
"unique.default" 0.04 1.49 0.04 1.49
"unique" 0.04 1.49 0.00 0.00
"is.factor" 0.02 0.75 0.02 0.75
"match" 0.02 0.75 0.02 0.75
"structure" 0.02 0.75 0.02 0.75
"unlist" 0.02 0.75 0.02 0.75
"[.data.frame" 0.02 0.75 0.00 0.00
"[" 0.02 0.75 0.00 0.00

$sample.interval
[1] 0.02

$sampling.time
[1] 2.68

有什么办法可以加快这个操作吗?我注意到“Species”、“SizeClass”和“Infected”每个类别都有一小部分(<5),并且我事先知道这些是什么。

注释:

  • stringr::str_split_fixed 执行此任务,但速度并不快
  • 数据框实际上最初是通过在数组上调用 reshape::melt 生成的,其中 Class 及其关联级别是一个维度。如果有一种更快的方式从那里到达这里,那就太好了。
  • data.rds 位于 http://dl.getdropbox.com/u/3356641/data.rds

最佳答案

这可能会带来相当大的增长:

library(data.table)
DT <- data.table(data.df)


DT[, c("Species", "SizeClass", "Infected")
:= as.list(strsplit(Class, "\\.")[[1]]), by=Class ]
<小时/>

增加的原因:

  1. data.table 为列预分配内存
  2. data.frame 中的每一列分配都会重新分配整个数据(相比之下,data.table 则不会)
  3. by 语句允许您为每个唯一值实现一次 strsplit 任务。
<小时/>

这是整个过程的一个很好的快速方法。

# Save the new col names as a character vector 
newCols <- c("Species", "SizeClass", "Infected")

# split the string, then convert the new cols to columns
DT[, c(newCols) := as.list(strsplit(as.character(Class), "\\.")[[1]]), by=Class ]
DT[, c(newCols) := lapply(.SD, factor), .SDcols=newCols]

# remove the old column. This is instantaneous.
DT[, Class := NULL]

## Have a look:
DT[, lapply(.SD, class)]
# Time Location Replicate Population Species SizeClass Infected
# 1: integer integer integer numeric factor factor factor

DT

关于r - 当已知可能的输出时加速 `strsplit`,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/16641024/

28 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com