- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我围绕 ftable
编写了一个包装器,因为我需要为许多变量计算具有频率和百分比的平面表。由于类“公式”的 ftable
方法使用非标准评估,包装器依赖于 do.call
和 match.call
来允许使用ftable
的 subset
参数(更多细节在 my previous question 中)。
mytable <- function(...) {
do.call(what = ftable,
args = as.list(x = match.call()[-1]))
# etc
}
但是,我不能将此包装器与 lapply
或 with
一起使用:
# example 1: error with "lapply"
lapply(X = warpbreaks[c("breaks",
"wool",
"tension")],
FUN = mytable,
row.vars = 1)
Error in (function (x, ...) : object 'X' not found
# example 2: error with "with"
with(data = warpbreaks[warpbreaks$tension == "L", ],
expr = mytable(wool))
Error in (function (x, ...) : object 'wool' not found
这些错误似乎是由于 match.call
没有在正确的环境中进行评估。
由于此问题与 my previous one 密切相关,这里是我的问题的总结:
do.call
和 match.call
的包装器不能与 lapply
或 with
一起使用。do.call
和match.call
的包装器不能使用ftable
的subset
参数。总结一下我的问题:
ftable
的 subset
参数,又可以与 lapply
和 一起使用与
?我想避免使用 lapply
和 with
,但我希望了解并纠正这些错误以提高我对 R 的了解。lapply
的错误是否与来自 ?lapply
的以下注释有关?
For historical reasons, the calls created by lapply are unevaluated, and code has been written (e.g., bquote) that relies on this. This means that the recorded call is always of the form FUN(X[[i]], ...), with i replaced by the current (integer or double) index. This is not normally a problem, but it can be if FUN uses sys.call or match.call or if it is a primitive function that makes use of the call. This means that it is often safer to call primitive functions with a wrapper, so that e.g. lapply(ll, function(x) is.numeric(x)) is required to ensure that method dispatch for is.numeric occurs correctly.
最佳答案
将 match.call
与 lapply
一起使用的问题在于 match.call
返回的是 literal 调用传递进去,没有任何解释。为了看看发生了什么,让我们创建一个更简单的函数,它可以准确显示您的函数如何解释传递给它的参数:
match_call_fun <- function(...) {
call = as.list(match.call()[-1])
print(call)
}
当我们直接调用它时,match.call
正确地获取参数并将它们放入我们可以与 do.call
一起使用的列表中:
match_call_fun(iris['Species'], 9)
[[1]]
iris["Species"]
[[2]]
[1] 9
但请注意当我们使用 lapply
时会发生什么(我只包含了内部 print
语句的输出):
lapply('Species', function(x) match_call_fun(iris[x], 9))
[[1]]
iris[x]
[[2]]
[1] 9
由于 match.call
获取传递给它的literal 参数,它接收 iris[x]
,而不是正确解释的 我们想要的 iris['Species']
。当我们使用 do.call
将这些参数传递给 ftable
时,它会在当前环境中查找对象 x
,然后返回一个错误它找不到它。我们需要解释
如您所见,添加 envir = parent.frame()
可以解决问题。这是因为,添加该参数会告诉 do.call
评估父框架中的 iris[x]
,这是 lapply
中的匿名函数其中 x
具有正确的含义。为了解实际效果,让我们创建另一个简单函数,该函数使用 do.call
从 3 个不同的环境级别打印 ls
:
z <- function(...) {
print(do.call(ls, list()))
print(do.call(ls, list(), envir = parent.frame()))
print(do.call(ls, list(), envir = parent.frame(2)))
}
当我们从全局环境调用 z()
时,我们看到函数内部的空环境,然后是全局环境:
z()
character(0) # Interior function environment
[1] "match_call_fun" "y" "z" # GlobalEnv
[1] "match_call_fun" "y" "z" # GlobalEnv
但是当我们从lapply
内部调用时,我们看到parent.frame
上一层是lapply
中的匿名函数:
lapply(1, z)
character(0) # Interior function environment
[1] "FUN" "i" "X" # lapply
[1] "match_call_fun" "y" "z" # GlobalEnv
因此,通过添加 envir = parent.frame()
,do.call
知道在 中评估
环境,它知道 iris[x]
>lapplyx
实际上是 'Species'
,并且正确评估。
mytable_envir <- function(...) {
tab <- do.call(what = ftable,
args = as.list(match.call()[-1]),
envir = parent.frame())
prop <- prop.table(x = tab,
margin = 2) * 100
bind <- cbind(as.matrix(x = tab),
as.matrix(x = prop))
margin <- addmargins(A = bind,
margin = 1)
round(x = margin,
digits = 1)
}
# This works!
lapply(X = c("breaks","wool","tension"),
FUN = function(x) mytable_envir(warpbreaks[x],row.vars = 1))
至于为什么添加 envir = parent.frame()
会有所不同,因为这似乎是默认选项。我不是 100% 确定,但我的猜测是,当使用默认参数时,parent.frame
会在 do.call
中计算> 函数,返回运行 do.call
的环境。然而,我们正在做的是调用 parent.frame
outside do.call
,这意味着它返回比默认版本高一级.
这是一个将 parent.frame()
作为默认值的测试函数:
fun <- function(y=parent.frame()) {
print(y)
print(parent.frame())
print(parent.frame(2))
print(parent.frame(3))
}
现在看看当我们从 lapply
中调用它时会发生什么,无论是否传入 parent.frame()
作为参数:
lapply(1, function(y) fun())
<environment: 0x12c5bc1b0> # y argument
<environment: 0x12c5bc1b0> # parent.frame called inside
<environment: 0x12c5bc760> # 1 level up = lapply
<environment: R_GlobalEnv> # 2 levels up = globalEnv
lapply(1, function(y) fun(y = parent.frame()))
<environment: 0x104931358> # y argument
<environment: 0x104930da8> # parent.frame called inside
<environment: 0x104931358> # 1 level up = lapply
<environment: R_GlobalEnv> # 2 levels up = globalEnv
在第一个示例中,y
的值与您在函数内部调用 parent.frame()
时获得的值相同。在第二个示例中,y
的值与上一级环境(在 lapply
内)相同。因此,虽然它们看起来相同,但实际上它们在做不同的事情:在第一个示例中,parent.frame
在发现没有 y=< 时在函数内部进行计算
参数,在第二个参数中,parent.frame
在 lapply
匿名函数 first 中求值,然后调用 fun
,然后传入其中。
关于r - 在 R 中使用 lapply 或 with 的用户定义函数中的非标准评估,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55813859/
我正在从 Stata 迁移到 R(plm 包),以便进行面板模型计量经济学。在 Stata 中,面板模型(例如随机效应)通常报告组内、组间和整体 R 平方。 I have found plm 随机效应
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 想改进这个问题?将问题更新为 on-topic对于堆栈溢出。 6年前关闭。 Improve this qu
我想要求用户输入整数值列表。用户可以输入单个值或一组多个值,如 1 2 3(spcae 或逗号分隔)然后使用输入的数据进行进一步计算。 我正在使用下面的代码 EXP <- as.integer(rea
当 R 使用分类变量执行回归时,它实际上是虚拟编码。也就是说,省略了一个级别作为基础或引用,并且回归公式包括所有其他级别的虚拟变量。但是,R 选择了哪一个作为引用,以及我如何影响这个选择? 具有四个级
这个问题基本上是我之前问过的问题的延伸:How to only print (adjusted) R-squared of regression model? 我想建立一个线性回归模型来预测具有 15
我在一台安装了多个软件包的 Linux 计算机上安装了 R。现在我正在另一台 Linux 计算机上设置 R。从他们的存储库安装 R 很容易,但我将不得不使用 安装许多包 install.package
我正在阅读 Hadley 的高级 R 编程,当它讨论字符的内存大小时,它说: R has a global string pool. This means that each unique strin
我们可以将 Shiny 代码写在两个单独的文件中,"ui.R"和 "server.R" , 或者我们可以将两个模块写入一个文件 "app.R"并调用函数shinyApp() 这两种方法中的任何一种在性
我正在使用 R 通过 RGP 包进行遗传编程。环境创造了解决问题的功能。我想将这些函数保存在它们自己的 .R 源文件中。我这辈子都想不通怎么办。我尝试过的一种方法是: bf_str = print(b
假设我创建了一个函数“function.r”,在编辑该函数后我必须通过 source('function.r') 重新加载到我的全局环境中。无论如何,每次我进行编辑时,我是否可以避免将其重新加载到我的
例如,test.R 是一个单行文件: $ cat test.R # print('Hello, world!') 我们可以通过Rscript test.R 或R CMD BATCH test.R 来
我知道我可以使用 Rmd 来构建包插图,但想知道是否可以更具体地使用 R Notebooks 来制作包插图。如果是这样,我需要将 R Notebooks 编写为包小插图有什么不同吗?我正在使用最新版本
我正在考虑使用 R 包的共享库进行 R 的站点安装。 多台计算机将访问该库,以便每个人共享相同的设置。 问题是我注意到有时您无法更新包,因为另一个 R 实例正在锁定库。我不能要求每个人都关闭它的 R
我知道如何从命令行启动 R 并执行表达式(例如, R -e 'print("hello")' )或从文件中获取输入(例如, R -f filename.r )。但是,在这两种情况下,R 都会运行文件中
我正在尝试使我当前的项目可重现,因此我正在创建一个主文档(最终是一个 .rmd 文件),用于调用和执行其他几个文档。这样我自己和其他调查员只需要打开和运行一个文件。 当前设置分为三层:主文件、2 个读
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 想改进这个问题?将问题更新为 on-topic对于堆栈溢出。 5年前关闭。 Improve this qu
我的 R 包中有以下描述文件 Package: blah Title: What the Package Does (one line, title case) Version: 0.0.0.9000
有没有办法更有效地编写以下语句?accel 是一个数据框。 accel[[2]]<- accel[[2]]-weighted.mean(accel[[2]]) accel[[3]]<- accel[[
例如,在尝试安装 R 包时 curl作为 usethis 的依赖项: * installing *source* package ‘curl’ ... ** package ‘curl’ succes
我想将一些软件作为一个包共享,但我的一些脚本似乎并不能很自然地作为函数运行。例如,考虑以下代码块,其中“raw.df”是一个包含离散和连续类型变量的数据框。函数“count.unique”和“squa
我是一名优秀的程序员,十分优秀!