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