lifetime-elision.md
commit: f8e76ee9368f498f7f044c719de68c7d95da9972
本章译文最后维护日期:2020-11-16
Rust 拥有一套允许在多种位置省略生命周期的规则,但前提是编译器在这些位置上能推断出合理的默认生命周期。
为了使常用模式使用起来更方便,可以在函数项、函数指针和闭包trait的签名中省略生命周期类型参数。以下规则用于推断出被省略的生命周期类型参数。省略不能被推断出的生命周期类型参数是错误的。占位符形式的生命周期,'_
,也可以用这一套规则来推断出。对于路径中的生命周期,首选使用 '_
。trait对象的生命周期类型参数遵循不同的规则,具体这里讨论。
- 参数中省略的每个生命周期类型参数都会(被推断)成为一个独立的生命周期类型参数。
- 如果参数中只使用了一个生命周期(省略或不省略都行),则将该生命周期作为所有省略的输出生命周期类型参数。
在方法签名中有另一条规则
- 如果接受者(receiver)类型为
&Self
或 &mut Self
,那么对 Self
的引用的生命周期会被作为所有省略的输出生命周期类型参数。
示例:
#![allow(unused)]
fn main() {
trait T {}
trait ToCStr {}
struct Thing<'a> {f: &'a i32}
struct Command;
trait Example {
fn print1(s: &str);
fn print2(s: &'_ str);
fn print3<'a>(s: &'a str);
fn debug1(lvl: usize, s: &str);
fn debug2<'a>(lvl: usize, s: &'a str);
fn substr1(s: &str, until: usize) -> &str;
fn substr2<'a>(s: &'a str, until: usize) -> &'a str;
fn get_mut1(&mut self) -> &mut dyn T;
fn get_mut2<'a>(&'a mut self) -> &'a mut dyn T;
fn args1<T: ToCStr>(&mut self, args: &[T]) -> &mut Command;
fn args2<'a, 'b, T: ToCStr>(&'a mut self, args: &'b [T]) -> &'a mut Command;
fn new1(buf: &mut [u8]) -> Thing<'_>;
fn new2(buf: &mut [u8]) -> Thing;
fn new3<'a>(buf: &'a mut [u8]) -> Thing<'a>;
}
type FunPtr1 = fn(&str) -> &str;
type FunPtr2 = for<'a> fn(&'a str) -> &'a str;
type FunTrait1 = dyn Fn(&str) -> &str;
type FunTrait2 = dyn for<'a> Fn(&'a str) -> &'a str;
}
#![allow(unused)]
fn main() {
trait Example {
fn get_str() -> &str;
fn frob(s: &str, t: &str) -> &str;
}
}
假定存在于(代表) trait对象(的那个胖指针)上的生命周期(assumed lifetime)类型参数称为此 trait对象的默认对象生命周期约束(default object lifetime bound)。这些在 RFC 599 中定义,在 RFC 1156 中修定增补。
当 trait对象的生命周期约束被完全省略时,会使用默认对象生命周期约束来替代上面定义的生命周期类型参数省略规则。但如果使用 '_
作为生命周期约束,则该约束仍遵循上面通常的省略规则。
如果将 trait对象用作泛型类型的类型参数,则首先使用此容器泛型来尝试为此 trait对象推断一个约束(来替代那个假定的生命周期)。
- 如果存在来自此容器泛型的唯一约束,则该约束就为此 trait对象的默认约束
- 如果此容器泛型有多个约束,则必须指定一个约显式束为此 trait对象的默认约束
如果这两个规则都不适用,则使用该 trait对象的声明时的 trait约束:
- 如果原 trait 声明为单生命周期约束,则此 trait对象使用该约束作为默认约束。
- 如果
'static
被用做原 trait声明的任一一个生命周期约束,则此 trait对象使用 'static
作为默认约束。
- 如果原 trait声明没有生命周期约束,那么此 trait对象的生命周期会在表达式中根据上下文被推断出来,在表达式之外直接用
'static
。
#![allow(unused)]
fn main() {
trait Foo { }
type T1 = Box<dyn Foo>;
type T2 = Box<dyn Foo + 'static>;
impl dyn Foo {}
impl dyn Foo + 'static {}
type T3<'a> = &'a dyn Foo;
type T4<'a> = &'a (dyn Foo + 'a);
type T5<'a> = std::cell::Ref<'a, dyn Foo>;
type T6<'a> = std::cell::Ref<'a, dyn Foo + 'a>;
}
#![allow(unused)]
fn main() {
trait Foo { }
struct TwoBounds<'a, 'b, T: ?Sized + 'a + 'b> {
f1: &'a i32,
f2: &'b i32,
f3: T,
}
type T7<'a, 'b> = TwoBounds<'a, 'b, dyn Foo>;
}
注意,像 &'a Box<dyn Foo>
这样多层包装的,只需要看最内层包装 dyn Foo
的那层,所以扩展后仍然为 &'a Box<dyn Foo + 'static>
#![allow(unused)]
fn main() {
trait Bar<'a>: 'a { }
type T1<'a> = Box<dyn Bar<'a>>;
type T2<'a> = Box<dyn Bar<'a> + 'a>;
impl<'a> dyn Bar<'a> {}
impl<'a> dyn Bar<'a> + 'a {}
}
除非指定了显式的生命周期,引用类型的常量项声明和静态项声明都具有隐式的静态('static
)生命周期。因此,有 'static
生命周期的常量项声明在编写时可以略去其生命周期。
#![allow(unused)]
fn main() {
const STRING: &str = "bitstring";
struct BitsNStrings<'a> {
mybits: [u32; 2],
mystring: &'a str,
}
const BITS_N_STRINGS: BitsNStrings<'_> = BitsNStrings {
mybits: [1, 2],
mystring: STRING,
};
}
注意,如果静态项(static
)或常量项(const
)包含对函数或闭包的引用,而这些函数或闭包本身也包含引用,此时编译器将首先尝试使用标准的省略规则来推断生命周期类型参数。如果它不能通过通常的生命周期省略规则来推断出生命周期类型参数,那么它将报错。举个例子:
#![allow(unused)]
fn main() {
struct Foo;
struct Bar;
struct Baz;
fn somefunc(a: &Foo, b: &Bar, c: &Baz) -> usize {42}
const RESOLVED_SINGLE: fn(&str) -> &str = |x| x;
const RESOLVED_MULTIPLE: &dyn Fn(&Foo, &Bar, &Baz) -> usize = &somefunc;
}
#![allow(unused)]
fn main() {
struct Foo;
struct Bar;
struct Baz;
fn somefunc<'a,'b>(a: &'a Foo, b: &'b Bar) -> &'a Baz {unimplemented!()}
const RESOLVED_STATIC: &dyn Fn(&Foo, &Bar) -> &Baz = &somefunc;
}