2016-04-30 10 views
1

Ich habe einen Charakter Vektor von hierarchischen IDs wie diese:R: Erstellen Sie verschachtelte Liste von IDs in Zeichenvektor

ids <- c("0","1","2","3","1.1","1.2","1.3","2.1","2.2","2.11","2.21","2.22") 

Die hierarchische Struktur ist wie folgt:

1 
    1.1 
    1.2 
    1.3 
2 
    2.1 
     2.11 
    2.2 
     2.21 
     2.22 

ich verwenden möchte diagonalNetwork() aus dem Paket networkD3, um diese Hierarchie zu visualisieren. Aber diagonalNetwork() erfordert eine verschachtelte Liste definieren, die jeweils Knoten Kinder wie diese:

l <- list(name = "0", 
      children = list(
       list(name = "1", 
        children = list(
        list(name = "1.1"), 
        list(name = "1.2"), 
        list(name = "1.3") 
        ) 
      ), 
       list(name = "2", 
        children = list(
        list(name = "2.1", 
          children = list(
          list(name = "2.11") 
         ) 
        ), 
        list(name = "2.2", 
          children = list(
          list(name = "2.21"), 
          list(name = "2.22") 
         ) 
        ) 
        ) 
      ) 
      ) 
) 

Meine tatsächliche Satz von ids ist viel größer und tiefer (bis zu 6 Ziffern lang), so brauche ich einen Weg, um diese verschachtelte Liste zu erstellen automatisch . Ich begann einen data.frame mit der Erstellung, dass die IDs' Ziffern in mehreren Spalten wie dieser speichert:

df <- data.frame(root = 0, 
       a = c(1, 1, 1, 1, 2, 2, 2, 2, 2, 2), 
       b = c(NA, 1, 2, 3,NA, 1, 1, 2, 2, 2), 
       c = c(NA,NA,NA,NA,NA,NA, 1,NA, 1, 2)) 

Aber ich kann nicht denken Sie an eine Art und Weise weiter mit meiner Sorge zu bekommen. Gibt es einen vielversprechenden Ansatz?

Antwort

2

Hier ist eine mögliche Lösung basierend auf rekursive Funktion. Und es ist keine schnelle Lösung, sondern sollte für Sie arbeiten.

library(network3D) 
findChildren <- function(pa, ids) { 
    lapply(ids, function(ch){ 
     if(grepl(paste("^", pa, sep = ""), ch) && ch != pa && 
      nchar(gsub("\\.", "", ch)) == nchar(gsub("\\.", "", pa)) + 1) 
     { 
      childrenTmp = Filter(Negate(is.null), findChildren(ch, ids)) 
      if(length(childrenTmp) != 0) list(name = ch, children = childrenTmp) 
      else list(name = ch) 
     } 
    } 
    ) 
} 

myList <- list(name = "0", 
       children = Filter(
        function(x){nchar(x$name) == 1 },      
        lapply(ids[-1], function(id) { 
         childrenTmp = Filter(Negate(is.null), findChildren(id, ids)) 
         if(length(childrenTmp) != 0) list(name = id, children = childrenTmp) 
         else list(name = id) 
        } 
        ) 
       ) 
       ) 
diagonalNetwork(myList) 

enter image description here

Verwandte Themen