1

我在 R6 类中使用活动绑定在分配给字段之前检查值。我以为我可以使用闭包来生成如下绑定,但这不起作用。

绑定没有以我期望的方式进行评估(完全?),因为错误显示了闭包的name参数。我错过了什么?

library(R6)
library(pryr)

# pass a field name to create its binding 
generate_binding <- function(name) {
  function(value) {
    if (!missing(value) && length(value) > 0) {
      private$name <- value
    } 
    private$name
  }
}

bind_x = generate_binding(x_)
# created as intended:
unenclose(bind_x)
# function (value) 
# {
#     if (!missing(value) && length(value) > 0) {
#         private$x_ <- value
#     }
#     private$x_
# }

MyClass <- R6::R6Class("MyClass",
  private = list(
    x_ = NULL
  ),
  active = list(
    x = bind_x
  ),
)

my_class_instance <- MyClass$new()
my_class_instance$x <- "foo"
# Error in private$name <- value :
#   cannot add bindings to a locked environment
4

1 回答 1

4

我认为您误解了闭包的工作原理。unenclose在这里是一个红鲱鱼(因为它实际上并没有向您展示关闭的样子)。闭包包含语句private$name <- value——它不包含语句private$x_ <- value

这个问题的通常解决方案是重写闭包,以便将未评估的name参数解析为其字符串表示,然后用于子集private环境(private[[name]] <- value)。但是,这在这里不起作用,因为 R6 活动绑定会剥离其封闭环境的闭包。

这就是unenclose进来的地方:

MyClass <- R6::R6Class("MyClass",
  private = list(
    x_ = NULL
  ),
  active = list(
    x = pryr::unenclose(bind_x)
  ),
)
于 2016-03-03T10:44:13.197 回答