r - 解除多列嵌套

标签 r tidyr

我有这种类型的数据,具有多个列表列:

df <- structure(list(go = c("go after it", "here we go", "he went bust", 
                            "go get it go", "i 'm gonna go", "she 's going berserk"), left = list(
                              "", "we", "he", c("", "it"), c("'m", "gonna"), "'s"), node = list(
                                "go", "go", "went", c("go", "go"), c("gonna", "go"), "going"), 
                     right = list("after", "", "bust", c("get", ""), c("go", ""
                     ), "berserk")), class = "data.frame", row.names = c(NA, -6L
                     ))

我的目标是取消列出 leftnoderight 三列。 unnest_longerunnest_auto 似乎都是朝着正确方向迈出的一步,但我不确定如何从那里继续:

library(tidyr)
df %>%
  unnest_longer(node) # or:   unnest_auto(node)
# A tibble: 8 × 4
  go                   left      node   right    
  <chr>                <list>    <chr>  <list>   
1 go after it          <chr [0]> go     <chr [1]>
2 here we go           <chr [1]> go     <chr [0]>
3 he went bust         <chr [1]> went   <chr [1]>
4 go get it go         <chr [1]> go     <chr [1]>
5 go get it go         <chr [1]> go     <chr [1]>
6 i 'm gon na go       <chr [2]> gon na <chr [1]>
7 i 'm gon na go       <chr [2]> go     <chr [1]>
8 she 's going berserk <chr [1]> going  <chr [1]>

预期结果是这样的:

# go                         left      node     right  
# <chr>                      <chr>    <chr>     <chr>  
# 1 go after it                         go       after  
# 2 here we go                 we       go       
# 3 he went bust               he       went     bust   
# 4 go get it go                        go       get    
# 5 go get it go               it       go           
# 6 i 'm gon na go             'm       gon na   go     
# 7 i 'm gon na go         gon na       go            
# 8 she 's going berserk       's       going    berserk 

最佳答案

我们可以使用where

library(dplyr)
library(tidyr)
library(purrr)
df %>% 
   mutate(mx = invoke(pmax, across(where(is.list), lengths))) %>% 
   mutate(across(where(is.list), ~ map2(.x, mx, ~ {
        length(.x) <- .y
        if(cur_column() == "left") .x <- .x[order(!is.na(.x))]
        .x})), mx = NULL) %>%
   unnest(where(is.list))
# A tibble: 8 × 4
  go                   left  node   right  
  <chr>                <chr> <chr>  <chr>  
1 go after it          <NA>  go     after  
2 here we go           we    go     <NA>   
3 he went bust         he    went   bust   
4 go get it go         <NA>  go     get    
5 go get it go         it    go     <NA>   
6 i 'm gon na go       'm    gon na go     
7 i 'm gon na go       na    go     <NA>   
8 she 's going berserk 's    going  berserk

更新

根据OP的评论,以前的解决方案有效

df %>%
    unnest(where(is.list))

如果有NULL元素,请指定keep_empty = TRUE(在OP的数据中,某些元素为空("")而不是 NULL,所以前一个应该也可以工作

df %>%
     unnest(where(is.list), keep_empty = TRUE)
# A tibble: 8 × 4
  go                   left    node  right    
  <chr>                <chr>   <chr> <chr>    
1 go after it          ""      go    "after"  
2 here we go           "we"    go    ""       
3 he went bust         "he"    went  "bust"   
4 go get it go         ""      go    "get"    
5 go get it go         "it"    go    ""       
6 i 'm gonna go        "'m"    gonna "go"     
7 i 'm gonna go        "gonna" go    "" 

关于r - 解除多列嵌套,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/70228513/

相关文章:

r - 如何重新缩放 bookdown/rmarkdown 网站的本地镜像?

r - abline 不尊重 R 中的截距

r - 在构面上固定位置的文本注释

r - 传播/聚集错误 : Must supply a symbol or a string as argument

r - 将 tibble 作为带有行名称的矩阵展开

r - 需要通过 r 中的指定间隔识别连续观察的新方法

c - R-C接口(interface): extracting an object from an environment

r - 如何在 R 中使用分组平均值填充 NA 值

r - dplyr 和 tidyr : convert long to wide format and arrange columns

r - 根据组的第一个值填充每组的 NA 值