gpt4 book ai didi

serialization - 如何抽象化Rust中可远程或本地化的可序列化对象?

转载 作者:行者123 更新时间:2023-12-03 11:38:42 25 4
gpt4 key购买 nike

我有一个枚举来表示节点是本地节点(指当前节点)还是远程节点,在这种情况下,通信是通过RPC调用进行的:

pub enum Node {
Local,
Remote(SocketAddr),
}
我有一个代表本地数据库的 DB结构。我需要序列化此 Node枚举,因为它用作元数据来确定某些键值对的存储位置(本地或远程)。我将此元数据与其余数据存储在同一数据库中,因此 Node::Local指的是它存储在其中的数据库。
我希望反序列化的节点枚举包含对该数据库的引用,以便可以将该 Local变体转换为 Local(DB)。然后,对于 LocalRemote变体,我将具有相同的语义,因为它们都包含足够的信息来执行 Node.get(key)Node.insert(key, value)操作。
对于远程变体,我只需要打开到 SocketAddr的连接并发出RPC请求,但是对于 Local变体,我必须在我对本地数据库有引用的地方进行模式匹配,并应用特殊的逻辑而不是通常可以说 Node.get(key)。基本问题是 Node::Local包含的信息不足,无法对本地数据库执行请求,而 SocketAddr中的 Node::Remote足以执行RPC调用。
我可以通过创建一个自定义反序列化方法来解决此问题,在该方法中,我将当前的 DB传递给该反序列化方法的引用,但是我想知道是否还有其他好的方法可以解决此问题。

最佳答案

我最终将这种区别表示为枚举。使用单独的枚举进行序列化和实际工作,因为我无法序列化对本地数据库的引用,所以如果不需要序列化,那么当然可以避开NodeRepr枚举:

#[derive(Serialize, Deserialize)]
pub enum NodeRepr {
Local,
Remote(NodeRef),
}

pub enum Node<'a> {
Local(&'a LocalDB),
Remote(NodeRef),
}

然后,节点枚举具有impl方法,通过match语句将必要的方法调用委派给localdb或NodeRef方法。
impl Node {
pub async fn get(&self, key: String) -> Option<String> {
match self {
Node::Local(db) => db.get(key),
Node::Remote(noderef) => noderef.get(key).await,
}
}
}
这种处理事物的方式使我们仍然可以保留节点是本地还是远程的信息,因此我们可以在必要时在代码中的任何地方使用此信息。这很有用,因为您通常不希望总是隐藏有关某物是本地还是远程的信息。
例如,如果您始终想隐藏某个节点是远程节点还是本地节点以提高代码清洁度,则可以使用trait对象,在该对象中,LocalDB和NodeRef都实现相同的特征。
特征特别有用,因为它允许添加新的节点类型(本地和远程),而无需更改任何旧代码。使用枚举时,预期只有两种不同的枚举情况的代码将在添加新的节点类型时中断。通过应用经典的OOP模式(如访客模式),也可以克服特征对象的局限性。

关于serialization - 如何抽象化Rust中可远程或本地化的可序列化对象?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/64847691/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com