Rust

关于rust中的“安全”与“非安全”

2022-04-21  本文已影响0人  神奇的考拉

在实际使用Rust过程中很多时候,基于rust自身的来实现功能和代码的编写,并依托编译期自身来帮助我们进行“编译检查”,这时候相对来说我们使用的Rust是“安全的”;
不过另外一些“特殊”场景下的需求需要我们来处理底层实现,比如直接与系统层面交互或与汇编指令操作等,那此刻rust可能就变得“不安全”。换种说法:rust是一种同时包含安全和非安全特性的编程语言。

安全的Rust是一种真正安全的编程语言。

不代表Rust不能进行“非安全的行为”

关于unsafe

1、声明代码中存在编译期无法检查的安全规范
2、开发者会自觉遵守相关规范并不会主动破坏它(只能默认认为开发者“确保”了安全)

在实际应用中我们可以使用unsafe来定义函数和trait特型中存在不受编译期检查的规范:

当给一个代码块添加unsafe关键字,也就是意味着该声明块中的代码都已进行“人工检查”并符合相关规范;

当实现一个trait时使用了unsafe关键字,声明实现符合trait的安全规范;比如实现Send(一个标记trait,只是声明;其安全性均有实现该trait的开发者者来“确保”)的类型可以绝对安全move到另一个线程中。

3、标准库中的非安全函数
4、非安全的trait

许多 Rust 标准库其实内部也使用了非安全 Rust。这些库的实现方法都经过了严苛的人工检查,所以这些基于非安全 Rust 实现的安全 Rust 接口依然可以认为是安全的.

安全和非安全的 Rust 之间存在一种不对称的信任关系。安全 Rust 必须无条件信任非安全 Rust,假定所有与之打交道的非安全代码都是正确的。反过来,非安全 Rust 却要谨慎对待安全 Rust 的代码。

非安全 Rust 能做什么

非安全 Rust 比安全 Rust 可以多做的事情只有以下几个:

就这些操作被归为非安全的,是因为使用得不正确就会导致可怕的未定义行为。一旦触发了未定义行为,编译器就可以放飞自我,肆意破坏你的程序。切记,一定不能给未定义行为任何的机会。

与 C 不同,Rust 充分限制了可能出现的未定义行为的种类。语言核心只需要防止这几种行为:

只有这些Rust语言自身可以导致未定义行为的操作。当然,非安全函数和 trait 可以声明自己专有的安全规范,要求开发者必须遵守以避免未定义行为。比如,allocator API 声明回收一段未分配的内存是未定义行为。

但是,违背这些专有的规范通常也只是间接地触发上面列出的行为。另外,编译器内联函数也可能引入一些规则,一般是针对代码优化的假设条件。比如,Vec 和 Box 使用的内联函数要求传入的指针永远不能为 null。

Rust 对于一些模糊的操作则通常比较宽容。Rust 会认为下列操作是安全的:

当然,有以上行为的程序极有可能就是错误的。Rust 提供了一系列的工具减少这种事情的发生,但是完全地杜绝它们其实是不现实的。

引用

meet-safe-and-unsafe

上一篇 下一篇

猜你喜欢

热点阅读