这主要是一个功能编程问题而不是Elixir问题,但是因为我正在学习Elixir,如果有人能用这种语言回答它,那将会很好。即便如此,如果有人想要提供更一般的答案,我们将不胜感激。
我自己就是OO程序员,我无法理解如何根据配置文件更改组件的行为(例如)。
实施例: 我有一个从数据库加载/保存用户的应用程序。在生产环境中,我希望从MongoDB数据库中保存和检索我的用户,而在开发和测试中,我想使用内存映射。如果我用OO语言编写给定系统(比如说Java),我只需要创建一个名为“UserRepository”的接口,它有两个实现:“MemoryUserRepository”和“MongoDBUserRepository”。然后,我将在启动时基于配置文件(或硬编码,无关紧要)实例化相应的存储库,在它之后,与存储库交互的所有对象将永远不会知道它的实现(它们将使用存储库,但永远不会关心它是在记忆中还是在mongo中)。 这使我能够创建任意数量的实现,并且我需要做的唯一改变系统行为的方法是实例化我想要使用的实现。
我想在Elixir中使用相同的行为(让我们使用相同的例子)。由于它不是面向对象的语言,我不能使用上述方法。显然我希望它是可扩展的(我可以轻松地传递一个String,其中包含我想在每次调用中使用的存储库类型,并使用模式匹配来确定要使用的行为,但这不能很好地扩展,因为每次我都会想要添加一个实现,我将不得不查看每一段代码,我是模式匹配类型并添加新的实现)。实现这一目标的最佳方法是什么?
提前致谢!
答案 0 :(得分:7)
假设您有两个(或更多)存储库实现,它们实现相同的接口:
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
新创建的混合项目中的默认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)