注意:此回复有点旧。由于它涉及内部和不稳定的特性,事情已经有了一些变化。但基本机制仍然相同,因此该答案仍能够解释box
的底层机制。
box x
通常使用标记为exchange_malloc
的语言项函数进行分配和标记为exchange_free
的函数进行释放。您可以在默认标准库中的heap.rs#L112和heap.rs#L125中看到这些的实现。
最终,box x
语法取决于以下语言项:
- 使用
owned_box
在一个Box
结构体中封装分配的指针。这个结构体不需要Drop
实现,编译器会自动实现。
- 使用
exchange_malloc
来分配内存。
- 使用
exchange_free
来释放之前分配的内存。
在不稳定的Rust书籍的lang items章节中可以通过这个no_std
示例有效地看到这一点:
#![feature(lang_items, box_syntax, start, no_std, libc)]
#![no_std]
extern crate libc;
extern {
fn abort() -> !;
}
#[lang = "owned_box"]
pub struct Box<T>(*mut T);
#[lang = "exchange_malloc"]
unsafe fn allocate(size: usize, _align: usize) -> *mut u8 {
let p = libc::malloc(size as libc::size_t) as *mut u8;
if p as usize == 0 {
abort();
}
p
}
#[lang = "exchange_free"]
unsafe fn deallocate(ptr: *mut u8, _size: usize, _align: usize) {
libc::free(ptr as *mut libc::c_void)
}
#[start]
fn main(argc: isize, argv: *const *const u8) -> isize {
let x = box 1;
0
}
#[lang = "stack_exhausted"] extern fn stack_exhausted() {}
#[lang = "eh_personality"] extern fn eh_personality() {}
#[lang = "panic_fmt"] fn panic_fmt() -> ! { loop {} }
请注意,
Box
结构体没有实现
Drop
方法。那么我们来看一下生成的
main
的 LLVM IR:
define internal i64 @_ZN4main20hbd13b522fdb5b7d4ebaE(i64, i8**) unnamed_addr #1 {
entry-block:
%argc = alloca i64
%argv = alloca i8**
%x = alloca i32*
store i64 %0, i64* %argc, align 8
store i8** %1, i8*** %argv, align 8
%2 = call i8* @_ZN8allocate20hf9df30890c435d76naaE(i64 4, i64 4)
%3 = bitcast i8* %2 to i32*
store i32 1, i32* %3, align 4
store i32* %3, i32** %x, align 8
call void @"_ZN14Box$LT$i32$GT$9drop.103617h8817b938807fc41eE"(i32** %x)
ret i64 0
}
预期调用allocate
(_ZN8allocate20hf9df30890c435d76naaE
)来构建Box
,同时...看!Box
(_ZN14Box$LT$i32$GT$9drop.103617h8817b938807fc41eE
) 的Drop
方法!让我们看一下这个方法的IR:
define internal void @"_ZN14Box$LT$i32$GT$9drop.103617h8817b938807fc41eE"(i32**) unnamed_addr #0 {
entry-block:
%1 = load i32** %0
%2 = ptrtoint i32* %1 to i64
%3 = icmp ne i64 %2, 2097865012304223517
br i1 %3, label %cond, label %next
next: ; preds = %cond, %entry- block
ret void
cond: ; preds = %entry-block
%4 = bitcast i32* %1 to i8*
call void @_ZN10deallocate20he2bff5e01707ad50VaaE(i8* %4, i64 4, i64 4)
br label %next
}
这里有一个编译器生成的 Drop,它调用了
deallocate
(
ZN10deallocate20he2bff5e01707ad50VaaE
)。即使在
标准库上,
Drop
特质也不是由用户代码实现的。事实上,
Box
是一个有点神奇的结构体。
box x
是放置新语法吗? - George HilliardBox
最终将不再像我创建的任何类型一样被特殊处理,它将简单地使用其他(1.0版本后)语言功能。 - George Hilliard