r - 是否有一个 R 函数可以读取带有\n 作为(列)分隔符的文本文件?

标签 r data.table readr

问题

我正在尝试想出一种简洁/快速的方法来将由换行符 (\n) 字符分隔的文件读取到多个列中。

本质上,在给定的输入文件中,输入文件中的多行应该成为输出中的单行,但是大多数文件读取函数明智地将换行符解释为表示新行,因此它们最终成为数据框单列。这是一个例子:

输入文件如下所示:

Header Info
2021-01-01
text
...
@
2021-01-02
text
...
@
...

哪里...代表输入文件中可能的多行,@表示输出数据框中一行的真正结尾。所以在读取这个文件时,它应该变成这样的数据框(忽略标题):

<表类="s-表"> <头> X1 X2 <日>... Xn <正文> 2021-01-01 文本 ... ... 2021-01-02 文本 ... ... ... ... ... ...

我的尝试

我试过了 base , data.table , readrvroom ,它们都有两个输出之一,或者是具有单列的数据框,或者是向量。我想避免 for 循环,所以我当前的解决方案是使用 base::readLines() , 将其作为字符向量读取,然后手动添加一些“适当的”列分隔符(例如 ; ),然后再次加入和拆分。

# Save the example data to use as input
writeLines(c("Header Info", "2021-01-01", "text", "@", "2021-01-02", "text", "@"), "input.txt")

input <- readLines("input.txt")
input <- paste(input[2:length(input)], collapse = ";") # Skip the header
input <- gsub(";@;*", replacement = "\n", x = input)
input <- strsplit(unlist(strsplit(input, "\n")), ";")
input <- do.call(rbind.data.frame, input)

# Clean up the example input
unlink("input.txt")

我上面的代码有效并给出了预期的结果,但肯定有更好的方法?? 编辑:这是函数内部的,因此任何简化的部分(也许是大部分)目的是提高速度。

提前致谢!

最佳答案

1) 读入数据,找到给定逻辑变量 at 的 @ 符号,然后创建一个分组变量 g,该变量对每个所需行都有不同的值。最后使用 tapply with paste 将其重新加工成可以使用 read.table 读取的行并读取它。 (如果数据中有逗号,则使用其他一些分隔符。)

L <- readLines("input.txt")[-1]
at <- grepl("@", L)
g <- cumsum(at)
read.table(text = tapply(L[!at], g[!at], paste, collapse = ","), 
  sep = ",", col.names = cnames)

给出这个数据框:

          V1   V2
1 2021-01-01 text
2 2021-01-02 text

2) 另一种方法是将数据重新处理为 dcf 格式,方法是删除 @ 符号并在其他行前面加上列名和冒号。然后使用 read.dcf。 cnames 是您要使用的列名称的字符向量。

cnames <- c("Date", "Text")

L <- readLines("input.txt")[-1]
LL <- sub("@", "", paste0(c(paste0(cnames, ": "), ""), L))
DF <- as.data.frame(read.dcf(textConnection(LL)))
DF[] <- lapply(DF, type.convert, as.is = TRUE)
DF

给出这个数据框:

        Date Text
1 2021-01-01 text
2 2021-01-02 text

3) 这种方法只是将数据重新整形为矩阵,然后将其转换为数据框。请注意,(1) 将数字列转换为数字列,而这个只是将它们保留为字符。

L <- readLines("input.txt")[-1]
k <- grep("@", L)[1]
as.data.frame(matrix(L, ncol = k, byrow = TRUE))[, -k]
##           V1   V2
## 1 2021-01-01 text
## 2 2021-01-02 text

基准

这个问题没有提到速度是一个考虑因素,但后来在评论中提到了它。根据以下基准测试中的数据,(1) 运行速度是问题中代码的两倍多,(3) 运行速度快近 25 倍。

library(microbenchmark)

writeLines(c("Header Info", 
   rep(c("2021-01-01", "text", "@", "2021-01-02", "text", "@"), 10000)), 
   "input.txt")

library(microbenchmark)
writeLines(c("Header Info", rep(c("2021-01-01", "text", "@", "2021-01-02", "text", "@"), 10000)), "input.txt")

microbenchmark(times = 10,
ques = {
  input <- readLines("input.txt")
  input <- paste(input[2:length(input)], collapse = ";") # Skip the header
  input <- gsub(";@;*", replacement = "\n", x = input)
  input <- strsplit(unlist(strsplit(input, "\n")), ";")
  input <- do.call(rbind.data.frame, input)
},
ans1 = {
  L <- readLines("input.txt")[-1]
  at <- grepl("@", L)
  g <- cumsum(at)
  read.table(text = tapply(L[!at], g[!at], paste, collapse = ","), sep = ",")
},
ans3 = {
  L <- readLines("input.txt")[-1]
  k <- grep("@", L)[1]
  as.data.frame(matrix(L, ncol = k, byrow = TRUE))[, -k]
})
## Unit: milliseconds
##  expr     min      lq    mean  median      uq     max neval cld
##  ques 1146.62 1179.65 1188.74 1194.78 1200.11 1219.01    10   c
##  ans1  518.95  522.75  548.33  532.59  561.55  647.14    10  b 
##  ans3   50.47   51.19   51.68   51.69   52.25   52.52    10 a  

关于r - 是否有一个 R 函数可以读取带有\n 作为(列)分隔符的文本文件?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/67060174/

相关文章:

r - 如何在 R 中创建带有图案而不是颜色的堆栈条形图

r - 将 JSON 传递到 R 函数

R数据表: update with shift() does not work as expected

string - 为数据中的每一行构造一个标识符字符串

r - 如何有效地将所有光栅文件导入 R?

r - 49 个图以 7x7 矩阵排列

r - data.table 上的数学运算(在 R 中)

R:如何读取带有 data.table::fread 的 CSV 文件,其中逗号为小数,点为千位分隔符 ="."

r - 如何将空格分隔的字符串转换为r中的数据框

将 txt 文件的目录逐行读取到 R 数据框中,文件名作为一列