# We build all combinations of names, greetings and separators from our
# list of data and pass each one to paste()
data <- list(
id = c("John", "Jane"),
greeting = c("Hello.", "Bonjour."),
sep = c("! ", "... ")
)
data %>%
cross_n() %>%
map(lift_dl(paste))
# cross_n() returns the combinations in long format: many elements,
# each representing one combination. With cross_d() we'll get a
# data frame in long format. If we cross three objects, we'll get a
# data frame of three columns, with each row being a particular
# combination. This is the same format that expand.grid() returns.
args <- data %>% cross_d()
# In case a list (as opposed to a data frame) in long format is
# needed, just run as.list() after cross_d()
args %>% as.list()
# This format is often less pratical for functional programming
# because applying a function to the combinations requires a loop
out <- vector("list", length = nrow(args))
for (i in seq_along(out))
out[[i]] <- map(args, i) %>% map_call("paste")
out
# In this case, the combinations could be manipulated using dplyr
args %>% dplyr::do(
map_call(., "paste") %>% dplyr::data_frame()
)
# Unwanted combinations can be filtered out with a predicate function
filter <- function(x, y) x >= y
cross2(1:5, 1:5, .filter = filter) %>% str()
# To give names to the components of the combinations, we map
# setNames() on the product:
seq_len(3) %>%
cross2(., ., .filter = function(x, y) x == y) %>%
map(setNames, c("x", "y"))
# We can also encapsulate the arguments in a named list before
# crossing:
seq_len(3) %>%
list(x = ., y = .) %>%
cross_n(.filter = ~ .x == .y)
Run the code above in your browser using DataLab