Получить выражение, которое оценивается как точка в функции, вызываемой `magrittr`
У меня есть функция x_expression()
который печатает выражение, переданное в аргумент x
,
pacman::p_load(magrittr, rlang)
x_expression <- function(x) {
print(enquo(x))
}
y <- 1
x_expression(y)
#> <quosure>
#> expr: ^y
#> env: global
y %>% x_expression()
#> <quosure>
#> expr: ^.
#> env: 0x7ff27c36a610
Итак, вы можете видеть, что он знает y
был передан ему, но когда y
с помощью %>%
, функция возвращает отпечатки .
, Есть ли способ восстановить y
в том случае, если он по трубопроводу или он ушел навсегда? Вкратце, то, что я хочу, это функция, как x_expression()
но тот, который будет печатать y
в обоих случаях выше.
Этот вопрос действительно похож на " Get name of dataframe", переданного через pipe в R, но он немного более общий. Этот человек просто хочет имя фрейма данных, я хочу выражение, что бы это ни было. Тем не менее, один и тот же ответ, вероятно, будет применяться к обоим. Мне не нравится ответ на этот почти дублирующий вопрос, а также автор этого ответа.
1 ответ
y
не "ушел навсегда", потому что канал вызывает вашу функцию, и он также знает о y
, Есть способ восстановить y
, но это требует некоторого обхода стека вызовов. Чтобы понять, что происходит, мы будем использовать ?sys.frames
а также ?sys.calls
:
"sys.calls" и "sys.frames" дают список всех активных вызовов и фреймов соответственно, а "sys.parents" возвращает целочисленный вектор индексов родительских фреймов каждого из этих фреймов.
Если мы рассыпаем их по всему вашему x_expression()
мы можем видеть, что происходит, когда мы звоним y %>% x_expression()
из глобальной среды:
x_expression <- function(x) {
print( enquo(x) )
# <quosure>
# expr: ^.
# env: 0x55c03f142828 <---
str(sys.frames())
# Dotted pair list of 9
# $ :<environment: 0x55c03f151fa0>
# $ :<environment: 0x55c03f142010>
# ...
# $ :<environment: 0x55c03f142828> <---
# $ :<environment: 0x55c03f142940>
str(sys.calls())
# Dotted pair list of 9
# $ : language y %>% x_expression() <---
# $ : language withVisible(eval(...
# ...
# $ : language function_list[[k]...
# $ : language x_expression(.)
}
Я выделил важные части <---
, Обратите внимание на то, что enquo
живет в родительской среде функции (второй из нижней части стека), в то время как вызов канала, который знает о y
все пути наверху стека.
Есть несколько способов пройтись по стеку. @ MrFlick ответ на аналогичный вопрос, а также этот вопрос GitHub пересекают фреймы / среды из sys.frames()
, Здесь я покажу альтернативу, которая пересекает sys.calls()
и анализирует выражения, чтобы найти %>%
,
Первая часть головоломки - определить функцию, которая преобразует выражение в его абстрактное синтаксическое дерево (AST):
# Recursively constructs Abstract Syntax Tree for a given expression
getAST <- function( ee ) { as.list(ee) %>% purrr::map_if(is.call, getAST) }
# Example: getAST( quote(a %>% b) )
# List of 3
# $ : symbol %>%
# $ : symbol a
# $ : symbol b
Теперь мы можем систематически применять эту функцию ко всему sys.calls()
стек. Цель состоит в том, чтобы идентифицировать AST, где первый элемент %>%
; тогда второй элемент будет соответствовать левой стороне трубы (symbol a
в a %>% b
пример). Если существует более одного такого AST, то мы находимся во вложенном %>%
трубный сценарий. В этом случае последний AST в списке будет самым низким в стеке вызовов и ближайшим к нашей функции.
x_expression2 <- function(x) {
sc <- sys.calls()
ASTs <- purrr::map( as.list(sc), getAST ) %>%
purrr::keep( ~identical(.[[1]], quote(`%>%`)) ) # Match first element to %>%
if( length(ASTs) == 0 ) return( enexpr(x) ) # Not in a pipe
dplyr::last( ASTs )[[2]] # Second element is the left-hand side
}
(Незначительное примечание: я использовал enexpr()
вместо enquo()
для обеспечения согласованного поведения функции в и из трубы. поскольку sys.calls()
traversal возвращает выражение, а не выражение, мы хотим сделать то же самое в случае по умолчанию.)
Новая функция довольно надежна и работает внутри других функций, включая вложенные %>%
трубы:
x_expression2(y)
# y
y %>% x_expression2()
# y
f <- function() {x_expression2(v)}
f()
# v
g <- function() {u <- 1; u %>% x_expression2()}
g()
# u
y %>% (function(z) {w <- 1; w %>% x_expression2()}) # Note the nested pipes
# w