Rust: 聊聊AtomicPtr<()>和 *const ()

在Bytes库在github源码(https://docs.rs/bytes/1.1.0/src/bytes/bytes.rs.html#94-100)有关Bytes的定义中,

复制代码
pub struct Bytes {
 	ptr: *const u8,
 	 len: usize, // inlined "trait object"
   	data: AtomicPtr<()>, 
  	 vtable: &'static Vtable,
   }

其中的data字段中有(),一般地,()表示一个空元组。但在这里,是不是这个意思?

我们看到,data可以用这个来赋值:

复制代码
AtomicPtr::new(ptr::null_mut())

那ptr::null_mut()又是啥?查看https://doc.rust-lang.org/stable/std/ptr/fn.null_mut.html,可以看到:

复制代码
pub const fn null_mut<T: ?Sized + Thin>() -> *mut T {
    from_raw_parts_mut(invalid_mut(0), ())
}
//再查invaild_mut
pub const fn invalid_mut<T>(addr: usize) -> *mut T {
    // FIXME(strict_provenance_magic): I am magic and should be a compiler intrinsic.
    // We use transmute rather than a cast so tools like Miri can tell that this
    // is *not* the same as from_exposed_addr.
    // SAFETY: every valid integer is also a valid pointer (as long as you don't dereference that
    // pointer).
    unsafe { mem::transmute(addr) }
}

表示:Creates a null mutable raw pointer.类似,创建一个不变的raw pointer有,

复制代码
pub const fn null<T>() -> *const T
where
    T: Thin + ?Sized,

进而,查看https://doc.rust-lang.org/src/core/sync/atomic.rs.html#175,可以了解AtomicPtr的定义,

复制代码
pub struct AtomicPtr<T> {
    p: UnsafeCell<*mut T>,
}

也就是可以了解,Bytes中的data字段是一个用AtomicPtr封装起来的一个函数的可变raw pointer.

当然,上面的data还可以这样赋值:

复制代码
// slice是Box<[u8]>
//let ptr = Box::into_raw(slice) as *mut u8;
//let data = ptr as usize
data: AtomicPtr::new(data as *mut _),

另外,*const ()也自然可以联想到,它可能是表示指向一个固定的函数的raw pointer.

通过下面例子,也可以看到,*const ()与函数指针的转化。

复制代码
fn foo(i32) -> i32 {
    666666
}
fn bar()->String{
    println!("i am bar!");
    String::from("bar")
}

let pointer_1 = foo as *const ();
let pointer_2 = bar as *const ();
let fn_1 = unsafe {
    std::mem::transmute::<*const (), fn(i32) -> i32>(pointer_1)
};
let fn_2 = unsafe {
    std::mem::transmute::<*const (), fn() >(pointer_2)
};
assert_eq!(fn_1(42), 666666);
assert_eq!(fn_2(),String::from("bar"));
相关推荐
缺点内向13 小时前
C#: 高效移动与删除Excel工作表
开发语言·c#·.net·excel
老前端的功夫13 小时前
Web应用的永生之术:PWA落地与实践深度指南
java·开发语言·前端·javascript·css·node.js
程序员爱钓鱼14 小时前
Python编程实战:面向对象与进阶语法——类型注解与代码规范(PEP 8)
后端·python·ipython
程序员爱钓鱼14 小时前
Python实战:用高德地图API批量获取地址所属街道并写回Excel
后端·python·ipython
Yeats_Liao14 小时前
时序数据库系列(三):InfluxDB数据写入Line Protocol详解
数据库·后端·时序数据库
王元_SmallA14 小时前
Redis Desktop Manager(Redis可视化工具)安装
java·后端
ᐇ95914 小时前
Java HashMap深度解析:数据结构、原理与实战指南
java·开发语言·数据结构
好好研究14 小时前
Spring框架 - 开发方式
java·后端·spring
QT 小鲜肉14 小时前
【个人成长笔记】在 Linux 系统下撰写老化测试脚本以实现自动压测效果(亲测有效)
linux·开发语言·笔记·单片机·压力测试
程序员龙一15 小时前
C++之static_cast关键字
开发语言·c++·static_cast