Rust编程(三)生命周期与异常处理

生命周期

生命周期,简而言之就是引用的有效作用域。在大多数时候,我们无需手动的声明生命周期,因为编译器可以自动进行推导。生命周期的主要作用是避免悬垂引用,它会导致程序引用了本不该引用的数据:

rust 复制代码
{
	let r;
	{
		let x = 5;
		r = &x;
	}
	println!("R:{}",r);
}

这种情况r就是悬垂指针,r是引用的x,但是x的生命周期是到大括号结束就结束了,所以下面再使用r时,r就是一个悬垂指针。避免悬垂指针的方法就是:确保被引用的变量的生命周期长于引用本身。

rust 复制代码
fn main() {
    let string1 = String::from("abcd");
    let string2 = "xyz";

    let result = longest(string1.as_str(), string2);
    //会报错,因为编译器不知道返回是x还是y,自然就没法对result进行生命周期分析
    println!("The longest string is {}", result);
}

fn longest(x: &str, y: &str) -> &str {
    if x.len() > y.len() {
        x
    } else {
        y
    }
} 

对于一些复杂情况,编译器无法自己推导出生命周期,就需要我们自己进行生命周期标注:

rust 复制代码
&i32        // 一个引用
&'a i32     // 具有显式生命周期的引用
&'a mut i32 // 具有显式生命周期的可变引用

生命周期标注不对代码的逻辑起到任何作用,它自身并不具有什么意义,因为生命周期的作用就是告诉编译器多个引用之间的关系。

标注后的longest函数:

rust 复制代码
fn longest<'a>(x: &'a str, y: &'a str) -> &'a str {
    //这里告诉编译器,x和y的生命周期要长于`a
    //然后将二者的生命周期统一设定为较短的那个的生命周期
    //这样就不管是返回x还是y,result的生命周期都是一样的
    if x.len() > y.len() {
        x
    } else {
        y
    }
}

当只与一个参数有关时,可以只标注一个的生命周期:

rust 复制代码
fn function<'a>(arg1:&'a mut String,arg2:& mut String) -> &'a mut String{
    return arg1;
}

fn main(){
    let mut s1 : String = String::new();
    let mut s2 : String = String::new();
    let s3 = function(&mut s1,&mut s2);
    println!("s3:{}",s3);
}

生命周期问题只针对于引用,主要是为了解决引用/借用还在,被引用/借用的没了的情况,对于move就没有这么麻烦,一旦move,直接生命周期就重新计算了,与之前的变量就没关系了。

结构体也有生命周期的要求,结构体内部有引用,被引用的对象的生命周期要长于结构体对象的生命周期:

rust 复制代码
struct ImportantExcerpt<'a> {
    part: &'a str,
}

fn main() {
    let novel1 = String::from("Call me Ishmael. Some years ago...");
    let first_sentence1 = novel1.split('.').next().expect("Could not find a '.'");
    let i = ImportantExcerpt {
        part: first_sentence1,
    }; //这种没有问题
    
    let i;
    {
        let novel2 = String::from("Call me Ishmael. Some years ago...");
        let first_sentence2 = novel2.split('.').next().expect("Could not find a '.'");
        i = ImportantExcerpt {
            part: first_sentence2,
        };
    }
    println!("{:?}",i); //这种就会报错,因为first_sentence2在大括号结束的时候生命周期已经结束了
}

函数情况下分为输入生命周期和输出生命周期,一些编译器可以推断出输入输出生命周期的情况,就不需要进行生命周期标注,编译器推断生命周期有三个规则:

  • 每一个引用参数都会获得独自的生命周期

    例如一个引用参数的函数就有一个生命周期标注: fn foo<'a>(x: &'a i32),两个引用参数的有两个生命周期标注:fn foo<'a, 'b>(x: &'a i32, y: &'b i32), 依此类推。

  • 若只有一个输入生命周期(函数参数中只有一个引用类型),那么该生命周期会被赋给所有的输出生命周期,也就是所有返回值的生命周期都等于该输入生命周期

  • 例如函数 fn foo(x: &i32) -> &i32,x 参数的生命周期会被自动赋给返回值 &i32,因此该函数等同于 fn foo<'a>(x: &'a i32) -> &'a i32

  • 若存在多个输入生命周期,且其中一个是 &self 或 &mut self,则 &self 的生命周期被赋给所有的输出生命周期。拥有 &self 形式的参数,说明该函数是一个 方法,该规则让方法的使用便利度大幅提升。

    更多关于生命周期的内容,请参考

函数返回/异常处理

Rust的函数返回为了解决是否有值的问题,引入了一个枚举类型Option,有值时返回Some(T),没值时返回None,再也不用绞尽脑汁如何思考返回-1或者None了。

rust 复制代码
enum Option<T> {
    Some(T),
    None,
}

Rust的异常分为可恢复异常Result,出了异常不直接崩溃,还可以根据异常的类别继续运行,类似于其他语言中函数返回了-1或者None。另一种是不可恢复异常panic,出了问题直接崩溃,类似于其他语言中的assert。

Result<T,E>

对应可以挽救的error,Result是一个枚举类型

rust 复制代码
enum Result<T, E> {
    Ok(T),
    Err(E),
}

使用案例:

rust 复制代码
fn plus_one(x: Option<i32>) -> Option<i32> {
    match x { //match就类似于C里面的switch语句
        None => None,
        Some(i) => Some(i + 1),
    }
}

let five = Some(5);
let six = plus_one(five);
let none = plus_one(None);

当函数正常运行时,返回Ok包裹着的返回结果,当函数运行失败时,返回Err包裹的错误类型,

一般使用Result的工作流程就是:函数返回一个Result,然后使用match匹配结果:

rust 复制代码
use std::fs::File;
use std::io::ErrorKind;

fn main() {
    let f = File::open("hello.txt");

    let f = match f {
        Ok(file) => file,
        Err(error) => match error.kind() {
            ErrorKind::NotFound => match File::create("hello.txt") {
                Ok(fc) => fc,
                Err(e) => panic!("Problem creating the file: {:?}", e),
            },
            other_error => panic!("Problem opening the file: {:?}", other_error),
        },
    };
}

Result也可以通过unwrap或者except转换为panic:

rust 复制代码
use std::fs::File;

fn main() {
    let f = File::open("hello.txt").unwrap();
    //不出问题就提取出来返回结果,出问题就直接panic
    let f = File::open("hello.txt").expect("Failed to open hello.txt");
    //跟上面一样的效果,不过会报错自定义的错误提示
}

panic

对应不可恢复error,出现panic代码直接崩溃

rust 复制代码
fn main() {
    panic!("crash and burn");
}

更多关于panic的请看Rust圣经

相关推荐
神奇夜光杯4 分钟前
Python酷库之旅-第三方库Pandas(202)
开发语言·人工智能·python·excel·pandas·标准库及第三方库·学习与成长
Themberfue6 分钟前
Java多线程详解⑤(全程干货!!!)线程安全问题 || 锁 || synchronized
java·开发语言·线程·多线程·synchronized·
plmm烟酒僧8 分钟前
Windows下QT调用MinGW编译的OpenCV
开发语言·windows·qt·opencv
测试界的酸菜鱼19 分钟前
Python 大数据展示屏实例
大数据·开发语言·python
晨曦_子画29 分钟前
编程语言之战:AI 之后的 Kotlin 与 Java
android·java·开发语言·人工智能·kotlin
Black_Friend37 分钟前
关于在VS中使用Qt不同版本报错的问题
开发语言·qt
假装我不帅1 小时前
asp.net framework从webform开始创建mvc项目
后端·asp.net·mvc
神仙别闹1 小时前
基于ASP.NET+SQL Server实现简单小说网站(包括PC版本和移动版本)
后端·asp.net
希言JY1 小时前
C字符串 | 字符串处理函数 | 使用 | 原理 | 实现
c语言·开发语言
残月只会敲键盘1 小时前
php代码审计--常见函数整理
开发语言·php