A więc: witaj w świecie R. ;-)
Proszę bardzo
Konfigurowanie kodu
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)
}
Stosowanie kodu
> 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
Badanie produkcji
> 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
Dodatkowe uwagi
próbuj złapać
tryCatch
zwraca wartość związaną z wykonaniem, expr
chyba że wystąpi błąd lub ostrzeżenie. W takim przypadku określone wartości zwracane (patrz return(NA)
wyżej) można określić, podając odpowiednią funkcję modułu obsługi (patrz argumenty error
i warning
in ?tryCatch
). Mogą to być funkcje, które już istnieją, ale możesz je również zdefiniować w sobie tryCatch()
(tak jak to zrobiłem powyżej).
Implikacje wyboru określonych wartości zwracanych przez funkcje obsługi
Jak ustaliliśmy, że NA
powinien zostać zwrócony w przypadku błędu, trzecim elementem y
jest NA
. Gdybyśmy wybrali NULL
wartość zwracaną, długość y
byłaby po prostu 2
zamiast, 3
ponieważ lapply()
po prostu „ignoruje” zwracane wartości, które są NULL
. Zauważ również, że jeśli nie określisz wyraźnej wartości zwracanej przez return()
, funkcje modułu obsługi zwrócą NULL
(tj. W przypadku błędu lub warunku ostrzegawczego).
Komunikat ostrzegawczy „niepożądany”
Ponieważ warn=FALSE
wydaje się, że nie ma to żadnego efektu, alternatywnym sposobem tłumienia ostrzeżenia (które w tym przypadku tak naprawdę nie jest interesujące) jest użycie
suppressWarnings(readLines(con=url))
zamiast
readLines(con=url, warn=FALSE)
Wiele wyrażeń
Pamiętaj, że możesz także umieścić wiele wyrażeń w „rzeczywiste wyrażeń części” (teza expr
o tryCatch()
) jeśli zawinąć je w nawiasach klamrowych (podobnie jak ja pokazano w finally
części).
paste
funkcjach kończy się spacją, dlaczego nie pominąć spacji isep=""
?paste0
do tego!paste0()
jest w bazie. Wewnętrznie zarównopaste()
ipaste0()
wywołajdo_paste
w paste.c . Jedyną różnicą jest to,paste0()
że nie przekazujesep
argumentu.readLines(con=url, warn=FALSE)
to, co może się nie powieść ). Tak więc, jeśli chcesz dodać wiadomość, musisz zapisać aktualną wartość retunu w zmiennej:out <- readLines(con=url, warn=FALSE)
po niejmessage("Everything worked")
następujeout
, aby uczynić to ostatnim zwracanym obiektemR używa funkcji do implementacji bloku try-catch:
Składnia wygląda mniej więcej tak:
W tryCatch () można spełnić dwa „warunki”: „ostrzeżenia” i „błędy”. Ważną rzeczą do zrozumienia podczas pisania każdego bloku kodu jest stan wykonania i zakres. @źródło
źródło
error-handler-code
zcat("web url is wrong, can't get")
tryCatch
ma nieco złożoną strukturę składni. Jednak gdy zrozumiemy 4 części, które stanowią kompletne wywołanie tryCatch, jak pokazano poniżej, łatwo jest zapamiętać:wyrażenie : [ wymagane ] R kod (y) do oceny
error : [ Opcjonalnie ] Co powinno się uruchomić, jeśli wystąpił błąd podczas oceny kodów w wyrażeniu
ostrzeżenie : [ Opcjonalnie ] Co powinno się uruchomić, jeśli wystąpi ostrzeżenie podczas oceny kodów w wyrażeniu
na koniec : [ Opcjonalnie ] Co powinno działać tuż przed zakończeniem wywołania tryCatch, niezależnie od tego, czy wyrażenie uruchomione pomyślnie, z błędem czy z ostrzeżeniem
Przykład zabawkowy do obliczenia dziennika wartości może wyglądać następująco:
Teraz uruchamiam trzy sprawy:
Ważna sprawa
Przypadek „ostrzegawczy”
Przypadek „błędu”
Pisałem o kilku użytecznych przypadkach, z których regularnie korzystam. Znajdź więcej szczegółów tutaj: https://rsangole.netlify.com/post/try-catch/
Mam nadzieję, że to jest pomocne.
źródło
Oto prosty przykład :
Jeśli chcesz również uchwycić „ostrzeżenie”, po prostu dodaj
warning=
podobny doerror=
części.źródło
expr
części powinny znajdować się nawiasy klamrowe , ponieważ zamiast jednej są dwie linie?Error: unexpected ')' in " )"
iError: unexpected ')' in " )"
. Dodanie pary nawiasów klamrowych rozwiązuje problem.Ponieważ właśnie straciłem dwa dni mojego życia, próbując rozwiązać problemy związane z funkcją tryCatch, pomyślałem, że powinienem podzielić się swoją mądrością (i tym, czego brakuje). FYI - irr jest faktyczną funkcją FinCal w tym przypadku, w której wystąpiły błędy w kilku przypadkach na dużym zbiorze danych.
Skonfiguruj tryCatch jako część funkcji. Na przykład:
Aby błąd (lub ostrzeżenie) zadziałał, faktycznie musisz utworzyć funkcję. Pierwotnie z powodu błędu właśnie napisałem,
error = return(NULL)
a WSZYSTKIE wartości wróciły do zera.Pamiętaj, aby utworzyć pod-wyjście (jak mój „out”) i
return(out)
.źródło