Elixir-오류 처리
Elixir에는 오류, throw 및 종료의 세 가지 오류 메커니즘이 있습니다. 각 메커니즘을 자세히 살펴 보겠습니다.
오류
코드에서 예외가 발생할 때 오류 (또는 예외)가 사용됩니다. 문자열에 숫자를 추가하여 샘플 오류를 검색 할 수 있습니다.
IO.puts(1 + "Hello")
위의 프로그램이 실행되면 다음과 같은 오류가 발생합니다.
** (ArithmeticError) bad argument in arithmetic expression
:erlang.+(1, "Hello")
이것은 샘플 내장 오류입니다.
오류 발생
우리는 할 수 있습니다 raise올림 함수를 사용하는 오류. 같은 것을 이해하는 예를 고려해 보겠습니다.
#Runtime Error with just a message
raise "oops" # ** (RuntimeError) oops
raise / 2가 오류 이름과 키워드 인수 목록을 전달하면 다른 오류가 발생할 수 있습니다.
#Other error type with a message
raise ArgumentError, message: "invalid argument foo"
자신의 오류를 정의하고이를 제기 할 수도 있습니다. 다음 예를 고려하십시오-
defmodule MyError do
defexception message: "default message"
end
raise MyError # Raises error with default message
raise MyError, message: "custom message" # Raises error with custom message
구조 오류
우리는 프로그램이 갑작스럽게 종료되는 것이 아니라 오류를 신중하게 처리해야합니다. 이를 위해 우리는 오류 처리를 사용합니다. 우리rescue 사용하는 오류 try/rescue구성. 같은 것을 이해하기 위해 다음 예제를 고려해 보겠습니다.
err = try do
raise "oops"
rescue
e in RuntimeError -> e
end
IO.puts(err.message)
위의 프로그램이 실행되면 다음과 같은 결과가 생성됩니다.
oops
패턴 매칭을 사용하여 구조 문의 오류를 처리했습니다. 오류를 사용하지 않고 식별 목적으로 사용하려는 경우 다음 양식을 사용할 수도 있습니다.
err = try do
1 + "Hello"
rescue
RuntimeError -> "You've got a runtime error!"
ArithmeticError -> "You've got a Argument error!"
end
IO.puts(err)
위의 프로그램을 실행하면 다음과 같은 결과가 생성됩니다.
You've got a Argument error!
NOTE− Elixir 표준 라이브러리에있는 대부분의 함수는 한 번 튜플을 반환하고 다른 한 번은 오류를 발생시키는 두 번 구현됩니다. 예를 들어File.read 그리고 File.read!기능. 첫 번째 파일은 파일을 성공적으로 읽은 경우 튜플을 반환하고 오류가 발생하면이 튜플을 사용하여 오류의 원인을 제공했습니다. 두 번째 오류가 발생하면 오류가 발생했습니다.
첫 번째 함수 접근 방식을 사용하는 경우 오류와 일치하는 패턴에 대한 사례를 사용하고 이에 따라 조치를 취해야합니다. 두 번째 경우에는 오류가 발생하기 쉬운 코드에 대해 try rescue 접근 방식을 사용하고 그에 따라 오류를 처리합니다.
던짐
Elixir에서는 값을 던져 나중에 잡을 수 있습니다. Throw 및 Catch는 throw 및 catch를 사용하지 않는 한 값을 검색 할 수없는 상황을 위해 예약되어 있습니다.
인스턴스는 라이브러리와 인터페이스하는 경우를 제외하고 실제로는 매우 드뭅니다. 예를 들어, 이제 Enum 모듈이 값을 찾기위한 API를 제공하지 않았고 숫자 목록에서 13의 첫 번째 배수를 찾아야한다고 가정 해 보겠습니다.
val = try do
Enum.each 20..100, fn(x) ->
if rem(x, 13) == 0, do: throw(x)
end
"Got nothing"
catch
x -> "Got #{x}"
end
IO.puts(val)
위의 프로그램이 실행되면 다음과 같은 결과가 생성됩니다.
Got 26
출구
프로세스가 "자연적 원인"(예 : 처리되지 않은 예외)으로 죽으면 종료 신호를 보냅니다. 프로세스는 명시 적으로 종료 신호를 전송하여 죽을 수도 있습니다. 다음 예를 살펴 보겠습니다.
spawn_link fn -> exit(1) end
위의 예에서 연결된 프로세스는 값이 1 인 종료 신호를 전송하여 종료되었습니다. try / catch를 사용하여 종료를 "잡을"수도 있습니다. 예를 들면-
val = try do
exit "I am exiting"
catch
:exit, _ -> "not really"
end
IO.puts(val)
위의 프로그램이 실행되면 다음과 같은 결과가 생성됩니다.
not really
후
때때로 오류를 일으킬 수있는 일부 작업 후에 리소스를 정리해야합니다. try / after 구조를 사용하면 그렇게 할 수 있습니다. 예를 들어 파일을 열고 after 절을 사용하여 파일을 닫을 수 있습니다. 문제가 발생한 경우에도 마찬가지입니다.
{:ok, file} = File.open "sample", [:utf8, :write]
try do
IO.write file, "olá"
raise "oops, something went wrong"
after
File.close(file)
end
이 프로그램을 실행하면 오류가 발생합니다. 하지만after 문은 이러한 이벤트가 발생하면 파일 설명자가 닫히도록합니다.