如何使用 tryCatch() 函数?

How to use the tryCatch() function?

提问人:Dd Pp 提问时间:8/30/2012 最后编辑:Konrad RudolphDd Pp 更新时间:11/14/2023 访问量:536236

问:

我想编写代码来处理从网络下载数据的错误。tryCatch

url <- c(
    "http://stat.ethz.ch/R-manual/R-devel/library/base/html/connections.html",
    "http://en.wikipedia.org/wiki/Xz")
y <- mapply(readLines, con=url)

这两个语句成功运行。下面,我创建一个不存在的网址:

url <- c("xxxxx", "http://en.wikipedia.org/wiki/Xz")

url[1]不存在。如何编写一个循环(函数),以便:tryCatch

  1. 当 URL 错误时,输出将是:“Web URL 错误,无法获取”。
  2. 当 URL 错误时,代码不会停止,而是继续下载,直到 URL 列表的末尾?
异常 try-catch r-faq

评论


答:

88赞 heretolearn 8/30/2012 #1

R 使用函数来实现 try-catch 块:

语法有点像这样:

result = tryCatch({
    expr
}, warning = function(warning_condition) {
    warning-handler-code
}, error = function(error_condition) {
    error-handler-code
}, finally={
    cleanup-code
})

在 tryCatch() 中,有两个可以处理的“条件”:“警告”和“错误”。编写每个代码块时要了解的重要事项是执行状态和范围。@source

评论

6赞 seancarmody 8/30/2012
替换为error-handler-codecat("web url is wrong, can't get")
3赞 rawr 12/16/2015
您遗漏了消息捕获
915赞 15 revs, 3 users 73%Rappster #2

设置代码

urls <- c(
    "http://stat.ethz.ch/R-manual/R-devel/library/base/html/connections.html",
    "http://en.wikipedia.org/wiki/Xz",
    "xxxxx"
)

readUrl <- function(url) {
    tryCatch(
        {
            # Just to highlight: if you want to use more than one
            # R expression in the "try" part then you'll have to
            # use curly brackets.
            # 'tryCatch()' will return the last evaluated expression
            # in case the "try" part was completed successfully

            message("This is the 'try' part")

            suppressWarnings(readLines(url))
            # The return value of `readLines()` is the actual value
            # that will be returned in case there is no condition
            # (e.g. warning or error).
        },
        error = function(cond) {
            message(paste("URL does not seem to exist:", url))
            message("Here's the original error message:")
            message(conditionMessage(cond))
            # Choose a return value in case of error
            NA
        },
        warning = function(cond) {
            message(paste("URL caused a warning:", url))
            message("Here's the original warning message:")
            message(conditionMessage(cond))
            # Choose a return value in case of warning
            NULL
        },
        finally = {
            # NOTE:
            # Here goes everything that should be executed at the end,
            # regardless of success or error.
            # If you want more than one expression to be executed, then you
            # need to wrap them in curly brackets ({...}); otherwise you could
            # just have written 'finally = <expression>' 
            message(paste("Processed URL:", url))
            message("Some other message at the end")
        }
    )
}

使用代码

> y <- lapply(urls, readUrl)
This is the 'try' part
Processed URL: http://stat.ethz.ch/R-manual/R-devel/library/base/html/connections.html
Some other message at the end
This is the 'try' part
Processed URL: http://en.wikipedia.org/wiki/Xz
Some other message at the end
This is the 'try' part
URL does not seem to exist: xxxxx
Here's the original error message:
cannot open the connection
Processed URL: xxxxx
Some other message at the end

调查输出

> head(y[[1]])
[1] "<!DOCTYPE html><html><head><title>R: Functions to Manipulate Connections (Files, URLs, ...)</title>"
[2] "<meta http-equiv=\"Content-Type\" content=\"text/html; charset=utf-8\" />"
[3] "<meta name=\"viewport\" content=\"width=device-width, initial-scale=1.0, user-scalable=yes\" />"
[4] "<link rel=\"stylesheet\" href=\"https://cdn.jsdelivr.net/npm/[email protected]/dist/katex.min.css\">"
[5] "<script type=\"text/javascript\">"
[6] "const macros = { \"\\\\R\": \"\\\\textsf{R}\", \"\\\\code\": \"\\\\texttt\"};"

> length(y)
[1] 3

> y[[3]]
[1] NA

补充说明

tryCatch

tryCatch返回与执行关联的值,除非出现错误或警告。在这种情况下,可以通过提供相应的处理函数来指定特定的返回值(见上文)。这些可以是已经存在的函数,但您也可以在其中定义它们(就像我上面所做的那样)。exprNAerrorwarning?tryCatchtryCatch()

选择处理程序函数的特定返回值的含义

正如我们指定的那样,在出现错误时应返回,因此 中的第三个元素是 .NAyNA

评论

4赞 seancarmody 8/30/2012
你应该为此使用!paste0
8赞 jthetzel 8/30/2012
paste0() 基地。在内部,两者都调用了 paste.c。唯一的区别是不传递参数。paste()paste0()do_pastepaste0()sep
0赞 Rappster 8/30/2012
@jthetzel + @seancarmody:对不起,R-2.15.1 也是如此。我正在运行过时的 R 版本 (2.14.1),当时没有这样的函数。非常好,恕我直言,这应该是从一开始就默认的行为paste
2赞 Rappster 4/23/2015
@JulienNavarre:请记住,“try 部分”总是返回最后一个对象(当前是可能出错的实际对象)。因此,如果要添加消息,则需要将实际的 retun 值存储在变量中:后跟 后跟,以使其成为实际返回的最后一个对象readLines(con=url, warn=FALSE)out <- readLines(con=url, warn=FALSE)message("Everything worked")out
1赞 Lazarus Thurston 1/31/2020
很棒的例子,有据可查。我可以要求删除所有关于无聊问题的评论者,例如 / ,这样我们就不会用不相关的东西挤满这一部分吗?谢谢。pastepaste0
33赞 James 9/26/2016 #3

由于我刚刚浪费了两天的时间试图解决 irr 函数的 tryCatch,我想我应该分享我的智慧(以及缺少什么)。仅供参考 - irr 是 FinCal 的一个实际函数,在这种情况下,在大型数据集上出现错误。

  1. 将 tryCatch 设置为函数的一部分。例如:

    irr2 <- function (x) {
      out <- tryCatch(irr(x), error = function(e) NULL)
      return(out)
    }
    
  2. 要使错误(或警告)起作用,您实际上需要创建一个函数。我最初只是写了错误部分,所有值都返回 null。error = return(NULL)

  3. 请记住创建一个子输出(例如我的“out”)和 .return(out)

评论

5赞 jan-glx 12/21/2017
为什么需要数字 3?
69赞 Paul 4/13/2017 #4

下面是一个简单的例子

# Do something, or tell me why it failed
my_update_function <- function(x){
    tryCatch(
        # This is what I want to do...
        {
        y = x * 2
        return(y)
        },
        # ... but if an error occurs, tell me what happened: 
        error=function(error_message) {
            message("This is my custom message.")
            message("And below is the error message from R:")
            message(error_message)
            return(NA)
        }
    )
}

如果您还想捕获“警告”,只需添加与该部分相似的内容即可。warning=error=

评论

1赞 Paul 8/15/2018
零件周围是否应该有大括号,因为有两条线而不是一条线?expr
0赞 Paul 8/16/2018
谢谢!经过仔细检查,我认为不需要大括号
0赞 Paul 8/16/2018
感谢您的仔细检查。当我运行你的代码时,我得到了 和 .添加一对大括号可以解决此问题。Error: unexpected ')' in " )"Error: unexpected ')' in " )"
1赞 Paul 11/11/2018
对于大多数用例,你是对的,谢谢!它已被修复。
182赞 Rahul 12/21/2018 #5

tryCatch具有略微复杂的语法结构。但是,一旦我们理解了构成完整 tryCatch 调用的 4 个部分,如下所示,就很容易记住:

expr:[必需] 要评估的 R 代码

error : [可选] 如果在 expr 中评估代码时发生错误,应该运行什么

warning : [可选] 如果在 expr 中评估代码时出现警告,应运行什么

finally : [可选] 在退出 tryCatch 调用之前应该运行什么,无论 expr 是否成功运行、出现错误或出现警告

tryCatch(
    expr = {
        # Your code...
        # goes here...
        # ...
    },
    error = function(e){ 
        # (Optional)
        # Do this if an error is caught...
    },
    warning = function(w){
        # (Optional)
        # Do this if a warning is caught...
    },
    finally = {
        # (Optional)
        # Do this at the end before quitting the tryCatch structure...
    }
)

因此,一个玩具示例,用于计算值的对数可能如下所示:

log_calculator <- function(x){
    tryCatch(
        expr = {
            message(log(x))
            message("Successfully executed the log(x) call.")
        },
        error = function(e){
            message('Caught an error!')
            print(e)
        },
        warning = function(w){
            message('Caught an warning!')
            print(w)
        },
        finally = {
            message('All done, quitting.')
        }
    )    
}

现在,运行三个案例:

有效案例

log_calculator(10)
# 2.30258509299405
# Successfully executed the log(x) call.
# All done, quitting.

“警告”案例

log_calculator(-10)
# Caught an warning!
# <simpleWarning in log(x): NaNs produced>
# All done, quitting.

“错误”案例

log_calculator("log_me")
# Caught an error!
# <simpleError in log(x): non-numeric argument to mathematical function>
# All done, quitting.

我写过一些我经常使用的有用用例。在此处查找更多详细信息:将 tryCatch 用于可靠的 R 脚本

希望这对您有所帮助。

评论

0赞 Julien 3/17/2023
如何使返回值?log_calculator
2赞 Michael Dewar 4/10/2023
@Julien,如果没有错误,则返回最后一行。expr
0赞 H. berg 4/25/2023
当表达式正确执行时,是否也可以有不同于 NULL 的返回值?
3赞 Gregor Thomas 5/1/2023 #6

该软件包提供了比 更易于设置的替代功能。从文档中,它们被描述为:purrrtryCatch?safely

  • safely:wrapped 函数返回一个带有组件和 .如果发生错误,则为对象并具有默认值 ()。否则错误是 。listresulterrorerrorerrorresultotherwiseNULL

  • quietly:包装函数返回一个带有组件、 和 的函数。listresultoutputmessageswarnings

  • possibly:每当发生错误时,wrapped 函数使用默认值 ()。otherwise

请注意,与 不同,这些函数应包装函数,而不是表达式,并且它们返回修改后的函数。 对于如前所述的 OP 问题,我们可能会直接使用 and wrap 来修改它。tryCatch()possiblyreadLines

url <- c(
    "http://stat.ethz.ch/R-manual/R-devel/library/base/html/connections.html",
    "http://en.wikipedia.org/wiki/Xz", 
    "xxx")

library(purrr)
lapply(url, possibly(readLines, otherwise = "web URL is wrong, can't get"))
## with possibly, the error prints as a warning
## and the final value is the `otherwise` string

但也要注意,我们可以创建一个修改后的版本 ,例如可以在我们代码中的多个地方使用。readLinesmy_readLines <- possibly(readLines, otherwise = "web URL is wrong, can't get")

我在上面进行了说明,但我们可以很容易地想象出我们想要使用的情况(之后我们可以从每个列表项中提取组件,可能跳过或以其他方式处理具有非空组件的项目,甚至可能根据错误以不同的方式处理它们),或者还单独捕获警告和消息。possiblysafely()resulterrorquietly