Question
I have a list of many data.frames that I want to merge. The issue here is that
each data.frame differs in terms of the number of rows and columns, but they
all share the key variables (which I've called "var1"
and "var2"
in the
code below). If the data.frames were identical in terms of columns, I could
merely rbind
, for which plyr's
[rbind.fill](https://stackoverflow.com/questions/2851327/r-converting-a-list-
of-data-frames-into-one-data-frame) would do the job, but that's not the case
with these data.
Because the merge
command only works on 2 data.frames, I turned to the
Internet for ideas. I got this one from [here](http://www.mail-
archive.com/r-help@r-project.org/msg46572.html), which worked perfectly in R
2.7.2, which is what I had at the time:
merge.rec <- function(.list, ...){
if(length(.list)==1) return(.list[[1]])
Recall(c(list(merge(.list[[1]], .list[[2]], ...)), .list[-(1:2)]), ...)
}
And I would call the function like so:
df <- merge.rec(my.list, by.x = c("var1", "var2"),
by.y = c("var1", "var2"), all = T, suffixes=c("", ""))
But in any R version after 2.7.2, including 2.11 and 2.12, this code fails with the following error:
Error in match.names(clabs, names(xi)) :
names do not match previous names
(Incidently, I see other references to this error [elsewhere](https://stackoverflow.com/questions/2504543/merge-multiple-data- frames-error-in-match-namesclabs-namesxi-names-do-n) with no resolution).
Is there any way to solve this?
Answer
Another question asked specifically [how to perform multiple left joins using dplyr in R ](https://stackoverflow.com/questions/32066402/how-to-perform- multiple-left-joins-using-dplyr-in-r). The question was marked as a duplicate of this one so I answer here, using the 3 sample data frames below:
x <- data.frame(i = c("a","b","c"), j = 1:3, stringsAsFactors=FALSE)
y <- data.frame(i = c("b","c","d"), k = 4:6, stringsAsFactors=FALSE)
z <- data.frame(i = c("c","d","a"), l = 7:9, stringsAsFactors=FALSE)
The answer is divided in three sections representing three different ways to
perform the merge. You probably want to use the purrr
way if you are already
using the tidyverse packages. For comparison purposes below, you'll find a
base R version using the same sample dataset.
1) Join them withreduce
from the purrr
package:
The purrr
package provides a reduce
function which has a concise syntax:
library(tidyverse)
list(x, y, z) %>% reduce(left_join, by = "i")
# A tibble: 3 x 4
# i j k l
# <chr> <int> <int> <int>
# 1 a 1 NA 9
# 2 b 2 4 NA
# 3 c 3 5 7
You can also perform other joins, such as a full_join
or inner_join
:
list(x, y, z) %>% reduce(full_join, by = "i")
# A tibble: 4 x 4
# i j k l
# <chr> <int> <int> <int>
# 1 a 1 NA 9
# 2 b 2 4 NA
# 3 c 3 5 7
# 4 d NA 6 8
list(x, y, z) %>% reduce(inner_join, by = "i")
# A tibble: 1 x 4
# i j k l
# <chr> <int> <int> <int>
# 1 c 3 5 7
2)dplyr::left_join()
with base R Reduce()
:
list(x,y,z) %>%
Reduce(function(dtf1,dtf2) left_join(dtf1,dtf2,by="i"), .)
# i j k l
# 1 a 1 NA 9
# 2 b 2 4 NA
# 3 c 3 5 7
3) Base Rmerge()
with base R Reduce()
:
And for comparison purposes, here is a base R version of the left join based on Charles's answer.
Reduce(function(dtf1, dtf2) merge(dtf1, dtf2, by = "i", all.x = TRUE),
list(x,y,z))
# i j k l
# 1 a 1 NA 9
# 2 b 2 4 NA
# 3 c 3 5 7