Назначить несколько объектов .GlobalEnv из функции - PullRequest
21 голосов
/ 15 марта 2012

В сообщении, опубликованном здесь, день назад, я задаюсь вопросом, как назначать значения нескольким объектам в глобальной среде изнутри функции.Это моя попытка использовать lapply (assign может быть безопаснее, чем <<-, но я никогда не использовал его и не знаком с ним).

#fake data set
df <- data.frame(
  x.2=rnorm(25),
  y.2=rnorm(25),
  g=rep(factor(LETTERS[1:5]), 5)
)

#split it into a list of data frames
LIST <- split(df, df$g)

#pre-allot 5 objects in R with class data.frame()
V <- W <- X <- Y <- Z <- data.frame()

#attempt to assign the data frames in the LIST to the objects just created
lapply(seq_along(LIST), function(x) c(V, W, X, Y, Z)[x] <<- LIST[[x]])

Пожалуйста, не стесняйтесь сокращать любые / все части моего кода, чтобы эта работа работала (или работала лучше / быстрее).

1 Ответ

29 голосов
/ 15 марта 2012

Обновление 2018-10-10:

Самый краткий способ выполнить эту конкретную задачу - использовать list2env() примерно так:

## Create an example list of five data.frames
df <- data.frame(x = rnorm(25),
                 g = rep(factor(LETTERS[1:5]), 5))
LIST <- split(df, df$g)

## Assign them to the global environment
list2env(LIST, envir = .GlobalEnv)

## Check that it worked
ls()
## [1] "A"    "B"    "C"    "D"    "df"   "E"    "LIST"

Оригинальный ответ, демонстрирующий использование assign ()

Вы правы, что assign() - правильный инструмент для работы. Его аргумент envir дает вам точный контроль над местом назначения - контроль, который недоступен ни с <-, ни с <<-.

Так, например, чтобы присвоить значение X объекту с именем NAME в глобальной среде, вы должны сделать:

assign("NAME", X, envir = .GlobalEnv)

В вашем случае:

df <- data.frame(x = rnorm(25),
                 g = rep(factor(LETTERS[1:5]), 5))
LIST <- split(df, df$g)
NAMES <- c("V", "W", "X", "Y", "Z")

lapply(seq_along(LIST), 
       function(x) {
           assign(NAMES[x], LIST[[x]], envir=.GlobalEnv)
        }
)

ls()
[1] "df"    "LIST"  "NAMES" "V"     "W"     "X"     "Y"     "Z"    
...