当前位置: 首页 > news >正文

Rust生命周期、文件与IO

文章目录

    • Rust生命周期
      • 生命周期注释
      • 结构体如何使用字符串
      • 静态生命周期
    • Rust文件与IO
      • 接收命令行参数
      • 命令行输入
      • 文件读取
      • 文件写入

Rust生命周期

终于讲到Rust最重要的机制之一了,生命周期机制

我们先复习一下垂悬引用

{let r;{let x = 5;r = &x;}println!("r: {}", r);
}

这一段代码是编译失败的,原因如下

我们可以看到有两个作用域

rust-lifetime1

a和b,但是其实生命周期也是类似的概念,r的生命周期中a的范围里,x的生命周期中b的范围内

r保存的x的引用,当x销毁之后,r的引用也就失效了,因此也就产生了垂悬引用

这里有一个案例

fn longer(s1: &str, s2: &str) -> &str {if s2.len() > s1.len() {s2} else {s1}
}

接收两个字符串引用,返回两个字符串引用中较长的一个

这段代码是不能通过编译的,因为返回值返回的引用可能会过期,例如这样调用

fn main() {let r;{let s1 = "rust";let s2 = "ecmascript";r = longer(s1, s2);}println!("{} is longer", r);
}

r最后接收的是s2的引用,但是等到我们使用r的时候,s2已经释放了,Rust是会消除一切可能导致危险的情况

生命周期注释

生命周期注释是描述引用生命周期的方法,相当于是给生命周期做一个标记

虽然这样并不能改变引用的生命周期,但是可以在合适的地方声明两个引用的生命周期一致

例如说

&i32        // 常规引用
&'a i32     // 含有生命周期注释的引用
&'a mut i32 // 可变型含有生命周期注释的引用

然后我们就可以用生命周期注释来改造这个函数

fn longer<'a>(s1: &'a str, s2: &'a str) -> &'a str {if s2.len() > s1.len() {s2} else {s1}
}

调用的时候就可以正常调用了

fn main() {let r;{let s1 = "rust";let s2 = "ecmascript";r = longer(s1, s2);println!("{} is longer", r);}
}

结构体如何使用字符串

之前的样例中,我们使用字符串都是使用的String类型,而不是str类型,主要就是考虑到生命周期的问题

那如果我们需要使用str,就需要设定生命周期

struct Str<'a> {content: &'a str}

这里定义了一个Str结构体,包含一个str类型的字符串,生命周期和结构体生命周期相同

然后使用是这样的

fn main() {let s = Str {content: "string_slice"};println!("s.content = {}", s.content);
}

这里都是没有问题的

那如果我们要给这个结构体赋予一个方法

impl<'a> Str<'a> {fn get_content(&self) -> &str {self.content}
}

这里的返回值类型没有写生命周期注释,加不加都可以的,现在已经可以自动判断生命周期了

静态生命周期

生命周期还有一个注释,'static,所有使用双引号包括的字符串都是`&'static str

表示生命周期从程序的开始一直到程序的结束

Rust文件与IO

接收命令行参数

我们在学C/C++的时候是通过main函数接收参数的,一个字符串数组

但是Rust的main函数是一个午餐的函数,环境的参数是直接通过std::env直接取出来的

例如

fn main() {let args = std::env::args();println!("{:?}", args);
}

这个打印出来的第一个参数是当前运行的目录,而后续的参数可能是环境变量或者是命令行参数

命令行输入

我们可以直接使用std::io这个模块来调用输入

use std::io::stdin;fn main() {
let mut str_buf = String::new();stdin().read_line(&mut str_buf).expect("Failed to read line.");println!("Your input line is \n{}", str_buf);
}

read_line可以直接读取一行信息到缓冲区,返回值是Result枚举,用于传递读取中的错误,可以用expect或者unwrap方法来处理错误

Rust中还没有提供直接从命令行读取数字或者格式化数据的方法,主要还是读取一行字符串,然后再用字符串识别函数处理数据

文件读取

这个就是用std::fs模块即可

例如

use std::fs;fn main() {let text = fs::read_to_string("~\text.txt").unwrap();println!("{}", text);
}

如果要读取二进制文件,则直接使用模块中的read函数即可

如果是需要读取大型文件,则可能需要流式处理的方式

例如说

use std::io::prelude::*;
use std::fs;fn main() {let mut buffer = [0u8; 5];let mut file = fs::File::open("~\text.txt").unwrap();file.read(&mut buffer).unwrap();println!("{:?}", buffer);file.read(&mut buffer).unwrap();println!("{:?}", buffer);
}

File是描述文件的一个类,打开文件之后,就可以获取一个文件对象

然后可以通过对象的read方法读取字节到缓冲区,读取的字节数等于缓冲区的长度

文件写入

文件写入也分为一次写入和流式写入,流写入有打开方式,可以设置create和append

一次写入很简单

use std::fs;fn main() {fs::write("~\text.txt", "FROM RUST PROGRAM").unwrap();
}

流写入create需要调用对应的方法

use std::io::prelude::*;
use std::fs::File;fn main() {let mut file = File::create("~\text.txt").unwrap();file.write(b"FROM RUST PROGRAM").unwrap();
}

写入的话就需要设置为mut

append就需要OpenOptions来设置

use std::io::prelude::*;
use std::fs::OpenOptions;fn main() -> std::io::Result<()> {let mut file = OpenOptions::new().append(true).open("~\text.txt")?;file.write(b" APPEND WORD")?;Ok(())
}

相关文章:

  • 20.3 使用技巧3
  • Qwen2.5-VL视觉大语言模型复现过程,没碰到什么坑
  • 【AI量化第24篇】KhQuant 策略框架深度解析:让策略开发回归本质——基于miniQMT的量化交易回测系统开发实记
  • Go语言入门到入土——三、处理并返回异常
  • Python爬虫之线程、进程、协程详解
  • Python 实现日志备份守护进程
  • JavaScript模块化开发:CommonJS、AMD到ES模块
  • AUTOSAR图解==>AUTOSAR_SWS_CryptoDriver
  • blender里面的材质列表
  • JavaEE——线程安全
  • Java工具类——实体类列表写入excel
  • Java 工厂设计模式详解:用统一入口打造灵活可扩展的登录系统----掌握 Spring 源码的基础第一步
  • 【Semantic Kernel核心组件】Plugin:连接AI与业务逻辑的桥梁
  • EmbeddingBag介绍与案例
  • Android问题整理
  • 数据加盐/加密
  • CentOS 中安装 vim
  • qt中关于思源雅黑字体的使用
  • OpenCV 图形API(43)颜色空间转换-----将 BGR 图像转换为 LUV 色彩空间函数BGR2LUV()
  • 《vue3学习手记4》
  • 可量产9MWh超大容量储能系统亮相慕尼黑,宁德时代:大储技术迈入新时代
  • 大四本科生已发14篇SCI论文?学校工作人员:已记录汇报
  • 李彦宏:技术迭代速度之快从业30年来未见过,要提升执行力战胜对手
  • 从“重规模”向“重回报”转变,公募基金迎系统性改革
  • 全国首例在沪完成,这项近视治疗手术不到10秒
  • 金融监管总局将出八大增量政策,李云泽详解稳楼市稳股市“组合拳”