json - 将Json文件读取到没有嵌套列表的data.frame中

标签 json r jsonlite

我正在尝试将json文件加载到r中的data.frame中。我在jsonlite包中的fromJSON函数方面有些运气-但是正在嵌套列表,并且不确定如何将输入拼合为二维data.frame。 Jsonlite读取文件为data.frame,但将嵌套列表保留在某些变量中。

在使用嵌套列表读取JSON文件到data.frame中时,是否有人有任何提示?

#*#*#*#*#*#*#*#*#*##*#*#*#*#*#*#*#*#*# HERE IS MY EXAMPLE #*#*#*#*#*#*#*#*#*##*#*#*#*#*#*#*#*#*#
# loads the packages
library("httr")
library( "jsonlite")

# downloads an example file
providers <- fromJSON( "http://fm.formularynavigator.com/jsonFiles/publish/11/47/providers.json" , simplifyDataFrame=TRUE ) 

# the flatten function breaks the name variable into three vars ( first name, middle name, last name)
providers <- flatten( providers )

# but many of the columns are still lists:
sapply( providers , class)

# Some of these lists have a single level
head( providers$facility_type )

# Some have lot more than two - for example nine
providers[ , 6][[1]]


我希望每个npi排一行,而不是每个单独列表的每个切片分开的列-以便数据框的cols分别为“ plan_id_type”,“ plan_id”,“ network_tier”的9倍,也许是colname,从0到8 。
我已经可以使用该站点:http://www.convertcsv.com/json-to-csv.htm来二维获取此文件,但是由于我要执行数百个操作,因此我希望能够动态地进行操作。这是文件:http://s000.tinyupload.com/download.php?file_id=10808537503095762868&t=1080853750309576286812811-我想使用fromJson函数获取具有此结构加载的文件作为data.frame。

这里是我尝试过的一些东西;
因此,我想到了两种方法:
第一:使用其他功能读取Json文件,我已经看过了

rjson but that reads in a list
library( rjson )
providers <- fromJSON( getURL( "https://fm.formularynavigator.com/jsonFiles/publish/11/47/providers.json") )
class( providers )


并且我尝试了RJSONIO-我尝试了Getting imported json data into a data frame in R

json-data-into-a-data-frame-in-r
library( RJSONIO )
providers <- fromJSON( getURL( "https://fm.formularynavigator.com/jsonFiles/publish/11/47/providers.json") )

json_file <- lapply(providers, function(x) {
  x[sapply(x, is.null)] <- NA
  unlist(x)
})

# but When converting the lists to a data.frame I get an error
a <- do.call("rbind", json_file)


因此,我尝试的第二种方法是将所有列表转换为data.frame中的变量

detach("package:RJSONIO", unload = TRUE )
detach("package:rjson", unload = TRUE )

library( "jsonlite")
providers <- fromJSON( "http://fm.formularynavigator.com/jsonFiles/publish/11/47/providers.json" , simplifyDataFrame=TRUE ) 
providers <- flatten( providers )


我能够拉出其中一个列表-但由于缺少,我无法合并回我的数据框

a <- data.frame(Reduce(rbind,  providers$facility_type))
length( a ) == nrow( providers )


我也尝试了以下建议:Converting nested list to dataframe。和其他东西一样好,但是没有运气

a <- sapply( providers$facility_type, unlist )
as.data.frame(t(sapply( providers$providers, unlist )) )


任何帮助非常感谢

最佳答案

更新:2016年2月21日

col_fixer更新为包含vec2col参数,该参数使您可以将列表列展平为单个字符串或一组列。



在您下载的data.frame中,我看到了几种不同的列类型。有包含相同类型向量的普通列。在列表列中,项目可能是NULL或它们本身可能是平面向量。有列表列,其中data.frame作为列表元素。有些列表列包含与主data.frame相同行数的data.frame

这是重新创建这些条件的样本数据集:

mydf <- data.frame(id = 1:3, type = c("A", "A", "B"), 
                   facility = I(list(c("x", "y"), NULL, "x")),
  address = I(list(data.frame(v1 = 1, v2 = 2, v4 = 3), 
                   data.frame(v1 = 1:2, v2 = 3:4, v3 = 5), 
                   data.frame(v1 = 1, v2 = NA, v3 = 3))))

mydf$person <- data.frame(name = c("AA", "BB", "CC"), age = c(20, 32, 23),
                          preference = c(TRUE, FALSE, TRUE))


此示例strdata.frame看起来像:

str(mydf)
## 'data.frame':    3 obs. of  5 variables:
##  $ id      : int  1 2 3
##  $ type    : Factor w/ 2 levels "A","B": 1 1 2
##  $ facility:List of 3
##   ..$ : chr  "x" "y"
##   ..$ : NULL
##   ..$ : chr "x"
##   ..- attr(*, "class")= chr "AsIs"
##  $ address :List of 3
##   ..$ :'data.frame': 1 obs. of  3 variables:
##   .. ..$ v1: num 1
##   .. ..$ v2: num 2
##   .. ..$ v4: num 3
##   ..$ :'data.frame': 2 obs. of  3 variables:
##   .. ..$ v1: int  1 2
##   .. ..$ v2: int  3 4
##   .. ..$ v3: num  5 5
##   ..$ :'data.frame': 1 obs. of  3 variables:
##   .. ..$ v1: num 1
##   .. ..$ v2: logi NA
##   .. ..$ v3: num 3
##   ..- attr(*, "class")= chr "AsIs"
##  $ person  :'data.frame':    3 obs. of  3 variables:
##   ..$ name      : Factor w/ 3 levels "AA","BB","CC": 1 2 3
##   ..$ age       : num  20 32 23
##   ..$ preference: logi  TRUE FALSE TRUE
## NULL


您可以“展平”此方法的一种方法是“修复”列表列。有三个修复程序。


flatten(来自“ jsonlite”)将处理诸如“ person”列之类的列。
可以使用toString固定诸如“ facility”列之类的列,该列会将每个元素转换为逗号分隔的项目,或者可以转换为多个列。
存在data.frame的列(有些具有多行),首先需要将其展平为单行(通过转换为“宽”格式),然后需要将它们绑定为单个data.table。 (我正在使用“ data.table”进行重塑并将行绑定在一起)。


我们可以使用以下函数来处理第二点和第三点:

col_fixer <- function(x, vec2col = FALSE) {
  if (!is.list(x[[1]])) {
    if (isTRUE(vec2col)) {
      as.data.table(data.table::transpose(x))
    } else {
      vapply(x, toString, character(1L))
    }
  } else {
    temp <- rbindlist(x, use.names = TRUE, fill = TRUE, idcol = TRUE)
    temp[, .time := sequence(.N), by = .id]
    value_vars <- setdiff(names(temp), c(".id", ".time"))
    dcast(temp, .id ~ .time, value.var = value_vars)[, .id := NULL]
  }
}


我们将把它和flatten函数集成到另一个将执行大部分处理的函数中。

Flattener <- function(indf, vec2col = FALSE) {
  require(data.table)
  require(jsonlite)
  indf <- flatten(indf)
  listcolumns <- sapply(indf, is.list)
  newcols <- do.call(cbind, lapply(indf[listcolumns], col_fixer, vec2col))
  indf[listcolumns] <- list(NULL)
  cbind(indf, newcols)
}


运行该函数将为我们提供:

Flattener(mydf)
##   id type person.name person.age person.preference facility address.v1_1
## 1  1    A          AA         20              TRUE     x, y            1
## 2  2    A          BB         32             FALSE                     1
## 3  3    B          CC         23              TRUE        x            1
##   address.v1_2 address.v2_1 address.v2_2 address.v4_1 address.v4_2 address.v3_1
## 1           NA            2           NA            3           NA           NA
## 2            2            3            4           NA           NA            5
## 3           NA           NA           NA           NA           NA            3
##   address.v3_2
## 1           NA
## 2            5
## 3           NA


或者,将向量分成不同的列:

Flattener(mydf, TRUE)
##   id type person.name person.age person.preference facility.V1 facility.V2
## 1  1    A          AA         20              TRUE           x           y
## 2  2    A          BB         32             FALSE        <NA>        <NA>
## 3  3    B          CC         23              TRUE           x        <NA>
##   address.v1_1 address.v1_2 address.v2_1 address.v2_2 address.v4_1 address.v4_2
## 1            1           NA            2           NA            3           NA
## 2            1            2            3            4           NA           NA
## 3            1           NA           NA           NA           NA           NA
##   address.v3_1 address.v3_2
## 1           NA           NA
## 2            5            5
## 3            3           NA


这是str

str(Flattener(mydf))
## 'data.frame':    3 obs. of  14 variables:
##  $ id               : int  1 2 3
##  $ type             : Factor w/ 2 levels "A","B": 1 1 2
##  $ person.name      : Factor w/ 3 levels "AA","BB","CC": 1 2 3
##  $ person.age       : num  20 32 23
##  $ person.preference: logi  TRUE FALSE TRUE
##  $ facility         : chr  "x, y" "" "x"
##  $ address.v1_1     : num  1 1 1
##  $ address.v1_2     : num  NA 2 NA
##  $ address.v2_1     : num  2 3 NA
##  $ address.v2_2     : num  NA 4 NA
##  $ address.v4_1     : num  3 NA NA
##  $ address.v4_2     : num  NA NA NA
##  $ address.v3_1     : num  NA 5 3
##  $ address.v3_2     : num  NA 5 NA
## NULL


在您的“提供者”对象上,这非常快速且一致地运行:

library(microbenchmark)
out <- microbenchmark(Flattener(providers), Flattener(providers, TRUE), flattenList(jsonRList))
out
# Unit: milliseconds
#                        expr        min         lq      mean    median        uq       max neval
#        Flattener(providers)  104.18939  126.59295  157.3744  138.4185  174.5222  308.5218   100
#  Flattener(providers, TRUE)   67.56471   86.37789  109.8921   96.3534  121.4443  301.4856   100
#      flattenList(jsonRList) 1780.44981 2065.50533 2485.1924 2269.4496 2694.1487 4397.4793   100

library(ggplot2)
qplot(y = time, data = out, colour = expr) ## Via @TylerRinker


enter image description here

关于json - 将Json文件读取到没有嵌套列表的data.frame中,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35444968/

相关文章:

json - 从Invoke-Command返回的无关数据

java - JSON 具有与不同数据类型的键相同的名称

string - 在 r 中编辑(即从中删除字符)矩阵的列名时,是否有与 paste() 相反的函数?

r - 箱线图是否在隐式绘图之前对数据进行分组?

json - 如何使用嵌套列表构造JSON

json - 提取、格式化和单独存储在数据框列中的 JSON

java - Jersey : JSON Unsupported Media Type

javascript - 原型(prototype)函数和json符号函数的区别?

r - 如何使用一些 apply 函数来解决 R 中需要两个 for 循环的问题

r - 将 JSON 列添加到 R 数据框