Szukaj…


Uwagi

To jest analiza podsumowująca, którą przeprowadziłem w oparciu o metody wymienione w Jak definiować stałe w modułach Elixir? . Publikuję go z kilku powodów:

  • Większość dokumentacji Elixir jest dość dokładna, ale ta kluczowa decyzja architektoniczna nie zawiera wskazówek - więc poprosiłbym o nią jako temat.
  • Chciałem uzyskać trochę widoczności i komentarzy od innych na ten temat.
  • Chciałem również przetestować nowy przepływ pracy w dokumentacji SO. ;)

Przesłałem również cały kod do koncepcji eliksir-stałe stałych repozytorium GitHub.

Stałe o modułach

defmodule MyModule do
  @my_favorite_number 13
  @use_snake_case "This is a string (use double-quotes)"
end

Są one dostępne tylko z poziomu tego modułu.

Stałe jako funkcje

Ogłosić:

defmodule MyApp.ViaFunctions.Constants do
  def app_version, do: "0.0.1"
  def app_author, do: "Felix Orr"
  def app_info, do: [app_version, app_author]
  def bar, do: "barrific constant in function"
end

Spożywać z wymagają:

defmodule MyApp.ViaFunctions.ConsumeWithRequire do
  require MyApp.ViaFunctions.Constants

  def foo() do
    IO.puts MyApp.ViaFunctions.Constants.app_version
    IO.puts MyApp.ViaFunctions.Constants.app_author
    IO.puts inspect MyApp.ViaFunctions.Constants.app_info
  end

  # This generates a compiler error, cannot invoke `bar/0` inside a guard.
  # def foo(_bar) when is_bitstring(bar) do
  #   IO.puts "We just used bar in a guard: #{bar}"
  # end
end

Zużyj przy imporcie:

defmodule MyApp.ViaFunctions.ConsumeWithImport do
  import MyApp.ViaFunctions.Constants

  def foo() do
    IO.puts app_version
    IO.puts app_author
    IO.puts inspect app_info
  end
end

Ta metoda pozwala na ponowne użycie stałych w różnych projektach, ale nie będą one przydatne w funkcjach ochrony, które wymagają stałych czasu kompilacji.

Stałe za pośrednictwem makr

Ogłosić:

defmodule MyApp.ViaMacros.Constants do
  @moduledoc """
  Apply with `use MyApp.ViaMacros.Constants, :app` or `import MyApp.ViaMacros.Constants, :app`.

  Each constant is private to avoid ambiguity when importing multiple modules
  that each have their own copies of these constants.
  """

  def app do
    quote do
      # This method allows sharing module constants which can be used in guards.
      @bar "barrific module constant"
      defp app_version, do: "0.0.1"
      defp app_author, do: "Felix Orr"
      defp app_info, do: [app_version, app_author]
    end
  end

  defmacro __using__(which) when is_atom(which) do
    apply(__MODULE__, which, [])
  end
end

Spożywać z use :

defmodule MyApp.ViaMacros.ConsumeWithUse do
  use MyApp.ViaMacros.Constants, :app

  def foo() do
    IO.puts app_version
    IO.puts app_author
    IO.puts inspect app_info
  end

  def foo(_bar) when is_bitstring(@bar) do
    IO.puts "We just used bar in a guard: #{@bar}"
  end
end

Ta metoda pozwala na użycie @some_constant wewnątrz strażników. Nie jestem nawet pewien, czy funkcje byłyby absolutnie konieczne.



Modified text is an extract of the original Stack Overflow Documentation
Licencjonowany na podstawie CC BY-SA 3.0
Nie związany z Stack Overflow