AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / coding / 问题 / 78231106
Accepted
ThoBo
ThoBo
Asked: 2024-03-27 18:51:38 +0800 CST2024-03-27 18:51:38 +0800 CST 2024-03-27 18:51:38 +0800 CST

可以采用什么结构来避免使用 RefCell?

  • 772

我在 Rust 中创建了一些代码,应该允许我设置递归端口、具有源的端口、可以具有源的端口等。但是,由于借用问题,我实现此结构的唯一方法是通过 RefCell,实际上我不想使用 RefCell (它允许我编译在运行时可能会出现恐慌的代码)或 std 中的任何其他库。

问题是,由于借用问题,我似乎无法为我的端口设置值。

struct Port<'a> {
    value: i32,
    source: Option<&'a Port<'a>>,
}

impl<'a> Port<'a> {
    fn new(value: i32) -> Port<'a> {
        Port { value, source: None }
    }

    fn link(&mut self, source: &'a Port<'a>) {
        self.source = Some(source);
    }

    fn get_source_value(&self) -> Option<i32> {
        match self.source {
            Some(port) => {
                let inner_source_value = port.get_source_value();
                match inner_source_value {
                    Some(value) => Some(value),
                    None => Some(port.value),
                }
            }
            None => None,
        }
    }
}

fn main() {
    let mut top_port = Port::new(10);
    let mut sub_port = Port::new(20);
    let mut sub_sub_port = Port::new(30);

    sub_port.link(&top_port);
    sub_sub_port.link(&sub_port);

    top_port.value = 400; // can't do this because top_port is borrowed

    println!("Value of sub_sub_port's source's source's source: {:?}", sub_sub_port.get_source_value());
}

出现问题的原因是当 sub_port 和 sub_sub_port 链接到 top_port 时,top_port 被借用。发生这种情况对我来说并不奇怪,我可以使用 RefCell 和 RC 来解决这个问题。

use std::cell::RefCell;

struct Port<'a> {
    value: i32,
    source: Option<&'a RefCell<Port<'a>>>,
}

impl<'a> Port<'a> {
    fn new(value: i32) -> RefCell<Port<'a>> {
        RefCell::new(Port { value, source: None })
    }

    fn link(&mut self, source: &'a RefCell<Port<'a>>) {
        self.source = Some(source);
    }

    fn get_source_value(&self) -> Option<i32> {
        match self.source {
            Some(port_ref) => {
                let port = port_ref.borrow();
                let inner_source_value = port.get_source_value();
                match inner_source_value {
                    Some(value) => Some(value),
                    None => Some(port.value),
                }
            }
            None => None,
        }
    }
}

fn main() {
    let top_port = Port::new(10);
    let sub_port = Port::new(20);
    let sub_sub_port = Port::new(30);

    sub_sub_port.borrow_mut().link(&sub_port);
    sub_port.borrow_mut().link(&top_port);

    top_port.borrow_mut().value = 400;

    println!("Value of sub_sub_port's source's source's source: {:?}", sub_sub_port.borrow().get_source_value());
}

然而,我的目标是不使用任何标准库功能,并且仅以安全的方式使用 Rust。我可以通过什么方式重组我的代码以避免使用 RefCell 或任何其他标准库?或者更好的是,我可以使用更多的生命周期来修复我的原始代码吗?

先感谢您!

recursion
  • 1 1 个回答
  • 62 Views

1 个回答

  • Voted
  1. Best Answer
    prog-fh
    2024-03-27T19:31:08+08:002024-03-27T19:31:08+08:00

    您可以使用AtomicI32代替,i32因为它的.store()功能不需要&mut.
    这是因为原子从根本上是共享的(否则不需要原子性),同时仍然允许突变(实际上仍然是内部可变性)。

    use core::sync::atomic::{AtomicI32, Ordering};
    
    struct Port<'a> {
        value: AtomicI32,
        source: Option<&'a Port<'a>>,
    }
    
    impl<'a> Port<'a> {
        fn new(value: i32) -> Port<'a> {
            Port {
                value: AtomicI32::new(value),
                source: None,
            }
        }
    
        fn link(
            &mut self,
            source: &'a Port<'a>,
        ) {
            self.source = Some(source);
        }
    
        fn get_source_value(&self) -> Option<i32> {
            match self.source {
                Some(port) => {
                    let inner_source_value = port.get_source_value();
                    match inner_source_value {
                        Some(value) => Some(value),
                        None => Some(port.value.load(Ordering::Relaxed)),
                    }
                }
                None => None,
            }
        }
    }
    
    fn main() {
        let top_port = Port::new(10);
        let mut sub_port = Port::new(20);
        let mut sub_sub_port = Port::new(30);
    
        sub_port.link(&top_port);
        sub_sub_port.link(&sub_port);
    
        top_port.value.store(400, Ordering::Relaxed);
    
        println!(
            "Value of sub_sub_port's source's source's source: {:?}",
            sub_sub_port.get_source_value()
        );
    }
    

    一般来说,众所周知,原子操作的效率低于常规操作,因为硬件需要小心谨慎,以便始终以一致的状态向所有内核/CPU 公开数据。
    在 x86(_64) 上,加载/存储没有区别,但在其他架构上可能是这种情况(但是对于添加、子...总是有惩罚)。这里使用的宽松的
    内存 顺序不应引入内存障碍,但其他顺序也可能会损害性能。

    然后,如果代码不涉及并行上下文,也许core::cell::Cell<i32>,正如评论中所建议的,会更有效一些。
    请注意,.get()此处有效,因为i32是Copy。

    use core::cell::Cell;
    
    struct Port<'a> {
        value: Cell<i32>,
        source: Option<&'a Port<'a>>,
    }
    
    impl<'a> Port<'a> {
        fn new(value: i32) -> Port<'a> {
            Port {
                value: Cell::new(value),
                source: None,
            }
        }
    
        fn link(
            &mut self,
            source: &'a Port<'a>,
        ) {
            self.source = Some(source);
        }
    
        fn get_source_value(&self) -> Option<i32> {
            match self.source {
                Some(port) => {
                    let inner_source_value = port.get_source_value();
                    match inner_source_value {
                        Some(value) => Some(value),
                        None => Some(port.value.get()),
                    }
                }
                None => None,
            }
        }
    }
    
    fn main() {
        let top_port = Port::new(10);
        let mut sub_port = Port::new(20);
        let mut sub_sub_port = Port::new(30);
    
        sub_port.link(&top_port);
        sub_sub_port.link(&sub_port);
    
        top_port.value.set(400);
    
        println!(
            "Value of sub_sub_port's source's source's source: {:?}",
            sub_sub_port.get_source_value()
        );
    }
    
    • 1

相关问题

  • 如何在递归函数中累加值?[复制]

  • OCaml 高阶函数

  • 如何使用 Cypher 递归创建树 (Neo4j)

  • 递归求幂中的浮点精度 MIPS

  • 在循环中调用自身的递归函数的时间复杂度

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    Vue 3:创建时出错“预期标识符但发现‘导入’”[重复]

    • 1 个回答
  • Marko Smith

    为什么这个简单而小的 Java 代码在所有 Graal JVM 上的运行速度都快 30 倍,但在任何 Oracle JVM 上却不行?

    • 1 个回答
  • Marko Smith

    具有指定基础类型但没有枚举器的“枚举类”的用途是什么?

    • 1 个回答
  • Marko Smith

    如何修复未手动导入的模块的 MODULE_NOT_FOUND 错误?

    • 6 个回答
  • Marko Smith

    `(表达式,左值) = 右值` 在 C 或 C++ 中是有效的赋值吗?为什么有些编译器会接受/拒绝它?

    • 3 个回答
  • Marko Smith

    何时应使用 std::inplace_vector 而不是 std::vector?

    • 3 个回答
  • Marko Smith

    在 C++ 中,一个不执行任何操作的空程序需要 204KB 的堆,但在 C 中则不需要

    • 1 个回答
  • Marko Smith

    PowerBI 目前与 BigQuery 不兼容:Simba 驱动程序与 Windows 更新有关

    • 2 个回答
  • Marko Smith

    AdMob:MobileAds.initialize() - 对于某些设备,“java.lang.Integer 无法转换为 java.lang.String”

    • 1 个回答
  • Marko Smith

    我正在尝试仅使用海龟随机和数学模块来制作吃豆人游戏

    • 1 个回答
  • Martin Hope
    Aleksandr Dubinsky 为什么 InetAddress 上的 switch 模式匹配会失败,并出现“未涵盖所有可能的输入值”? 2024-12-23 06:56:21 +0800 CST
  • Martin Hope
    Phillip Borge 为什么这个简单而小的 Java 代码在所有 Graal JVM 上的运行速度都快 30 倍,但在任何 Oracle JVM 上却不行? 2024-12-12 20:46:46 +0800 CST
  • Martin Hope
    Oodini 具有指定基础类型但没有枚举器的“枚举类”的用途是什么? 2024-12-12 06:27:11 +0800 CST
  • Martin Hope
    sleeptightAnsiC `(表达式,左值) = 右值` 在 C 或 C++ 中是有效的赋值吗?为什么有些编译器会接受/拒绝它? 2024-11-09 07:18:53 +0800 CST
  • Martin Hope
    The Mad Gamer 何时应使用 std::inplace_vector 而不是 std::vector? 2024-10-29 23:01:00 +0800 CST
  • Martin Hope
    Chad Feller 在 5.2 版中,bash 条件语句中的 [[ .. ]] 中的分号现在是可选的吗? 2024-10-21 05:50:33 +0800 CST
  • Martin Hope
    Wrench 为什么双破折号 (--) 会导致此 MariaDB 子句评估为 true? 2024-05-05 13:37:20 +0800 CST
  • Martin Hope
    Waket Zheng 为什么 `dict(id=1, **{'id': 2})` 有时会引发 `KeyError: 'id'` 而不是 TypeError? 2024-05-04 14:19:19 +0800 CST
  • Martin Hope
    user924 AdMob:MobileAds.initialize() - 对于某些设备,“java.lang.Integer 无法转换为 java.lang.String” 2024-03-20 03:12:31 +0800 CST
  • Martin Hope
    MarkB 为什么 GCC 生成有条件执行 SIMD 实现的代码? 2024-02-17 06:17:14 +0800 CST

热门标签

python javascript c++ c# java typescript sql reactjs html

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve