- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我读过 What are non-lexical lifetimes? .使用非词法借用检查器,可以编译以下代码:
fn main() {
let mut scores = vec![1, 2, 3];
let score = &scores[0]; // borrows `scores`, but never used
// its lifetime can end here
scores.push(4); // borrows `scores` mutably, and succeeds
}
在上面的例子中看起来是合理的,但是当涉及到互斥锁时,我们不希望它被过早地释放。
在下面的代码中,我想先锁定一个共享结构,然后执行一个闭包,主要是为了避免死锁。但是,我不确定是否会提前释放锁。
use lazy_static::lazy_static; // 1.3.0
use std::sync::Mutex;
struct Something;
lazy_static! {
static ref SHARED: Mutex<Something> = Mutex::new(Something);
}
pub fn lock_and_execute(f: Box<Fn()>) {
let _locked = SHARED.lock(); // `_locked` is never used.
// does its lifetime end here?
f();
}
Rust 是否对锁进行了特殊处理,以确保它们的生命周期可以延长到作用域的末尾?我们是否必须显式使用该变量以避免过早释放锁,如以下代码所示?
pub fn lock_and_execute(f: Box<Fn()>) {
let locked = SHARED.lock(); // - lifetime begins
f(); // |
drop(locked); // - lifetime ends
}
最佳答案
这里有一个误解:NLL(非词法生命周期)影响的是借用检查,而不是对象的实际生命周期。
Rust 广泛使用 RAII1,因此许多对象(例如锁)的 Drop
实现具有副作用,必须在正常情况下发生执行流程中确定且可预测的点。
NLL 没有改变这些对象的生命周期,因此它们的析构函数在与之前完全相同的点执行:在它们的词法范围的末尾,以相反的创建顺序。
NLL 确实改变了编译器对使用生命周期进行借用检查的理解。实际上,这不会导致任何代码更改;这纯粹是分析。此分析变得更加巧妙,以更好地识别使用引用的实际范围:
在 Ref<'a>
(来自 RefCell
)的情况下,Ref<'a>
将被丢弃在词法范围的末尾,此时它将使用对 RefCell
的引用来递减计数器。
NLL 不会剥离抽象层,因此必须考虑任何包含引用的对象(例如 Ref<'a>
)可能在其 Drop
实现中访问该引用。因此,任何包含引用的对象(例如锁)都将强制 NLL 认为引用的“使用中”范围会延长,直到它们被删除。
1 Resource Acquisition Is Initialization,其本意是变量构造函数一旦执行完毕,就获得了所需的资源,不处于半生不熟的状态,通常用于表示销毁所述变量将释放它拥有的任何资源。
关于rust - 非词法生命周期借用检查器是否会过早释放锁?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57467555/
我试图了解传递给 setTimeout 的箭头函数如何记住上一个执行上下文中的 this 的值。我知道在执行箭头函数时会使用词法作用域规则查找 this 值。这是否意味着箭头函数关闭变量和 this
这个问题已经有答案了: How does the "this" keyword in Javascript act within an object literal? [duplicate] (4 个
我已阅读 this问题,我想我已经理解了投票最多的答案,但他说 since basically every programming language in wide use today uses le
如何让这段宏发挥预期的作用? -- 我想从词法环境中捕获 p 而不必将其作为参数发送给宏。 (define-syntax-rule (fi a b) (if p a b)) ;--->capt
Program A() { x, y, z: integer; procedure B() { y: integer; y=0;
我正在用 Java 实现自己的链表。节点类只有一个名为“name”的字符串字段和一个名为“link”的节点。现在我有一个测试驱动程序类,它只按顺序插入几个名字。现在,我正在尝试编写一种排序方法来按字母
考虑到这个question SO,其中调用了整个 C# 内存中编译器。只有lexical and syntactic analyzing时是必需的:将文本解析为词素流,检查它们并退出。 在System
我有 2 个场景。 这失败了: class F { public X X { get; set; } } 错误 CS0102:类型“F” ' 已经包含 ' X 的定义| ' 这个有效: class
我有一个用 NodeJS 执行的 .js 文件。这是我的文件的内容: var ctry = "America"; function outer(msg) { console.log(msg +
我对编写汇编程序的概念非常陌生,即使在阅读了大量 Material 之后,我仍然很难理解几个概念。 将源文件实际分解为 token 的过程是什么?我相信这个过程称为词法分析,我已经到处搜索有意义的真实
在 static scoping,标识符可以通过分析/解析源代码来确定(与动态作用域不同,动态作用域或多或少需要了解调用者环境)。 我的问题是这样的,因为静态作用域只需要解析源代码以了解作用域和标识符
编辑:我在第一个答案后更改了示例代码,因为我想出了一个简单的版本来回避相同的问题。 我目前正在学习 Common Lisp 的作用域属性。在我认为我有一个坚实的理解之后,我决定编写一些我可以预测结果的
考虑这段代码: class Bar(object): pass class Foo(object): def bar(self): return Bar() f = Foo() def Bar
将 ES6 箭头函数与词法 this 绑定(bind)结合使用非常棒。 但是,我刚才在使用典型的 jQuery 单击绑定(bind)时遇到了一个问题: class Game { foo() {
将 ES6 箭头函数与词法 this 绑定(bind)结合使用非常好。 但是,我刚才在将它与典型的 jQuery 点击绑定(bind)一起使用时遇到了一个问题: class Game { foo(
我是一名优秀的程序员,十分优秀!