带有 lapply 的内部 S3 泛型

Jus*_*dis 5 oop r lapply

我有一个 S3 通用函数,我希望将其作为包的内部部分。如果可能的话我宁愿不导出它。这样做的一个有趣的缺点是似乎lapply无法找到或使用正确的 S3 方法。有谁知道这种行为背后的原因?下面是一个可重现的示例,其中涉及从我的 github 安装虚拟包。

在这种情况下,通用函数 is ,如果对象是,docheck则返回,否则返回。TRUE"foo"FALSE

library(remotes)
install_github("jtlandis/SOExample")
#> Skipping install of 'SOExample' from a github remote, the SHA1 (27ab918c) has not changed since last install.
#>   Use `force = TRUE` to force installation
library(SOExample)
Run Code Online (Sandbox Code Playgroud)

foo 函数只是一个NA带有类的对象"foo"

foo
#> function () 
#> structure(NA, class = "foo")
#> <bytecode: 0x0000000015033878>
#> <environment: namespace:SOExample>
Run Code Online (Sandbox Code Playgroud)

fun_success()仅显示内部(未导出)S3 功能起作用的情况。

fun_success
#> function (x) 
#> {
#>     docheck(x)
#> }
#> <bytecode: 0x0000000013f83100>
#> <environment: namespace:SOExample>

fun_success(foo())
#> [1] TRUE
Run Code Online (Sandbox Code Playgroud)

fun_failure()调用相同的内部泛型方法,但失败

fun_failure
#> function (x) 
#> {
#>     lapply(list(x), docheck)
#> }
#> <bytecode: 0x0000000015e68df0>
#> <environment: namespace:SOExample>

fun_failure(foo())
#> Error in UseMethod("docheck"): no applicable method for 'docheck' applied to an object of class "foo"
Run Code Online (Sandbox Code Playgroud)

由reprex 包(v2.0.0)于 2021-07-24 创建

MrF*_*ick 5

R 对于确保正确注册 S3 方法变得越来越挑剔。注册包泛型方法的常规方法是使用 NAMESPACE 文件中的 S3method() 。但如果您试图保留所有docheck未导出的内容,那么您需要自己注册 S3 方法。您可以使用该.S3method()功能来做到这一点。在您的 R 文件中,添加此行

# Existing line...
# docheck.foo <- function(x) T
# New line...
.S3method("docheck", "foo", docheck.foo)
Run Code Online (Sandbox Code Playgroud)

这将允许lapply找到该方法。