कैसे आर में trycatch लिखने के लिए


342

मैं 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 गलत है, तो आउटपुट होगा: "वेब URL गलत है, नहीं मिल सकता है"।
  2. जब URL गलत है, तो कोड बंद नहीं होता है, लेकिन URL की सूची के अंत तक डाउनलोड करना जारी रहता है?

जवाबों:


626

तो ठीक है: आर दुनिया में आपका स्वागत है ;-)

हेयर यू गो

कोड सेट करना

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) {
    out <- 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")

            readLines(con=url, warn=FALSE) 
            # The return value of `readLines()` is the actual value 
            # that will be returned in case there is no condition 
            # (e.g. warning or error). 
            # You don't need to state the return value via `return()` as code 
            # in the "try" part is not wrapped insided a function (unlike that
            # for the condition handlers for warnings and error below)
        },
        error=function(cond) {
            message(paste("URL does not seem to exist:", url))
            message("Here's the original error message:")
            message(cond)
            # Choose a return value in case of error
            return(NA)
        },
        warning=function(cond) {
            message(paste("URL caused a warning:", url))
            message("Here's the original warning message:")
            message(cond)
            # Choose a return value in case of warning
            return(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")
        }
    )    
    return(out)
}

कोड लागू करना

> y <- lapply(urls, readUrl)
Processed URL: http://stat.ethz.ch/R-manual/R-devel/library/base/html/connections.html
Some other message at the end
Processed URL: http://en.wikipedia.org/wiki/Xz
Some other message at the end
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
Warning message:
In file(con, "r") : cannot open file 'xxxxx': No such file or directory

आउटपुट की जांच

> head(y[[1]])
[1] "<!DOCTYPE html PUBLIC \"-//W3C//DTD HTML 4.01 Transitional//EN\">"      
[2] "<html><head><title>R: Functions to Manipulate Connections</title>"      
[3] "<meta http-equiv=\"Content-Type\" content=\"text/html; charset=utf-8\">"
[4] "<link rel=\"stylesheet\" type=\"text/css\" href=\"R.css\">"             
[5] "</head><body>"                                                          
[6] ""    

> length(y)
[1] 3

> y[[3]]
[1] NA

अतिरिक्त टिप्पणी

पकड़ने की कोशिश

tryCatchनिष्पादन से संबंधित मान लौटाता है exprजब तक कि कोई त्रुटि या चेतावनी न हो। इस मामले में, विशिष्ट वापसी मूल्यों (देखें return(NA)ऊपर) एक संबंधित हैंडलर समारोह (तर्कों को देखने की आपूर्ति करके निर्दिष्ट किया जा सकता errorहै और warningमें ?tryCatch)। ये ऐसे फ़ंक्शन हो सकते हैं जो पहले से मौजूद हैं, लेकिन आप उन्हें भीतर भी परिभाषित कर सकते हैं tryCatch()(जैसा मैंने ऊपर किया था)।

हैंडलर फ़ंक्शन के विशिष्ट रिटर्न मान चुनने के निहितार्थ

जैसा कि हमने निर्दिष्ट किया है कि NAत्रुटि के मामले में वापस आ जाना चाहिए, तीसरा तत्व yहै NA। हम चुना है चाहते हैं, तो NULLवापसी मान, की लंबाई होने के लिए yबस हो गया होता 2के बजाय 3के रूप में lapply()बस को "अनदेखा" होगा वापसी मान हैं कि NULL। यह भी ध्यान दें कि यदि आप के माध्यम से एक स्पष्ट वापसी मूल्य निर्दिष्ट नहीं करते हैं return(), तो हैंडलर फ़ंक्शन वापस आ जाएंगे NULL(यानी एक त्रुटि या चेतावनी की स्थिति में)।

"अनसेस्ड" चेतावनी संदेश

जैसा warn=FALSEकि कोई प्रभाव नहीं दिखता है, चेतावनी को दबाने का एक वैकल्पिक तरीका (जो इस मामले में वास्तव में रुचि नहीं है) का उपयोग करना है

suppressWarnings(readLines(con=url))

के बजाय

readLines(con=url, warn=FALSE)

एकाधिक भाव

ध्यान दें कि आप भी "वास्तविक भाव भाग" में एक से अधिक भाव (तर्क दे सकते हैं exprकी tryCatch()) यदि आप उन्हें कर्ली कोष्ठक में लपेट (जैसे मैं में सचित्र finallyभाग)।


यह देखते हुए कि आपके pasteफ़ंक्शंस में पहला स्ट्रिंग एक स्थान के साथ समाप्त होता है, क्यों नहीं अंतरिक्ष और को छोड़ दें sep=""?
सीनकार्मोडी

2
@ seancarmody: सच;; मैं सिर्फ इतना है कि मैं एक साथ लंबे समय तक / अधिक जटिल तार डाल रहा हूं, मुझे वास्तव में उन्हें लिखने के माध्यम से रिक्त स्थान को नियंत्रित करना होगा।
रैपरस्टर 12

3
आपको उसके paste0लिए उपयोग करना चाहिए !
13 अक्टूबर को seancarmody

6
paste0() है आधार में। आंतरिक रूप से, दोनों paste()और paste0()कॉल do_pasteमें paste.c । अंतर केवल paste0()एक sepतर्क से नहीं गुजरता है ।
jthetzel

1
@JulienNavarre: याद रखें कि "कोशिश वाला हिस्सा" हमेशा अंतिम वस्तु देता है (वर्तमान में readLines(con=url, warn=FALSE)जो वास्तविक चीज है जो गलत हो सकती है)। इसलिए यदि आप एक संदेश जोड़ना चाहते हैं, तो आपको वास्तविक रिटन मूल्य को एक चर में संग्रहीत करने की आवश्यकता होगी: out <- readLines(con=url, warn=FALSE)इसके message("Everything worked")बाद outइसे अंतिम वस्तु बनाने के लिए, जो वास्तव में वापस आ जाएगी
Rappster

69

आर-पकड़ने के ब्लॉक को लागू करने के लिए कार्यों का उपयोग करता है:

वाक्यविन्यास कुछ इस तरह दिखता है:

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

TryCatch () में दो 'स्थितियाँ' हैं जिन्हें संभाला जा सकता है: 'चेतावनियाँ' और 'त्रुटियाँ'। कोड के प्रत्येक ब्लॉक को लिखते समय समझने वाली महत्वपूर्ण बात निष्पादन की स्थिति और गुंजाइश है। @source


5
बदलें error-handler-codeके साथcat("web url is wrong, can't get")
seancarmody

2
आपने
21-37

52

tryCatchथोड़ी जटिल वाक्य रचना है। हालाँकि, जब हम 4 भागों को समझ लेते हैं, जो नीचे दिखाए गए अनुसार एक पूर्ण 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 an 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.

मैंने कुछ उपयोगी उपयोग-मामलों के बारे में लिखा है जो मैं नियमित रूप से उपयोग करता हूं। अधिक जानकारी यहां पाएं: https://rsangole.netlify.com/post/try-catch/

आशा है कि यह उपयोगी है।


34

यहाँ एक सीधा उदाहरण दिया गया है :

# 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
क्या exprभाग के चारों ओर घुंघराले कोष्ठक होने चाहिए , क्योंकि एक के बजाय दो रेखाएँ हैं?
पॉल

धन्यवाद! दोहरी जाँच के बाद, मुझे कर्ली कोष्ठक की कोई आवश्यकता नहीं
पॉल

दोहरी जाँच के लिए धन्यवाद। जब मैं अपने कोड चलाने के लिए, मुझे मिल गया Error: unexpected ')' in " )"और Error: unexpected ')' in " )"। घुंघराले ब्रैकेट की एक जोड़ी को जोड़ने से समस्या हल हो जाती है।
पॉल

अधिकांश उपयोग के मामलों के लिए, आप सही हैं, धन्यवाद! इसे ठीक कर लिया गया है।
पॉल

23

चूँकि मैंने अपने जीवन के दो दिन खो दिए थे, एक क्षोभ समारोह के लिए हल करने की कोशिश कर रहा था, मैंने सोचा कि मुझे अपना ज्ञान बांटना चाहिए (और क्या याद आ रहा है)। FYI करें - ir इस मामले में FinCal का एक वास्तविक कार्य है जहां एक बड़े डेटा सेट पर कुछ मामलों में त्रुटियां मिलीं।

  1. किसी फ़ंक्शन के भाग के रूप में tryCatch सेट करें। उदाहरण के लिए:

    irr2 <- function (x) {
      out <- tryCatch(irr(x), error = function(e) NULL)
      return(out)
    }
  2. कार्य करने के लिए त्रुटि (या चेतावनी) के लिए, आपको वास्तव में एक फ़ंक्शन बनाने की आवश्यकता है। मैंने मूल रूप से त्रुटि भाग के लिए लिखा था error = return(NULL)और सभी मान वापस आ गए थे।

  3. एक उप-आउटपुट बनाने के लिए याद रखें (जैसे मेरा "आउट") और return(out)


3
नंबर 3 क्यों जरूरी है?
-ग्लक्स
हमारी साइट का प्रयोग करके, आप स्वीकार करते हैं कि आपने हमारी Cookie Policy और निजता नीति को पढ़ और समझा लिया है।
Licensed under cc by-sa 3.0 with attribution required.