R'de trycatch nasıl yazılır


342

trycatchWeb'den indirme hatasıyla başa çıkmak için kod yazmak istiyorum .

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)

Bu iki ifade başarıyla çalışır. Aşağıda, var olmayan bir web adresi oluşturuyorum:

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

url[1]mevcut değil. Bir trycatchdöngü (işlev) nasıl yazılır, böylece:

  1. URL yanlış olduğunda, çıktı şöyle olur: "web URL'si yanlış, alınamıyor".
  2. URL yanlış olduğunda kod durmaz, ancak URL listesinin sonuna kadar indirilmeye devam eder mi?

Yanıtlar:


626

Peki o zaman: R dünyasına hoş geldiniz ;-)

Hadi bakalım

Kodun ayarlanması

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)
}

Kodu uygulama

> 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

Çıktıyı inceleme

> 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

Ek açıklamalar

tryCatch

tryCatchexprbir hata veya uyarı olmadığı sürece yürütmeyle ilişkili değeri döndürür . Bu durumda, belirli bir dönüş değerleri (bkz return(NA)yukarıda) karşılık gelen bir işleyici işlevi (bağımsız değişkenler bkz sağlayarak belirlenebilir errorve warningiçinde ?tryCatch). Bunlar zaten var olan işlevler olabilir, ancak bunları tryCatch()(yukarıda yaptığım gibi) içinde de tanımlayabilirsiniz .

İşleyici işlevlerinin belirli dönüş değerlerini seçmenin sonuçları

Biz o belirttiğiniz gibi NAhata durumunda iade edilmelidir, üçüncü unsur yolduğunu NA. Biz seçmiş olsaydın NULLdönüş değeri, uzunluğunu olmak ysadece olurdu 2yerine 3olarak lapply()sade dönüş değerleri "yok saymak" olacaktır NULL. Ayrıca , üzerinden açık bir dönüş değeri belirtmezseniz return(), işleyici işlevlerinin geri döneceğini unutmayın NULL(örn. Bir hata veya uyarı durumunda).

"İstenmeyen" uyarı mesajı

As warn=FALSEkullanımına herhangi bir etkisi, (bu durumda gerçekten ilgi değil) uyarıyı bastırmak için alternatif bir yolu var gibi görünmüyor

suppressWarnings(readLines(con=url))

onun yerine

readLines(con=url, warn=FALSE)

Birden çok ifade

Ayrıca (argüman "fiili ifadeler parçası" birden ifadeleri koyabilmesi Not exprait tryCatch()(Ben gösterildiği gibi süslü parantez koymayı varsa) finallykısmı).


pasteFonksiyonlarınızdaki ilk dizenin bir boşlukla bittiği göz önüne alındığında , neden boşluk ve sep=""?
seancarmody

2
@seancarmody: true ;-) Daha uzun / daha karmaşık dizeleri bir araya getirmeye çok alışkınım, boşlukları gerçekten yazarak kontrol etmek zorunda kalıyordum.
Rappster

3
Bunun paste0için kullanmalısın !
seancarmody

6
paste0() bir tabanı bulunmaktadır. Dahili olarak, hem paste()ve paste0()çağrı do_pasteiçinde paste.c . Tek fark, paste0()bir separgümanı geçmemesidir .
jthetzel

1
@JulienNavarre: "deneme bölümü" her zaman son nesneyi (şu anda readLines(con=url, warn=FALSE)yanlış gidebilecek gerçek şey) döndürür unutmayın . Bir ileti eklemek istiyorsa Yani, bir değişken saklanan gerçek retun değerine gerekir ediyorum: out <- readLines(con=url, warn=FALSE)takiben message("Everything worked")takip göre outaslında döndürülen bu son nesneyi yapmak için
Rappster

69

R, try-catch bloğu uygulamak için fonksiyonları kullanır:

Sözdizimi bir şekilde şöyle görünür:

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

TryCatch () yönteminde işlenebilecek iki 'koşul' vardır: 'uyarılar' ve 'hatalar'. Her bir kod bloğunu yazarken anlamanız gereken önemli şey, yürütme durumu ve kapsamdır. @kaynak


5
Değiştir error-handler-codeilecat("web url is wrong, can't get")
seancarmody

2
mesajlaşma dışarıda bıraktı
rawr

52

tryCatchbiraz karmaşık bir sözdizimi yapısına sahiptir. Bununla birlikte, aşağıda gösterildiği gibi eksiksiz bir tryCatch çağrısı oluşturan 4 bölümü anladıktan sonra hatırlamak kolaylaşır:

expr : [ Zorunlu ] Değerlendirilecek R kod (lar)

error : [ İsteğe bağlı ] İfadedeki kodlar değerlendirilirken bir hata oluşursa ne yapılmalı

uyarı : [ İsteğe bağlı ] İfadedeki kodları değerlendirirken bir uyarı oluşursa ne yapmalıdır?

son olarak : [ İsteğe bağlı ] İfade başarılı bir şekilde çalıştırılmışsa, bir hatayla veya bir uyarı ile olursa olsun, tryCatch çağrısından çıkmadan hemen önce ne çalıştırılmalıdır?

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...
    }
)

Bu nedenle, bir değerin günlüğünü hesaplamak için bir oyuncak örneği şöyle görünebilir:

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.')
        }
    )    
}

Şimdi, üç vaka çalışıyor:

Geçerli bir dava

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

"Uyarı" davası

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

Bir "hata" vakası

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

Düzenli kullandığım bazı yararlı kullanım durumları hakkında yazdım. Daha fazla ayrıntıyı burada bulabilirsiniz: bilgiyi https://rsangole.netlify.com/post/try-catch/

Umarım bu yardımcı olur.


34

İşte basit bir örnek :

# 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)
        }
    )
}

Ayrıca bir "uyarı" yakalamak istiyorsanız, sadece parçaya warning=benzer ekleyin error=.


1
Parçanın etrafında kıvrık parantez olmalı mı expr, çünkü bir yerine iki çizgi var mı?
Paul

Teşekkürler! Çift kontrol ettikten sonra, kıvrımlı parantezlere ihtiyaç duymuyorum
Paul

Çift kontrol için teşekkürler. Kodunu çalıştırdığımda, var Error: unexpected ')' in " )"ve Error: unexpected ')' in " )". Bir çift kıvrık parantez eklemek sorunu çözer.
Paul

Çoğu kullanım durumunda haklısınız, teşekkürler! Düzeltildi.
Paul

23

Bir irr işlevi için tryCatch için çözmeye çalışırken hayatımın iki gününü kaybettiğim için, bilgeliğimi (ve eksik olanı) paylaşmam gerektiğini düşündüm. FYI - irr, büyük bir veri setinde birkaç durumda hataların olduğu bu durumda FinCal'ın gerçek bir işlevidir.

  1. TryCatch işlevini bir işlevin parçası olarak ayarlayın. Örneğin:

    irr2 <- function (x) {
      out <- tryCatch(irr(x), error = function(e) NULL)
      return(out)
    }
  2. Hatanın (veya uyarının) çalışması için aslında bir işlev oluşturmanız gerekir. Başlangıçta hata kısmı için sadece yazdım error = return(NULL)ve TÜM değerler null geri geldi.

  3. Bir alt çıktı (benim "çıkışım" gibi) oluşturmayı unutmayın return(out).


3
Neden 3 numara gerekli?
jan-glx
Sitemizi kullandığınızda şunları okuyup anladığınızı kabul etmiş olursunuz: Çerez Politikası ve Gizlilik Politikası.
Licensed under cc by-sa 3.0 with attribution required.