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 / 问题 / 76985152
Accepted
ofo
ofo
Asked: 2023-08-27 07:05:38 +0800 CST2023-08-27 07:05:38 +0800 CST 2023-08-27 07:05:38 +0800 CST

在使用粗粒度锁访问的数据结构中使用 RefCell 是否安全?

  • 772

我正在编写一些 tokio 异步代码。我有一个多索引数据结构,将用户保留在我想要使用粗粒度锁(与每个对象锁相反)保护他们的位置。我有这样的事情:

use tokio::sync::RwLock;

struct User {
  id: u64,
  name: String,
}

// This class is not thread-safe.
struct UserDb {
  by_id: HashMap<u64, Arc<RefCell<User>>>,
  by_name: HashMap<String, Arc<RefCell<User>>>,
}

impl UserDb {
  pub fn add_user(&mut self, name: String) -> Result<(), Error> {
    // ...
  }
}

// This class is thread-safe.
struct AsyncDb {
  users: RwLock<UserDb>,
}

impl AsyncDb {
  pub async fn add_user(&self, name: String) -> Result<(), Error> {
    self.users.write().await.add_user(name)
  }
}

// QUESTION: Are these safe?
unsafe impl Send for AsyncDb {}
unsafe impl Sync for AsyncDb {}

如果末尾没有Sendand特征,编译器会抱怨不是and (合理地如此),因此通过 . 访问/修改是不安全的。SyncRefCell<User>SendSyncAsyncDb::add_user

我的解决方案是实现Send和数据结构,因为周围Sync有一个粗粒度锁,其中包含所述s。AsyncDbUserDbRefCell

这是正确的解决方案吗?它是否违反了任何不变量?有更好的方法来处理这个问题吗?

注意:这里是 Rust 初学者。我可能有很多概念上的差距,所以如果事情没有意义,请指出。

asynchronous
  • 2 2 个回答
  • 54 Views

2 个回答

  • Voted
  1. Best Answer
    cdhowie
    2023-08-27T07:55:49+08:002023-08-27T07:55:49+08:00

    这几乎肯定是不合理的,除非即使在从用户读取数据时也只采用写锁。

    这些RefCell::borrow*()函数不是线程安全的,因为它们不以原子方式维护内部引用计数。这意味着仅由读锁保护时使用borrow()从 a 读取是不合理的。RefCell<User>

    如果您已经对这个特定的设计感兴趣,我强烈建议您将RwLock其替换为Mutex,因为读锁几乎完全没有用处。

    替换RwLock为Mutex将使您的类型自动实现Sync,但它仍然不会实现Send。如果您unsafe impl Send在这种情况下保留 s,那么您需要非常小心 s Arc。您绝不能:

    • 直接返回一个Arc<RefCell<User>>值或对一个值的引用,或者嵌入到另一个值中,因为这将允许调用者克隆自己的值Arc,他们可以使用它来操作RefCell而不持有锁。
    • 通过、、 、在通道上发送一个等将一个Arc<RefCell<User>>或对一个线程/任务的引用发送到另一个线程/任务。thread::spawn()tokio::spawn()tokio::spawn_blocking()
    • 4
  2. kmdreko
    2023-08-27T07:57:39+08:002023-08-27T07:57:39+08:00

    这种结构并不安全。RefCell不能同时在两个线程上使用(即它没有实现Sync)并且您的结构无法防止这种情况。

    由于您使用 a RwLock,这意味着您可以为多个读取器提供对来自不同线程的值的不可变访问,这是不允许的。但即使您将其更改为Mutex独占的,您的Arc所有权也可以逃脱互斥锁的保护并在另一个线程中保留不可变的访问权限。

    话虽这么说,如果您确保没有逃脱的保护,那么实施Send和可能Sync是有效的。不过,如果您这样做,您甚至可以将它们降级为s。是的,它必须是而不是因为,同样,您不能同时从多个线程访问 a,甚至是不可变的。ArcMutexRcMutexRwLockRefCell

    • 2

相关问题

Sidebar

Stats

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

    使用 <font color="#xxx"> 突出显示 html 中的代码

    • 2 个回答
  • Marko Smith

    为什么在传递 {} 时重载解析更喜欢 std::nullptr_t 而不是类?

    • 1 个回答
  • Marko Smith

    您可以使用花括号初始化列表作为(默认)模板参数吗?

    • 2 个回答
  • Marko Smith

    为什么列表推导式在内部创建一个函数?

    • 1 个回答
  • Marko Smith

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

    • 1 个回答
  • Marko Smith

    java.lang.NoSuchMethodError: 'void org.openqa.selenium.remote.http.ClientConfig.<init>(java.net.URI, java.time.Duration, java.time.Duratio

    • 3 个回答
  • Marko Smith

    为什么 'char -> int' 是提升,而 'char -> Short' 是转换(但不是提升)?

    • 4 个回答
  • Marko Smith

    为什么库中不调用全局变量的构造函数?

    • 1 个回答
  • Marko Smith

    std::common_reference_with 在元组上的行为不一致。哪个是对的?

    • 1 个回答
  • Marko Smith

    C++17 中 std::byte 只能按位运算?

    • 1 个回答
  • Martin Hope
    fbrereto 为什么在传递 {} 时重载解析更喜欢 std::nullptr_t 而不是类? 2023-12-21 00:31:04 +0800 CST
  • Martin Hope
    比尔盖子 您可以使用花括号初始化列表作为(默认)模板参数吗? 2023-12-17 10:02:06 +0800 CST
  • Martin Hope
    Amir reza Riahi 为什么列表推导式在内部创建一个函数? 2023-11-16 20:53:19 +0800 CST
  • Martin Hope
    Michael A fmt 格式 %H:%M:%S 不带小数 2023-11-11 01:13:05 +0800 CST
  • Martin Hope
    God I Hate Python C++20 的 std::views::filter 未正确过滤视图 2023-08-27 18:40:35 +0800 CST
  • Martin Hope
    LiDa Cute 为什么 'char -> int' 是提升,而 'char -> Short' 是转换(但不是提升)? 2023-08-24 20:46:59 +0800 CST
  • Martin Hope
    jabaa 为什么库中不调用全局变量的构造函数? 2023-08-18 07:15:20 +0800 CST
  • Martin Hope
    Panagiotis Syskakis std::common_reference_with 在元组上的行为不一致。哪个是对的? 2023-08-17 21:24:06 +0800 CST
  • Martin Hope
    Alex Guteniev 为什么编译器在这里错过矢量化? 2023-08-17 18:58:07 +0800 CST
  • Martin Hope
    wimalopaan C++17 中 std::byte 只能按位运算? 2023-08-17 17:13:58 +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