Jde*_*llo 4 r dplyr data.table
考虑数据:
library(data.table)
library(magrittr)
vec1 <- c("Iron", "Copper")
vec2 <- c("Defective", "Passed", "Error")
set.seed(123)
a1 <- sample(x = vec1, size = 20, replace = T)
b1 <- sample(x = vec2, size = 20, replace = T)
set.seed(1234)
a2 <- sample(x = vec1, size = 20, replace = T)
b2 <- sample(x = vec2, size = 20, replace = T)
DT <- data.table(
c(1:20), a1, b1, a2, b2
) %>% .[order(V1)]
names(DT) <- c("id", "prod_name_1", "test_1", "prod_name_2", "test_2")
Run Code Online (Sandbox Code Playgroud)
我需要过滤行,其对价值test_1或者test_2是"Passed"。因此,如果这些列都不具有指定的值,则删除该行。使用dplyr,我们可以使用filter_at()动词:
> # dplyr solution...
>
> cols <- grep(x = names(DT), pattern = "test", value = T, ignore.case = T)
>
>
> DT %>%
+ dplyr::filter_at(.vars = grep(x = names(DT), pattern = "test", value = T, ignore.case = T),
+ dplyr::any_vars(. == "Passed")) -> DT.2
>
> DT.2
id prod_name_1 test_1 prod_name_2 test_2
1 3 Iron Passed Copper Defective
2 5 Copper Passed Copper Defective
3 7 Copper Passed Iron Passed
4 8 Copper Passed Iron Error
5 11 Copper Error Copper Passed
6 14 Copper Error Copper Passed
7 16 Copper Passed Copper Error
Run Code Online (Sandbox Code Playgroud)
凉。有没有类似的方法可以执行此操作data.table?
这是我最接近的:
> lapply(seq_along(cols), function(x){
+
+ setkeyv(DT, cols[[x]])
+
+ DT["Passed"]
+
+ }) %>%
+ do.call(rbind,.) %>%
+ unique -> DT.3
>
> DT.3
id prod_name_1 test_1 prod_name_2 test_2
1: 3 Iron Passed Copper Defective
2: 5 Copper Passed Copper Defective
3: 8 Copper Passed Iron Error
4: 16 Copper Passed Copper Error
5: 7 Copper Passed Iron Passed
6: 11 Copper Error Copper Passed
7: 14 Copper Error Copper Passed
>
> identical(data.table(DT.2)[order(id)], DT.3[order(id)])
[1] TRUE
Run Code Online (Sandbox Code Playgroud)
你们中有谁有更优雅的解决方案?最好是动词中包含的东西,如dplyr::filter_at()。
我们可以在中指定'cols' .SDcols,循环遍历Data.table(.SD)的Subset 以比较该值是否为“ Passed”,Reduce将其vector与|行合并并对其进行子集化
res2 <- DT[DT[, Reduce(`|`, lapply(.SD, `==`, "Passed")), .SDcols = cols]]
Run Code Online (Sandbox Code Playgroud)
与dplyrOP帖子中的输出进行比较
identical(as.data.table(res1), res2)
#[1] TRUE
Run Code Online (Sandbox Code Playgroud)