7

我在一个模块中有一组函数需要访问一些共享的初始化时间状态。实际上,我想用静态可变向量对此进行建模,例如:

static mut defs: Vec<String> = vec![];

fn initialize() {
    defs.push("One".to_string());
    defs.push("Two".to_string()); 
}

(例如:http://is.gd/TyNQVv,失败,“可变静态不允许有析构函数”。)

我的问题类似于是否可以在 Rust 中使用全局变量?,但使用 a Vec(即带有析构函数的类型),因此Option基于 - 的解决方案似乎不适用于该问题。也就是说,这与我第一次尝试的错误相同:

static mut defs: Option<Vec<String>> = None;

fn initialize() {
    let init_defs = vec![];
    init_defs.push("One".to_string());
    init_defs.push("Two".to_string()); 
    defs = Some(init_defs);
}
  1. 有没有办法访问在初始化时填充并在运行时可见的静态(“全局”)向量?

  2. 我应该考虑其他模式来支持这个用例吗?传递对状态向量的显式引用是可能的,但会混淆大量需要访问此状态的函数签名。

4

1 回答 1

11

您可以lazy_static为此目的使用:

lazy_static! {
    static ref defs: Vec<String> = {
        let mut init = vec!["One".to_string(), "Two".to_string()];
        // init.push(...); etc. etc.
        init
    }
}

这会在第一次访问时初始化一个向量,之后它是不可变的。如果您希望稍后对其进行修改,将其包装在 astd::sync::Mutex中是一个很好的第一步。

我应该考虑其他模式来支持这个用例吗?传递对状态向量的显式引用是可能的,但会混淆大量需要访问此状态的函数签名。

要考虑的一种模式是创建一个上下文对象来存储函数所需的所有信息,例如

struct Context {
    defs: Vec<String>
}

然后传递Context确保每个人都知道他们需要知道什么。您甚至可以考虑将所有/许多/某些功能作为方法放在 上Context,例如

impl Context {
    fn foo(&self) {
        if self.defs.len() > 10 {
             println!("lots of defs");
        }
    }
    // ...
}

如果您需要修改上下文(自动确保线程安全)和/或如果您希望在单个进程中拥有多个独立实例,则此模式特别好。

于 2015-03-03T06:14:49.907 回答