[英]Elixir : Mnesia : What is the most concise way to update a set of value in an element?
通过 Elixir 学习使用 Mnesia,我创建了一个包含多个 function 的表(例如读、写...)。 其中之一是更新一组字段(大小从 1 到 count - 1)而不更改数据的 rest,并将 mnesia 事务内部的逻辑限制为最小。
在我的搜索过程中,我碰巧发现了这个: Erlang: Mnesia: Updating a single field value in a row but (code below). 这是同一个问题,但对于 Erlang,而不是 Elixir。
据我了解,它在 Erlang 中工作,因为 read 返回一个直接设置在记录中的元组,这允许我们在写入操作中保存特定数据,因为它们被命名。
update_a(Tab, Key, Value) ->
fun() ->
[P] = mnesia:wread({Tab, Key}),
mnesia:write(P#pixel{a=Value})
end.
其中,对于 Elixir,即使 Records 存在,它也只是一个元组,您只能更改索引中的数据,并将完整的元组返回给写入操作。
Table: {table_name, id, data1, data2, data3, data4}
changes = [{2, new_val_for_data1}, {4, new_val_for_data3}]
def handle_call({:update_and_read, table, {id, changes}}, _, state) do
{:atomic, new_object} =
:mnesia.transaction(fn ->
object =
:mnesia.wread({table, id})
|> List.first()
ret =
Enum.reduce(changes, object, fn {index, value}, acc ->
acc |> Tuple.delete_at(index) |> Tuple.insert_at(index, value)
end)
:mnesia.write(object)
ret
end)
{:reply, {:ok, new_object}, state}
end
是否有可能在 Elixir 中有更短的 function (理想情况下,两行就像 Erlang 中的那一行)?
好吧,您可以通过一些初步步骤使用Record
(例如在您的应用程序中定义记录。)
defmodule Pixel do
require Record
Record.defrecord(:table_name, id: nil, data1: nil, data2: nil)
end
并在要更新表格的模块中
import Pixel
def handle_call({:update_and_read, table, {id, changes}}, _, state) do
# changes = [data2: new_val_for_data1, ...]
{:atomic, result} =
:mnesia.transaction(fn ->
case :mnesia.wread({table, id}) do
[object] ->
:mnesia.write(pixel(object, changes))
{:ok, object}
[] -> :error
end
end)
{:reply, result, state}
end
另一种可能性是通过 go
changes = %{2 => new_val_for_data1, 4 => new_val_for_data3}
object
|> Tuple.to_list()
|> Enum.with_index()
|> Enum.map(fn {value, idx} -> Map.get(changes, idx, value) end)
|> List.to_tuple()
另一种可能性是声明一个宏,该宏接受一个元组,表示一个表行和一个{idx, new_value}
元组列表,并就地更改相应的元素。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.