Elixir-エラー処理

Elixirには、エラー、スロー、終了の3つのエラーメカニズムがあります。それぞれのメカニズムを詳しく見ていきましょう。

エラー

エラー(または例外)は、コードで例外的なことが発生した場合に使用されます。文字列に数値を追加しようとすると、サンプルエラーを取得できます-

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標準ライブラリのほとんどの関数は2回実装されており、1回はタプルを返し、もう1回はエラーを発生させます。たとえば、File.read そしてその File.read!関数。最初のタプルは、ファイルが正常に読み取られた場合にタプルを返し、エラーが発生した場合は、このタプルを使用してエラーの理由を示しました。エラーが発生した場合、2番目のものはエラーを発生させました。

最初の関数アプローチを使用する場合は、エラーのパターンマッチングにユースケースを使用し、それに応じてアクションを実行する必要があります。2番目のケースでは、エラーが発生しやすいコードに対してトライレスキューアプローチを使用し、それに応じてエラーを処理します。

投げる

Elixirでは、値をスローして後でキャッチすることができます。ThrowとCatchは、throwとcatchを使用しない限り値を取得できない状況のために予約されています。

ライブラリとインターフェイスする場合を除いて、インスタンスは実際には非常にまれです。たとえば、列挙型モジュールが値を見つけるための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の終了シグナルを送信することによって終了しました。exitは、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 ステートメントは、そのようなイベントが発生したときにファイル記述子が閉じられるようにします。