【Rust】Rust获取命令行参数以及IO操作

news2025/6/5 10:50:37

在这里插入图片描述

✨✨ 欢迎大家来到景天科技苑✨✨

🎈🎈 养成好习惯,先赞后看哦~🎈🎈

🏆 作者简介:景天科技苑
🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。
🏆《博客》:Rust开发,Python全栈,Golang开发,云原生开发,PyQt5和Tkinter桌面开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi,flask等框架,云原生K8S,linux,shell脚本等实操经验,网站搭建,数据库等分享。

所属的专栏:Rust语言通关之路
景天的主页:景天科技苑

在这里插入图片描述

文章目录

  • Rust获取命令行参数以及IO操作
    • 1、接收命令行参数
      • 1.1 读取参数值
      • 1.2 将参数值保存进变量
    • 2、Rust读取文件
      • 2.1. 基本文件读取
        • 2.1.1 一次性读取整个文件到字符串
        • 2.1.2 一次性读取整个文件到字节向量
      • 2.2. 逐行读取文件
        • 2.2.1 使用 BufReader 逐行读取
        • 2.2.2 使用 BufReader 读取字节块
      • 2.3 使用内存映射文件 (memmap2)
    • 3、Rust向文件中写内容
    • 4、Rust序列化与反序列化

Rust获取命令行参数以及IO操作

Rust作为一门系统编程语言,提供了强大且安全的I/O操作支持。与C/C++不同,Rust通过所有权系统和丰富的类型系统,在编译期就能避免许多常见的I/O错误。Rust的标准库std::io模块包含了大多数I/O功能,而std::fs模块则专门处理文件系统操作。

1、接收命令行参数

1.1 读取参数值

Rust 标准库提供的std::env::args 能够获取传递给它的命令行参数的值
这个函数返回一个传递给程序的命令行参数的 迭代器(iterator)。

use std::env;

fn main() {
    let args: Vec<String> = env::args().collect();
    println!("{:?}", args);
}

首先使用 use 语句来将 std::env 模块引入作用域以便可以使用它的 args 函数。
注意 std::env::args 函数被嵌套进了两层模块中。
当所需函数嵌套了多于一层模块时,通常将父模块引入作用域,而不是其自身。
这便于我们利用 std::env 中的其他函数。这比增加了 use std::env::args; 后仅仅使用 args 调用函数要更明确一些,因为 args 容易被错认成一个定义于当前模块的函数。

得到的第一个参数是生成的程序本身,后面都是命令行参数
在这里插入图片描述

在 main 函数的第一行,我们调用了 env::args,并立即使用 collect 来创建了一个包含迭代器所有值的 vector。
collect 可以被用来创建很多类型的集合,所以这里显式注明 args 的类型来指定我们需要一个字符串 vector。
虽然在 Rust 中我们很少会需要注明类型,然而 collect 是一个经常需要注明类型的函数,因为 Rust 不能推断出你想要什么类型的集合。
最后,我们使用调试格式 :? 打印出 vector。

注意 vector 的第一个值是 “target\debug\myargs.exe”,它是我们二进制文件的名称。这与 C 中的参数列表的行为相匹配,让程序使用在执行时调用它们的名称。
如果要在消息中打印它或者根据用于调用程序的命令行别名更改程序的行为,通常可以方便地访问程序名称,不过考虑到本章的目的,我们将忽略它并只保存所需的两个参数。

注意args 函数和无效的 Unicode
注意 std::env::args 在其任何参数包含无效 Unicode 字符时会 panic。
如果你需要接受包含无效 Unicode 字符的参数,使用 std::env::args_os 代替。这在某些操作系统中可能发生
这个函数返回 OsString 值而不是 String 值。
OsString 值每个平台都不一样而且比 String 值处理起来更为复杂。

use std::env;

fn main() {
    for arg in env::args_os() {
        println!("{:?}", arg);
    }
}

在这里插入图片描述

1.2 将参数值保存进变量

打印出参数 vector 中的值展示了程序可以访问指定为命令行参数的值。现在需要将这两个参数的值保存进变量这样就可以在程序的余下部分使用这些值了。

use std::env;

fn main() {
    //获取命令行参数
    let args: Vec<String> = env::args().collect();

    //获取查询字符串和文件名
    let query = &args[1];
    let filename = &args[2];

    println!("Searching for {}", query);
    println!("In file {}", filename);
}

正如之前打印出 vector 时所看到的,程序的名称占据了 vector 的第一个值 args[0],所以我们从索引 1 开始。
myargs2 获取的第一个参数是需要搜索的字符串,所以将其将第一个参数的引用存放在变量 query 中。第二个参数将是文件名,所以将第二个参数的引用放入变量 filename 中。
在这里插入图片描述

2、Rust读取文件

在 Rust 中读取文件是常见的 I/O 操作,标准库提供了多种方法来处理文件读取。下面我将详细介绍各种文件读取方法及其适用场景

2.1. 基本文件读取

2.1.1 一次性读取整个文件到字符串
use std::fs;

use std::io;

//读取文件内容
fn read_file(filename: &str) -> io::Result<String> {
    // 使用?来处理错误
    // ?会将错误传递给调用者
    // ?只能在返回Result的函数中使用
    // ?会将错误类型从io::Error转换为io::Result<String>
    //fs::read_to_string返回Result<String, io::Error>
    let content = fs::read_to_string(filename)?;
    Ok(content)
}

fn main() {
    let filename = "hello.txt";
    match read_file(filename) {
        Ok(content) => println!("File content: {}", content),
        Err(e) => println!("Error: {}", e),
    }
}

在这里插入图片描述

2.1.2 一次性读取整个文件到字节向量

用于读取二进制文件

//读取文件到字节向量
use std::fs;
use std::io;

//读取二进制文件内容
fn read_file(filename: &str) -> io::Result<Vec<u8>> {
    // 使用?来处理错误
    // ?会将错误传递给调用者
    // ?只能在返回Result的函数中使用
    // ?会将错误类型从io::Error转换为io::Result<Vec<u8>>
    //fs::read返回Result<Vec<u8>, io::Error>
    let content = fs::read(filename)?;
    Ok(content)
}

fn main() {
    let filename = "123.vip";
    match read_file(filename) {
        //打印文件的字节长度
        Ok(content) => println!("File clength: {}", content.len()),
        Err(e) => println!("Error: {}", e),
    }
}

在这里插入图片描述

2.2. 逐行读取文件

对于大文件,逐行读取更高效且内存友好。

2.2.1 使用 BufReader 逐行读取
//读取大文件
use std::fs::File;
use std::io::{ BufRead, BufReader };

//读取大文件内容
fn read_file(filename: &str) -> std::io::Result<()> {
    let file = File::open(filename)?;
    let reader = BufReader::new(file);

    for line in reader.lines() {
        println!("{}", line?);
    }

    Ok(())
}

fn main() {
    let filename = "hello.txt";
    match read_file(filename) {
        Ok(()) => println!("File content: {}", filename),
        Err(e) => println!("Error: {}", e),
    }
}

在这里插入图片描述

特点:
内存效率高
自动处理换行符
lines() 返回 Result<String>,需要处理可能的错误

2.2.2 使用 BufReader 读取字节块
//使用 BufReader 读取字节块
use std::io::{ self, Read };
use std::fs::File;

//读取字节块
fn read_file(filename: &str) -> io::Result<()> {
    let file = File::open(filename)?;
    let mut reader = io::BufReader::new(file);

    //定义一个字节缓冲区
    //缓冲区的大小为1024字节
    let mut buffer = [0; 10];
    //循环读取字节块
    //read方法会将字节块读取到缓冲区中
    //read方法返回读取的字节数
    //如果读取到文件末尾,就退出循环
    loop {
        let n = reader.read(&mut buffer)?;
        if n == 0 {
            break; //读取到文件末尾,退出循环
        }
        println!("Read {} bytes: {:?}", n, &buffer[..n]);
        // 处理 buffer[..n]
    }

    Ok(())
}

fn main() {
    match read_file("Cargo.toml") {
        Ok(()) => println!("Read file successfully"),
        Err(e) => println!("Error: {}", e),
    }
}

在这里插入图片描述

特点:
适用于二进制文件
可以控制缓冲区大小
适合网络传输或处理大文件

2.3 使用内存映射文件 (memmap2)

对于超大文件,内存映射可以提高性能:

use std::fs::File;
use memmap2::Mmap;

fn main() -> std::io::Result<()> {
    let file = File::open("Cargo.toml")?;
    //使用memmap2映射文件到内存,需要使用unsafe,需要注意
    let mmap = unsafe { Mmap::map(&file)? };

    // 假设我们搜索某个字节模式
    let pattern = b"\xDE\xAD\xBE\xEF";
    if let Some(pos) = mmap.windows(pattern.len()).position(|w| w == pattern) {
        println!("Pattern found at offset: {}", pos);
    } else {
        println!("Pattern not found");
    }

    Ok(())
}

注意:内存映射涉及 unsafe 代码,需要谨慎使用。

3、Rust向文件中写内容

//写入文件
use std::fs;
use std::io;

fn write_file(filename: &str, content: &str) -> io::Result<()> {
    fs::write(filename, content)
}

fn main() {
    let filename = "hello.txt";
    let content = "你好,景天";
    match write_file(filename, content) {
        Ok(()) => println!("Write file successfully"),
        Err(e) => println!("Error: {}", e),
    }
}

在这里插入图片描述

4、Rust序列化与反序列化

使用 serde 库可以方便地进行 I/O 操作:
需要安装第三方库
cargo add serde
cargo add serde_json
并且在Cargo.toml中配置
在这里插入图片描述

//序列化和反序列化
use serde::{ Serialize, Deserialize };
use std::fs;
use serde_json;

#[derive(Serialize, Deserialize, Debug)]
#[allow(dead_code)]
struct Person {
    name: String,
    age: u8,
    phones: Vec<String>,
}

fn main() {
    let person = Person {
        name: "John".to_string(),
        age: 30,
        phones: vec!["1234567890".to_string(), "0987654321".to_string()],
    };

    //序列化
    let json = serde_json::to_string(&person).unwrap(); //将结构体序列化为json字符串
    println!("json: {}", json);
    //将序列化后的数据写入文件
    fs::write("person.json", json).unwrap(); //将数据写入文件,json的所有权已经移动到write中了

    //反序列化
    // let person2: Person = serde_json::from_str(&json).unwrap(); //这里json所有权已不存在
    // println!("person2: {:?}", person2);

    //从文件中读取数据并反序列化
    let json2 = fs::read_to_string("person.json").unwrap();
    let person3: Person = serde_json::from_str(&json2).unwrap();
    println!("person3: {:?}", person3);
}

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2397377.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Redis】Zset 有序集合

文章目录 常用命令zaddzcardzcountzrange && zrevrangezrangebyscorezpopmax && bzpopmaxzpopmin && zpopmaxzrank && zrevrankzscorezremzremrangebyrankzremrangebyscorezincrby 集合间操作交集 zinterstore并集 zunionstore 内部编码应用场…

manus对比ChatGPT-Deep reaserch进行研究类论文数据分析!谁更胜一筹?

目录 没有账号&#xff0c;只能挑选一个案例 1、manus的效果 Step-1&#xff1a;直接看结果 Step-2&#xff1a;看看其他文件的细节 Step-3&#xff1a;看最终报告 2、Deep reaserch 3、Deep reaserch进行行业分析 总结一下&#xff1a; 大家好这里是学术Anan&#xff…

【 HarmonyOS 5 入门系列 】鸿蒙HarmonyOS示例项目讲解

【 HarmonyOS 5 入门系列 】鸿蒙HarmonyOS示例项目讲解 一、前言&#xff1a;移动开发声明式 UI 框架的技术变革 在移动操作系统的发展历程中&#xff0c;UI 开发模式经历了从命令式到声明式的重大变革。 根据华为开发者联盟 2024 年数据报告显示&#xff0c;HarmonyOS 设备…

用提示词写程序(3),VSCODE+Claude3.5+deepseek开发edge扩展插件V2

edge扩展插件;筛选书签,跳转搜索,设置背景 链接: https://pan.baidu.com/s/1nfnwQXCkePRnRh5ltFyfag?pwd86se 提取码: 86se 导入解压的扩展文件夹: 导入扩展成功: edge扩展插件;筛选书签,跳转搜索,设置背景

初识PS(Photoshop)

初识PS&#xff08;Photoshop&#xff09; 1、Photoshop界面 2、常用快捷键

go语言的GMP(基础)

1.概念梳理 1.1线程 通常语义中的线程&#xff0c;指的是内核级线程&#xff0c;核心点如下&#xff1a; &#xff08;1&#xff09;是操作系统最小调度单元&#xff1b; &#xff08;2&#xff09;创建、销毁、调度交由内核完成&#xff0c;cpu 需完成用户态与内核态间的切…

电路图识图基础知识-高、低压供配电系统电气系统的继电自动装置(十三)

电气系统的继电自动装置 在供电系统中为保证系统的可靠性&#xff0c;保证重要负荷的不间断供电&#xff0c;常采用自动重合闸装置和备用电源自动投入装置。 1 自动重合闸装置 供配电系统多年运行实践表明&#xff0c;架空线路发生的故障多属于暂时性故障&#xff0c;如雷击…

Qt实现的水波进度条和温度进度条

一.效果 二.原理 1.水波 要模拟波浪,就要首先画出一条波浪线,正弦余弦曲线就很适合。 y=A*sin(ω*x+φ)+k y=A*cos(ω*x+φ)+k 这是正弦余弦曲线的公式,要想实现水波效果,那需要两条曲线,一条曲线的波峰对着另外一条曲线的波谷,要实现这样的曲线效果,只有让正弦曲线前移…

WEBSTORM前端 —— 第3章:移动 Web —— 第4节:移动适配-VM

目录 一、适配方案 二、VM布局 ​编辑 三、vh布局 四、案例—酷我音乐 一、适配方案 二、VM布局 三、vh布局 四、案例—酷我音乐

【Zephyr 系列 3】多线程与调度机制:让你的 MCU 同时干多件事

好的,下面是Zephyr 系列第 3 篇:聚焦 多线程与调度机制的实践应用,继续面向你这样的 Ubuntu + 真板实战开发者,代码清晰、讲解通俗、结构规范,符合 CSDN 高质量博客标准。 🧠关键词:Zephyr、线程调度、k_thread、k_sleep、RTOS、BluePill 📌适合人群:想从裸机开发进…

Kotlin-特殊类型

文章目录 数据类型枚举类型匿名类和伴生对象单例类伴生对象 数据类型 声明一个数据类非常简单: //在class前面添加data关键字表示为一个数据类 data class Student(var name: String, var age: Int)数据类声明后,编译器会根据主构造函数中声明的所有属性自动为其生成以下函数…

nssctf第二题[SWPUCTF 2021 新生赛]简简单单的逻辑

这是题目&#xff0c;下载后得到一个python文件,打开 解读代码&#xff1a; for i in range(len(list)):key (list[i]>>4)((list[i] & 0xf)<<4)result str(hex(ord(flag[i])^key))[2:].zfill(2)list[i]>>4&#xff1a;从列表中取数字同时高4位向右位…

《Discuz! X3.5开发从入门到生态共建》第3章 Discuz! X3.5 核心目录结构解析-优雅草卓伊凡

《Discuz! X3.5开发从入门到生态共建》第3章 Discuz! X3.5 核心目录结构解析-优雅草卓伊凡 3.1 系统核心目录结构 Discuz! X3.5采用模块化设计&#xff0c;主要目录结构如下&#xff1a; discuz_root/ ├─ api/ // API接口目录 ├─ config/ …

【HarmonyOS 5】鸿蒙应用实现发票扫描、文档扫描输出PDF图片或者表格的功能

【HarmonyOS 5】鸿蒙应用实现发票扫描、文档扫描输出PDF图片或者表格的功能 一、前言 图(1-1) HarmonyOS 系统提供的核心场景化视觉服务,旨在帮助开发者快速实现移动端文档数字化功能。 其核心能力包括:扫描合同、票据、会议记录并保存为 PDF 分享。拍摄课堂 PPT、书籍章…

Python_day43

DAY 43 复习日 作业&#xff1a; kaggle找到一个图像数据集&#xff0c;用cnn网络进行训练并且用grad-cam做可视化 进阶&#xff1a;并拆分成多个文件 关于 Dataset 从谷歌图片中抓取了 1000 多张猫和狗的图片。问题陈述是构建一个模型&#xff0c;该模型可以尽可能准确地在图像…

STM32CubeDAC及DMA配置

STM32CubeDAC及DMA配置 一&#xff0c;问题1二&#xff0c;解决11&#xff0c;宏观思路CubeMX配置2&#xff0c;HAL_TIM_Base_Start(&htim6) 的作用1&#xff0c;作用1&#xff1a;使能TIM6的时钟并让它开始计数2&#xff0c;作用2&#xff1a;当 TIM6 溢出时&#xff0c;会…

行业案例 | OPPO借助Azure AI Speech国际服务实现音频文件智能转录

OPPO是全球领先的智能终端与移动互联网服务提供商&#xff0c;业务覆盖50余国&#xff0c;通过超40万销售网点和2500个服务中心与全球用户共享科技。作为软硬服一体化科技公司&#xff0c;OPPO以ColorOS为核心优化软件平台&#xff0c;为4.4亿月活用户打造智能操作系统&#xf…

基于 OpenCV 和 DLib 实现面部特征调整(眼间距、鼻子、嘴巴)

摘 要 本文介绍如何利用Dlib面部特征点检测和OpenCV图像处理技术&#xff0c;通过Python实现面部特征的精准调整。我们将以改变眼间距为例&#xff0c;演示包括地标检测、三角剖分变形等关键技术&#xff0c;该方法可扩展至嘴唇、眉毛等面部特征的调整。 技术栈 Python 3.8 …

spring-boot接入websocket教程以及常见问题解决

我们使用spring-boot接入websocket有三种方式&#xff1a;使用EnableWebSocket、EnableWebSocketMessageBroker以及ServerEndpoint&#xff0c;本文主要介绍使用ServerEndpoint方式的流程以及碰到的问题解决 接入方式 添加依赖 确保spring-boot-starter-websocket依赖 <d…

迈向分布式智能:解析MCP到A2A的通信范式迁移

智能体与外部世界的桥梁之言&#xff1a; 在深入探讨智能体之间的协作机制之前&#xff0c;我们有必要先厘清一个更基础的问题&#xff1a;**单个智能体如何与外部世界建立连接&#xff1f;** 这就引出了我们此前介绍过的 **MCP&#xff08;Model Context Protocol&…