我试图将嵌套列表拆分为一个组变量。请考虑以下结构:
> str(L1)
List of 2
$ names:List of 2
..$ first: chr [1:5] "john" "lisa" "anna" "mike" ...
..$ last : chr [1:5] "johnsson" "larsson" "johnsson" "catell" ...
$ stats:List of 2
..$ physical:List of 2
.. ..$ age : num [1:5] 14 22 53 23 31
.. ..$ height: num [1:5] 165 176 179 182 191
..$ mental :List of 1
.. ..$ iq: num [1:5] 102 104 99 87 121现在,我需要生成两个列表,这两个列表都使用L1$names$last进行拼接,从而生成L2和L3,如下所示:
L1$names$last L2:按L2分组的结果
> str(L2)
List of 3
$ johnsson:List of 2
..$ names:List of 1
.. ..$ first: chr [1:2] "john" "anna"
..$ stats:List of 2
.. ..$ physical:List of 2
.. .. ..$ age : num [1:2] 14 53
.. .. ..$ height: num [1:2] 165 179
.. ..$ mental :List of 1
.. .. ..$ iq: num [1:2] 102 99
$ larsson :List of 2
..$ names:List of 1
.. ..$ first: chr [1:2] "lisa" "steven"
..$ stats:List of 2
.. ..$ physical:List of 2
.. .. ..$ age : num [1:2] 22 31
.. .. ..$ height: num [1:2] 176 191
.. ..$ mental :List of 1
.. .. ..$ iq: num [1:2] 104 121
$ catell :List of 2
..$ names:List of 1
.. ..$ first: chr "mike"
..$ stats:List of 2
.. ..$ physical:List of 2
.. .. ..$ age : num 23
.. .. ..$ height: num 182
.. ..$ mental :List of 1
.. .. ..$ iq: num 87L1$names$last L3:每个组只允许出现一个L3
List of 2
$ 1:List of 2
..$ names:List of 2
.. ..$ first: chr [1:3] "john" "lisa" "mike"
.. ..$ last : chr [1:3] "johnsson" "larsson" "catell"
..$ stats:List of 2
.. ..$ physical:List of 2
.. .. ..$ age : num [1:3] 14 22 23
.. .. ..$ height: num [1:3] 165 176 182
.. ..$ mental :List of 1
.. .. ..$ iq: num [1:3] 102 104 87
$ 2:List of 2
..$ names:List of 2
.. ..$ first: chr [1:2] "anna" "steven"
.. ..$ last : chr [1:2] "johnsson" "larsson"
..$ stats:List of 2
.. ..$ physical:List of 2
.. .. ..$ age : num [1:2] 53 31
.. .. ..$ height: num [1:2] 179 191
.. ..$ mental :List of 1
.. .. ..$ iq: num [1:2] 99 121我尝试过应用this solution,但这似乎不适用于嵌套列表。
可复制代码:
L1 <- list("names" = list("first" = c("john","lisa","anna","mike","steven"),"last" = c("johnsson","larsson","johnsson","catell","larsson")),"stats" = list("physical" = list("age" = c(14,22,53,23,31), "height" = c(165,176,179,182,191)), "mental" = list("iq" = c(102,104,99,87,121))))
L2 <- list("johnsson" = list("names" = list("first" = c("john","anna")),"stats" = list("physical" = list("age" = c(14,53), "height" = c(165,179)), "mental" = list("iq" = c(102,99)))), "larsson" = list("names" = list("first" = c("lisa","steven")),"stats" = list("physical" = list("age" = c(22,31), "height" = c(176,191)), "mental" = list("iq" = c(104,121)))), "catell" = list("names" = list("first" = "mike"),"stats" = list("physical" = list("age" = 23, "height" = 182), "mental" = list("iq" = 87))))
L3 <- list("1" = list("names" = list("first" = c("john","lisa","mike"),"last" = c("johnsson","larsson","catell")),"stats" = list("physical" = list("age" = c(14,22,23), "height" = c(165,176,182)), "mental" = list("iq" = c(102,104,87)))), "2" = list("names" = list("first" = c("anna","steven"),"last" = c("johnsson","larsson")),"stats" = list("physical" = list("age" = c(53,31), "height" = c(179,191)), "mental" = list("iq" = c(99,121)))))编辑:--请注意,与提供的示例相比,实际的数据集相当大,嵌套得更深。
发布于 2017-07-17 22:39:58
通常,为了修改列表,您需要使用递归。例如,考虑以下功能:
foo <- function(x, idx) {
if (is.list(x)) {
return(lapply(x, foo, idx = idx))
}
return(x[idx])
}它将一些列表作为x和一些索引( idx )。它将检查x是否是一个列表,如果是这样的话,它将自己应用于列表的所有子元素。一旦x不再是一个列表,我们就接受idx给出的元素。在整个过程中,原始清单的结构将保持不变。
这里是一个完整的例子。注意,这段代码假设列表中的所有向量都有5个元素。
L1 <- list("names" = list("first" = c("john","lisa","anna","mike","steven"),"last" = c("johnsson","larsson","johnsson","catell","larsson")),"stats" = list("physical" = list("age" = c(14,22,53,23,31), "height" = c(165,176,179,182,191)), "mental" = list("iq" = c(102,104,99,87,121))))
L2 <- list("johnsson" = list("names" = list("first" = c("john","anna")),"stats" = list("physical" = list("age" = c(14,53), "height" = c(165,179)), "mental" = list("iq" = c(102,99)))), "larsson" = list("names" = list("first" = c("lisa","steven")),"stats" = list("physical" = list("age" = c(22,31), "height" = c(176,191)), "mental" = list("iq" = c(104,121)))), "catell" = list("names" = list("first" = "mike"),"stats" = list("physical" = list("age" = 23, "height" = 182), "mental" = list("iq" = 87))))
L3 <- list("1" = list("names" = list("first" = c("john","lisa","mike"),"last" = c("johnsson","larsson","catell")),"stats" = list("physical" = list("age" = c(14,22,23), "height" = c(165,176,182)), "mental" = list("iq" = c(102,104,87)))), "2" = list("names" = list("first" = c("anna","steven"),"last" = c("johnsson","larsson")),"stats" = list("physical" = list("age" = c(53,31), "height" = c(179,191)), "mental" = list("iq" = c(99,121)))))
# make L2
foo <- function(x, idx) {
if (is.list(x)) {
return(lapply(x, foo, idx = idx))
}
return(x[idx])
}
levels <- unique(L1$names$last)
L2_2 <- vector("list", length(levels))
names(L2_2) <- levels
for (i in seq_along(L2_2)) {
idx <- L1$names$last == names(L2_2[i])
L2_2[[i]] <- list(names = foo(L1$names[-2], idx),
stats = foo(L1$stats, idx))
}
identical(L2, L2_2)
str(L2)
str(L2_2)
# make L3
dups <- duplicated(L1$names$last)
L3_2 <- vector("list", 2)
names(L3_2) <- 1:2
for (i in 1:2) {
if (i == 1)
idx <- !dups
else
idx <- dups
L3_2[[i]] <- foo(L1, idx)
}
identical(L3, L3_2)
str(L3)
str(L3_2)发布于 2017-07-25 15:45:13
这不是一个完整的答案,但我希望它能有所帮助。
看看这是否适用于L3:
x = data.frame(L1, stringsAsFactors = F)
y = x[order(x$names.last),]
y$seq = 1
y$seq = ifelse(y$names.last == shift(y$names.last),shift(y$seq)+1,1)
y$seq[1] = 1
z = list(list(names=list(first=z[[1]]$names.first, last=z[[1]]$names.last), stats=list(physical = list(age =z[[1]]$stats.physical.age, height= z[[1]]$stats.physical.height), mental=list(iq= z[[1]]$stats.iq))), list(names=list(first=z[[2]]$names.first, last=z[[2]]$names.last), stats=list(physical = list(age =z[[2]]$stats.physical.age, height= z[[2]]$stats.physical.height), mental=list(iq= z[[2]]$stats.iq))))最后一部分(z),将其转换回列表,可以通过循环完成。假设没有出现太多相同的名称,循环就不会太慢。
您说它更嵌套,在这种情况下,您需要添加is.null和或tryCatch函数来处理错误。
https://stackoverflow.com/questions/45154451
复制相似问题