functional-programming - Elixir - 改变行为

标签 functional-programming elixir behavior

这主要是一个函数式编程问题而不是 Elixir 问题,但由于我正在学习 Elixir,所以如果有人可以使用该语言回答它会很好。即便如此,如果有人想给出更笼统的答案,我们将不胜感激。

我自己是一名 OO 程序员,我无法全神贯注于如何根据配置文件更改组件的行为(例如)。

例子: 我有一个从数据库加载/保存用户的应用程序。在生产环境中,我希望在 MongoDB 数据库中保存和检索我的用户,而在开发和测试中,我想使用内存映射。如果我使用 OO 语言(比方说 Java)对给定系统进行编程,我会简单地创建一个名为“UserRepository”的接口(interface),其中包含 2 个实现:“MemoryUserRepository”和“MongoDBUserRepository”。然后我会在启动时基于配置文件(或对其进行硬编码,这无关紧要)实例化相应的存储库,并且在启动之后,所有与存储库交互的对象将永远不知道它的实现(它们将使用存储库,但永远不会关心它是在内存中还是在 mongo 中)。 这使我能够根据需要创建尽可能多的实现,而要更改系统的行为,我唯一需要做的就是实例化我想要使用的实现。

我想要相同的行为,但在 Elixir 中(让我们使用相同的示例)。因为它不是面向对象的语言,所以我不能使用上述方法。显然我希望它是可扩展的(我可以轻松地传递一个字符串,其中包含我想在每次调用中使用的存储库类型,并使用模式匹配来确定要使用的行为,但这不能很好地扩展,因为每次我都会想要添加一个实现我将不得不查看我模式匹配类型的每一段代码并添加新的实现)。实现这一目标的最佳方法是什么?

提前致谢!

最佳答案

假设您有这两个(或更多)存储库实现,它们实现了相同的接口(interface):

defmodule MyApp.Repository.Memory do
  def get(key) do
    # ...
  end

  def put(key, value) do
    # ...
  end
end

defmodule MyApp.Repository.Disk do
  def get(key) do
    # ...
  end

  def put(key, value) do
    # ...
  end
end

然后您可以编写一个通用存储库模块,该模块将根据 config/config.exs 文件中的配置值将函数调用转发到存储库后端之一:

defmodule MyApp.Repository do
  @backend Application.get_env(:my_app, :repository_backend)

  defdelegate [get(key), put(key, value)], to: @backend
end

可以进行配置,使其特定于环境(只需查看使用 mix new my_app 新创建的 mix 项目中的默认 config.exs):

# config/config.exs
import_config "#{Mix.env}.exs"

# config/dev.exs
config :my_app, repository_backend: MyApp.Repository.Memory

# config/prod.exs
config :my_app, repository_backend: MyApp.Repository.Disk

在整个代码中,您可以只使用 MyApp.Repository 模块,而无需明确引用其中一个特定实现:

MyApp.Repository.put(:foo, "Hello world!")
value = MyApp.Repository.get(:foo)

关于functional-programming - Elixir - 改变行为,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34969616/

相关文章:

javascript - 这个函数是纯的还是不纯的?

functional-programming - Erlang追踪: Causal consistency?

testing - 如何在 elixir 中为 escript 项目编写 ExUnit 测试用例

elixir - 在 Elixir 中实现外/内循环的最佳方式

php - 修改 beforeFind 回调中所需的 Containable 字段?

android - 如何在显示 snackbar 时上推现有 View ?

java - 如何交错(合并)两个 Java 8 Stream?

java - 在 Java 8 中仅使用 Function<> 实现 compose

elixir - 使用 Postgrex 和 Ecto 准备好的语句

.net - Control.Select() 和 Control.Focus() 有什么区别?