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 문은 이러한 이벤트가 발생하면 파일 설명자가 닫히도록합니다.