【问题标题】:Deduplicating/collapsing records in an R dataframeR 数据框中的重复数据删除/折叠记录
【发布时间】:2013-06-20 10:25:56
【问题描述】:

我有一个由不同个体组成的数据集,其中每个个体都有一个唯一的 ID。每个人都可以在数据集中出现多次,但据我了解,除了一个或两个变量不同(每个人大约有 80 个)之外,数据集中相同用户 ID 的每个条目的值应该相同。

如果可以,我想尝试折叠数据。我的主要障碍是我需要回填的某些空值。我正在寻找一个可以完成重复数据删除的函数,看起来像这样:

# Build sample dataset
df1 = data.frame(id=rep(1:6,2)                 
                ,classA=rep(c('a','b'),6)
                ,classB=rep(c(1001:1006),2)
                )
df1= df1[order(df1$id),]
df1$classC=c('a',NA,'b',NA,NA,NA,'e','d', NA, 'f', NA, NA)
df1[10,"classB"]=NA
df1=df1[df1$id!=6,]

#sample dataset
> df1
   id classA classB classC
1   1      a   1001      a
7   1      a   1001   <NA>
2   2      b   1002      b
8   2      b   1002   <NA>
3   3      a   1003   <NA>
9   3      a   1003   <NA>
4   4      b   1004      e
10  4      b   1004      d
5   5      a   1005   <NA>
11  5      a     NA      f        

# what I'm looking for
> deduplicate(df1, on='id')
  id classA classB classC
1  1      a   1001      a
2  2      b   1002      b
3  3      a   1003   <NA>
4  4      b   1004      d
5  4      b   1004      e
6  5      a   1005      f     

【问题讨论】:

  • 因为我在添加 id=5 的案例时搞砸了我的数据示例:p。固定
  • 也许您应该考虑将您的一些问题标记为已回答(如果他们回答了)??

标签: r join merge dataframe na


【解决方案1】:

这个怎么样? (使用data.table的解决方案)

require(data.table)
DT <- data.table(df1)
# ignore the warning here...
unique(DT[, lapply(.SD, function(x) x[!is.na(x)]), by = id])

   id classA classB classC
1:  1      a   1001      a
2:  2      b   1002      b
3:  3      a   1003     NA
4:  4      b   1004      e
5:  4      b   1004      d
6:  5      a   1005      f

一些解释:

  • by = id 部分将您的 data.table DT 拆分/分组id
  • .SD 是一个只读变量,它会自动为每个 id 一次提取每个拆分/组。
  • 因此,我们将DT 拆分为id,对于每个拆分部分,使用lapply(获取每一列)并删除所有NAs。现在,如果您说a, NA,那么NA 将被删除并返回a。但输入的长度为 2 (a, NA)。因此,它会自动回收 a 以适应大小 (=2)。所以,基本上我们用一些已经存在的值替换所有的 NA。当两者都是NA(如NA, NA)时,NAs 将被返回(再次通过回收)。
  • 如果你看这部分DT[, lapply(.SD, function(x) x[!is.na(x)]), by = id],你应该能明白做了什么。每个NA 都将被替换。所以,我们需要做的就是提取unique 行。这就是为什么它用unique 包裹起来。

希望这会有所帮助。您必须进行一些实验才能更好地理解。我建议从这里开始:DT[, print(.SD), by=id]


最终解决方案:

我刚刚意识到,如果您有 id=4 的另一行与 classC = NA (其他一切都相同),则上述解决方案将不起作用。这是由于回收问题。这段代码应该可以修复它。

unique(DT[, lapply(.SD, function(x) {x[is.na(x)] <- x[!is.na(x)][1]; x}), by = id])

【讨论】:

  • 这绝对是我想要的,但我最近才开始学习 data.table 包。你能不能带我了解一下这是做什么的,所以我不只是盲目地使用别人的代码?顺便说一句,这绝对看起来很棒。
  • 感谢您的解释。我还不能成功地使用它来删除我的完整数据集,但我认为这是我的变量和数据清洁度的一个属性。您的解决方案绝对满足我使用示例数据设置的要求(并进一步让我对 data.table 包感兴趣),因此我将其标记为已解决。谢谢您的帮助!我肯定会保留这个 sn-p 以备将来数据处理。
  • 如果你在这里,我会吻你。将其包装在一个函数中,该函数循环遍历每个变量,对 ID 进行重复数据删除,以确定哪些变量在我的 ID 变量中是唯一的......就像黄油一样,并且在我的数据的 10000x80 子集上超级快。
  • hmmm,只是为了确定,如果您有兴趣,请发布此解决方案本身失败的示例,我将对其进行调查以查看问题所在。 PS:我认为您使用了最终解决方案
  • 我认为我没有清楚地表达我的问题:问题不在于您的解决方案本身没有让我失望,而是我的数据集中的某些变量具有唯一的非空值对于每一行。这些变量可以忽略。是的,我确实使用了你的“最终解决方案”。
【解决方案2】:

我会,首先检查是否有重复的行 idmissing classC 并像这样删除它们:

dd <- df1[duplicated(df1[,1]) & is.na(df1$classC), ]
df1[setdiff(rownames(df1), rownames(dd)), ]
  id classA classB classC
1  1      a   1001      a
2  2      b   1002      b
3  3      a   1003   <NA>
4  4      b   1004      e
8  4      b   1004      d

编辑

我认为将以上内容概括为许多列,一个想法是使用melt 将您的数据以长格式放置,例如:

library(reshape2)
dat.m  <- melt(df1,id.vars='id')
dd <- dat.m[order(dat.m$id),]
rr <- dd[duplicated(dd$id) & is.na(dd$value),]
kk <- dd[setdiff(rownames(dd), rownames(rr)), ]
kk <- kk[!duplicated(kk),]
dcast(kk,id~variable,drop=FALSE,fun.aggregate=list,fill=list(NA))
  id classA classB classC
1  1      a   1001      a
2  2      b   1002      b
3  3      a   1003     NA
4  4      b   1004   e, d
5  5      a   1005      f

最终结果与您想要的输出略有不同,但您可以通过一些工作(例如 strsplit)得到它。

【讨论】:

  • 不错的开始,但我的数据中有大约 80 列:您能否将其概括为处理可能包含 NA 值的多个列?此外,这也不能满足第一个条目填充了 var1 和 var2 null,而第二个条目填充了 var1 null 和 var2 的情况(我没有在示例数据中显示)。
  • @DavidMarx 我为许多专栏编辑了我的答案。我没明白你关于 var1,var2,...也许你可以添加一个数据示例。
  • 用 id=5 更新了我的问题,以说明我试图描述的情况。基本上,我不是在假设空值都在一个记录中的情况下进行操作的。一条记录可能包含需要由另一条记录填充的空值,而另一条记录可能具有需要由第一条记录填充的空值。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2018-09-18
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2020-04-03
  • 2018-09-19
  • 2020-09-15
相关资源
最近更新 更多