使用 roxygen2 记录 S4 方法:在帮助文件中明确区分方法

Documenting S4 methods with roxygen2: clearly distinguishing methods in help files

提问人:Rappster 提问时间:9/18/2012 最后编辑:CommunityRappster 更新时间:6/21/2014 访问量:822

问:

在瞄准和射击之前

我知道这个问题与这篇文章密切相关。事实上,我遵循了各种答案中给出的建议,但仍然觉得生成的帮助文件有点“尴尬”,或者至少看起来有点“混在一起”,因此我的问题

实际问题

  1. 在记录 S4 泛型及其各自的方法时,如何最终获得一个 Rd 文件结构,让用户清楚地区分各种方法,从而非常清楚地知道文档的哪一部分属于哪个方法?
  2. 鉴于我提供了标签,那么在尝试获取特定方法的帮助时,我该如何实际使用它们?比如不仅仅是打字?@alias?foo-character-method?foo
  3. 最好不要在泛型方法中记录 a) 签名参数,因为每个方法肯定会包含相应的文档和 b) 返回值,因为这也取决于各自的方法,这是一个好的做法吗?

现在,我觉得我最终得到了一个帮助文件,它似乎无法真正处理通用方法和各种方法之间的正确区分 - 至少在视觉上不是。

但也许我还没有完全明白这一点;-)。在这种情况下,任何指针将不胜感激!


下面是一个可重现的示例,它生成包含 S4 泛型和两个方法的包。我试图遵循这篇文章编写 R 扩展中给出的建议mypkg

确保目录

sapply(c("src", "package"), dir.create, showWarnings=FALSE)

泛型方法定义

gnrc.roxy <- c(
    "#' Doing Something Useful",
    "#'", 
    "#' Description here.",
    "#'", 
    "#' @param x A signature argument for method dispatch.",
    "#' @param y A signature argument for method dispatch.",
    "#' @param arg.1 A \\code{logical} scalar. If \\code{TRUE} (default)", 
    "#'      something happens; else not.",
    "#' @param arg.2 A \\code{numeric} vector containing some useful numbers.", 
    "#' @param ... Further arguments.",
    "#' @return Depends on the actual method.", 
    "#' @references \\url{http://www.rappster.de/}",
    "#' @author Janko Thyson \\email{janko.thyson@@rappster.de}",
    "#' @examples", 
    "#'   foo(x=\"a\", y=letters[1:5])",
    "#'   foo(x=\"a\", y=1:5)",
    "#' @docType methods",
    "#' @rdname foo-methods",
    "#' @export"
)    
gnrc.def <- c(
    "setGeneric(",
    "    name=\"foo\",",
    "    signature=c(\"x\", \"y\"),",
    "    def=function(",
            "x,",
            "y,",
            "arg.1=TRUE,",
            "arg.2,",
            "...",
    ") {",
    "standardGeneric(\"foo\")",       
    "})"
)
path <- "src/gnrc_foo.R"
write(gnrc.roxy, file=path)
write(gnrc.def, file=path, append=TRUE)

相关方法的定义

mthd1.roxy <- c(
    "#' @param x A \\code{character} scalar.",
    "#' @param y A \\code{character} vector.",
    "#' @return A \\code{character} vector.", 
    "#' @docType methods",
    "#' @rdname foo-methods",
    "#' @rdname foo-methods",
    "#' @aliases foo,character,character-method",
    "#' @export"
)    
mthd1.def <- c(
    "setMethod(",
        "f=\"foo\",",
        "signature=signature(x=\"character\", y=\"character\"),", 
        "definition=function(",
            "x,",
            "y,", 
            "...",
        ") {",
        "print(x);print(y);return(y)",
        "})"
)    
path <- "src/mthds_foo.R"
write(mthd1.roxy, file=path)
write(mthd1.def, file=path, append=TRUE)

mthd2.roxy <- c(
    "#' @param x A \\code{character} scalar.",
    "#' @param y A \\code{numeric} vector.",
    "#' @return A \\code{numeric} vector.", 
    "#' @docType methods",
    "#' @rdname foo-methods",
    "#' @rdname foo-methods",
    "#' @aliases foo,character,numeric-method",
    "#' @export"
)    
mthd2.def <- c(
    "setMethod(",
        "f=\"foo\",",
        "signature=signature(x=\"character\", y=\"numeric\"),", 
        "definition=function(",
            "x,",
            "y,", 
            "...",
        ") {",
        "print(x);print(y);return(y)",
        "})"
)    
write(mthd2.roxy, file=path, append=TRUE)
write(mthd2.def, file=path, append=TRUE)

# Test source to see if methods were defined correctly
#sapply(list.files("src", full.names=TRUE), source)

创建包框架

# Ensure empty package directory
subdirs <- list.files("package", full.names=TRUE)
if (length(subdirs)) {
    sapply(subdirs, unlink, recursive=TRUE)
}

pkg.name    <- "mypkg"
path        <- file.path("package", pkg.name)

package.skeleton(
    name=pkg.name, 
    path="package", 
    code_files=list.files("src", full.names=TRUE, recursive=TRUE)
)

Roxygenize

require("roxygen2")
roxygenize(package.dir=path)

修补 Rd 文件

有些东西需要修补,以便允许自动检查和构建过程,不需要任何手动干预,例如手动编辑 Rd 文件等。

rdfiles <- list.files(file.path(path, "man"), full.names=TRUE)

# 1) Removing unnecessary file './package/mypgk/man/foo.Rd'
file <- grep("foo.Rd", rdfiles, value=TRUE)
if (length(file)) {
    unlink(file)
}

# 2) Patching './mypkg/man/mypkg-package.Rd' to pass R CMD CHECK
file <- grep(paste(pkg.name, "-package.Rd", sep=""), rdfiles, value=TRUE)
if (length(file)) {
    cnt <- readLines(file, warn=FALSE)
    idx.0 <- grep("\\\\examples\\{", cnt)
    if (length(idx.0)) {
        idx.1 <- grep("\\}", cnt)
        idx.1 <- max(idx.1[which(idx.1 > idx.0)])
        exnew <- c(
            "\\examples{",
            "# TODO: add examples",
            "}"
        )
        cnt <- cnt[-(idx.0:idx.1)]
        cnt <- append(cnt, exnew, after=idx.0-1)
        write(cnt, file=file)
    }
}

检查包装

shell(paste("R CMD check", path), intern=FALSE)

生成并安装软件包

shell(paste("R CMD INSTALL",  path), intern=FALSE)

调查帮助文件

require("mypkg")

foo(x="a", y=letters)
foo(x="a", y=1:10)

?foo

替代方法:构建和安装分离

shell(paste("R CMD INSTALL --build --data-compress=gzip ",  path), intern=FALSE)

install.packages(paste0(pkg.name, "_1.0.zip"), type="win.binary")

require("mypkg")

?foo
方法 R-S4 Roxygen2

评论

1赞 Henrik 9/18/2012
我想你应该看看 Roxygen3 的开发版本。最近,哈德利在这方面做了很多工作。顺便说一句,好问题(+1)。

答: 暂无答案