S4语境中表达式的非标准评估

时间:2014-10-20 09:47:05

标签: r expression s4 delayed-execution

这是借用shiny's函数exprToFunction,它在reactive函数中使用。

实际问题

如何延迟表达式的评估(例如通过arg expr指定)以便捕获"调用S4方法时的内容(与标准R函数相反)?

实施例

请注意,x_1确实存在,这就是为什么我们要延迟expr的评估并且只是"捕获"它的内容。

功能captureExpression

captureExpression <- function(
  expr,
  caller_offset = 1,
  brackets = TRUE
) {
  out <- eval(substitute(substitute(expr)), parent.frame(caller_offset))
  if (brackets && class(out) != "{") {
    out <- substitute({CODE}, list(CODE = out))
  }
  out
}

captureExpression(x_1 * 2)
# {
#     x_1 * 2
# }

在调用标准R函数时,通过未评估的 expr传递:

foo <- function(expr) {
  captureExpression(expr = expr)
}

foo(x_1 * 2)
# {
#     x_1 * 2
# }

通过未评估 expr通过调用S4方法时无效:

setGeneric(
  name = "bar",
  signature = c(
    "expr"
  ),
  def = function(
    expr,
    ...
  ) {
    standardGeneric("bar")       
  }
)
setMethod(
  f = "bar", 
  signature = signature(
    expr = "ANY"
  ), 
  definition = function(
    expr,
    ...
  ) {
  captureExpression(expr = expr, ...)    
})

应用S4方法:

bar(x_1 * 2)
# Error in bar(x_1 * 2) : 
#   error in evaluating the argument 'expr' in selecting a method for function 'bar': Error: 
# object 'x_1' not found

bar(x_1 * 2, caller_offset = 2)
# Error in bar(x_1 * 2, caller_offset = 2) :
#   error in evaluating the argument 'expr' in selecting a method for function 'bar': Error: 
# object 'x_1' not found

bar(x_1 * 2, caller_offset = 3)
# Error in bar(x_1 * 2, caller_offset = 3) :
#   error in evaluating the argument 'expr' in selecting a method for function 'bar': Error: 
# object 'x_1' not found

我想这与方法调度实际执行的方式有关。但是,也许有一种方法可以做到这一点。

0 个答案:

没有答案