【Rust】使用Rust实现一个简单的shell

一、Rust

Rust是一门系统编程语言,由Mozilla开发并开源,专注于安全、速度和并发性。它的主要目标是解决传统系统编程语言(如C和C++)中常见的内存安全和并发问题,同时保持高性能和底层控制能力。

Rust的特点包括:

  1. 内存安全:Rust通过其强大的所有权系统来确保内存安全,避免了空指针引用、悬挂指针和数据竞争等常见问题。编译器在编译时检查所有权规则,确保内存的正确管理,从而避免了运行时错误。

  2. 零成本抽象:Rust的设计哲学之一是提供零成本抽象,即高级特性(如泛型、闭包和模式匹配)在运行时不会引入额外的性能开销。这使得Rust能够在保持高级语言特性的同时,达到与C和C++相当的性能水平。

  3. 并发性:Rust内置了对并发编程的支持,通过其独特的所有权系统和借用检查器来防止数据竞争和其他并发问题。Rust的并发模型基于消息传递和锁,同时提供了异步编程的原生支持。

  4. 函数式编程元素:Rust融合了函数式编程的元素,如不可变性、纯函数、高阶函数和闭包。这些特性有助于编写更简洁、更可维护的代码,并减少状态管理和副作用带来的复杂性。

  5. 静态类型系统:Rust具有强大的静态类型系统,可以在编译时捕获许多类型错误。类型推断功能使得代码更简洁,同时保持了类型安全。

  6. 工具链和生态系统:Rust拥有一个完善的工具链和生态系统,包括包管理器Cargo、文档生成器rustdoc、构建工具rustc等。此外,Rust社区活跃,拥有大量的第三方库和框架,支持各种应用场景。

Rust适用于多种场景,包括系统级编程、网络服务、嵌入式开发、游戏开发等。由于其高性能和内存安全特性,Rust也被越来越多的公司采用,用于构建高性能、可靠的软件系统。

二、安装Rust和Hello World程序

安装Rust

使用官方安装脚本(推荐)安装。Rust 提供了一个方便的安装脚本,可以通过以下命令在 Unix-like 系统中使用:

curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh

这将下载一个脚本,并在本地执行,以安装 Rust 工具链。运行此脚本时,它会询问几个问题并安装 rustup,这是 Rust 版本管理器和安装工具。通过 rustup 将可以安装 Rust 编译器 rustc、Rust 包管理工具 cargo 和标准库。

安装完成后,可以通过运行以下命令来确保 Rust 安装成功:

rustc --version

这将输出已安装 Rust 编译器的版本信息。

为了升级现有的 Rust 版本,可以使用以下命令:

rustup update

Hello World程序

  • 创建一个新的Rust项目:cargo new hello_world

  • src/main.rs文件中编写Hello World程序:

fn main() {  println!("Hello, World!");  
}
  • 运行程序:在项目根目录下运行cargo run

三、一个简单的shell

制作一个简单的shell是一项很好的学习项目,尤其是当涉及到Unix系统编程和Rust的系统级能力时。以下是一个使用Rust实现的简单shell的基础框架。

这仅仅是起始点。一个完整的shell会涉及更多复杂的特性和错误处理。

首先,创建一个新的Rust项目:cargo new rust_shell。

Cargo.toml文件:

[package]
name = "rust_shell"
version = "0.1.0"
edition = "2021"# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html[dependencies]

接着,在main.rs文件中,编写以下代码:

use std::io::{self, Write};
use std::process::{Command, Stdio};fn main() {// 打印欢迎信息println!("Welcome to MyShell");// 进入REPL(Read-Evaluate-Print-Loop)循环loop {// 打印提示符print!("> ");io::stdout().flush().unwrap(); // 确保立即输出提示符// 获取用户输入let mut input = String::new();match io::stdin().read_line(&mut input) {Ok(_) => {// 移除字符串末端的换行符let input = input.trim_end();// 如果用户输入"exit",退出程序if input == "exit" {break;}// 解析命令和参数let parts: Vec<&str> = input.split_whitespace().collect();if parts.is_empty() {continue;}let command = parts[0];let args = &parts[1..];// 执行命令execute_command(command, args);}Err(error) => {eprintln!("Error reading input: {}", error);}}}
}fn execute_command(command: &str, args: &[&str]) {match Command::new(command).args(args).stdin(Stdio::inherit()).stdout(Stdio::inherit()).stderr(Stdio::inherit()).spawn() {Ok(mut child) => {// 等待命令执行完成match child.wait() {Ok(status) => {println!("Process exited with status: {:?}", status);}Err(e) => {eprintln!("Failed to wait on child: {}", e);}}}Err(e) => {eprintln!("Failed to execute command: {}", e);}}
}

在这段代码中,我们创建了一个基本的REPL循环,用户可以输入命令,然后我们解析这些命令并用`std::process::Command`去执行。每个命令在它自己的子进程中执行,而shell等待直到子进程结束。

要运行这个rust_shell,只需构建并运行项目。如果使用Cargo(Rust的构建系统和包管理器),在项目目录下运行以下命令:

cargo run

然后将能在自制shell中键入命令,比如`ls`或`echo Hello, world!`。

这个shell是非常简单的,没有实现像管道、重定向、变量扩展或是流程控制等shell的高级特性。在创建一个能和成熟shell(如bash或zsh)竞争的程序方面还有很长的路要走,但这个基本版本足够用于理解如何在Rust中开始这类型的项目。如果打算扩展这个基本shell,可能需要研究诸如异步IO、信号处理、作业控制等更复杂的概念。

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

车载诊断协议DoIP系列 —— 车辆以太网节点需求汇总

车载诊断协议DoIP系列 —— 车辆以太网节点需求汇总 我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师(Wechat:gongkenan2013)。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 本就是小人物,输了就是输了,不要在意别人怎么看自己。江湖一碗茶,…

《CSS 简易速速上手小册》第7章:CSS 预处理器与框架(2024 最新版)

文章目录 7.1 Sass&#xff1a;更高效的 CSS 编写7.1.1 基础知识7.1.2 重点案例&#xff1a;主题颜色和字体管理7.1.3 拓展案例 1&#xff1a;响应式辅助类7.1.4 拓展案例 2&#xff1a;深色模式支持 7.2 Bootstrap&#xff1a;快速原型设计和开发7.2.1 基础知识7.2.2 重点案例…

【NLP】MHA、MQA、GQA机制的区别

Note LLama2的注意力机制使用了GQA。三种机制的图如下&#xff1a; MHA机制&#xff08;Multi-head Attention&#xff09; MHA&#xff08;Multi-head Attention&#xff09;是标准的多头注意力机制&#xff0c;包含h个Query、Key 和 Value 矩阵。所有注意力头的 Key 和 V…

排序算法---计数排序

原创不易&#xff0c;转载请注明出处。欢迎点赞收藏~ 计数排序&#xff08;Counting Sort&#xff09;是一种线性时间复杂度的排序算法&#xff0c;其核心思想是通过统计待排序元素的个数来确定元素的相对位置&#xff0c;从而实现排序。 具体的计数排序算法步骤如下&#xff…

基于Python实现Midjourney集成到(个人/公司)平台中

目前Midjourney没有对外开放Api&#xff0c;想体验他们的服务只能在discord中进入他们的频道进行体验或者把他们的机器人拉入自己创建的服务器中&#xff1b;而且现在免费的也用不了了&#xff0c;想使用就得订阅。本教程使用midjourney-api这个开源项目&#xff0c;搭建Midjou…

鸿蒙开发-HarmonyOS UI架构

初步布局Index 当我们新建一个工程之后&#xff0c;首先会进入Index页。我们先简单的做一个文章列表的显示 class Article {title?: stringdesc?: stringlink?: string }Entry Component struct Index {State articles: Article[] []build() {Row() {Scroll() {Column() …

Web项目利用MybatisPlus进行分页查询

之前在写博客系统前台页面的时候&#xff0c;遇到了利用mp进行分页查询的情况&#xff0c;由于涉及到的知识点相对较为重要&#xff0c;固写一篇博客以此巩固。 一、功能需求 在首页和分类页面都需要查询文章列表。 首页&#xff1a;查询所有的文章分类页面&#xff1a;查询…

云计算基础-网络虚拟化

虚拟交换机 什么是虚拟交换机 虚拟交换机是一种运行在虚拟化环境中的网络设备&#xff0c;其运行在宿主机的内存中&#xff0c;通过软件方式在宿主机内部实现了部分物理交换机的功能&#xff0c;如 VLAN 划分、流量控制、QoS 支持和安全功能等网络管理特性 虚拟交换机在云平…

Invalid DataSize: cannot convert ‘30Mb‘ to Long

Invalid DataSize: cannot convert 30Mb to Long servlet:multipart:max-file-size: 30MBmax-request-size: 30MB

Mysql5.6忘记密码,如何找回(windows)

mysql5.6安装 第一步&#xff1a;关闭正在运行的数据库服务 net stop mysql第二步&#xff1a;在my.ini文件当中的[mysqld] 任意一个位置放入 skip-grant-tables第三步&#xff1a;启动mysql服务 net start mysql第四步&#xff1a;服务启动成功后就可以登录了&#xff0c;…

数据库管理-第149期 Oracle Vector DB AI-01(20240210)

数据库管理149期 2024-02-10 数据库管理-第149期 Oracle Vector DB & AI-01&#xff08;20240210&#xff09;1 机器学习2 向量3 向量嵌入4 向量检索5 向量数据库5 专用向量数据库的问题总结 数据库管理-第149期 Oracle Vector DB & AI-01&#xff08;20240210&#xf…

Typora+PicGO+腾讯云COS做图床教程

文章目录 Typora&#xff0b;PicGO&#xff0b;腾讯云COS做图床教程一、为什么使用图床二、Typora、PicGO和腾讯云COS介绍三、下载Typora和PicGOTyporaPicGO 四、配置Typora、PicGO和腾讯云COS腾讯云COS配置PicGO配置Typora配置 Typora&#xff0b;PicGO&#xff0b;腾讯云COS做…

【教学类-19-08】20240214《ABAB式-规律黏贴18格-手工纸15*15CM-一页3种图案,A空,纵向、无边框》(中班)

背景需求 利用15*15CM手工纸制作AB色块手环&#xff08;手工纸自带色彩&#xff09;&#xff0c;一页3个图案&#xff0c;2条为一组&#xff0c;黏贴成一个手环 素材准备 代码展示 # # 作者&#xff1a;阿夏 # 时间&#xff1a;2024年2月14日 # 名称&#xff1a;正方形数字卡…

OpenTitan- 开源安全芯片横空出世

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗&#xff1f;订阅我们的简报&#xff0c;深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同&#xff0c;从行业内部的深度分析和实用指南中受益。不要错过这个机会&#xff0c;成为AI领…

【PyQt6]全屏截图实现的带密码锁屏功能

文章目录 1. 简介2. Demo 1. 简介 书接上回&#xff0c;PyQt6 使用 QScreen 实现了截图&#xff0c;如果用个控件全屏显示截图&#xff0c;嘿嘿&#xff0c;那就是简单的一个锁屏界面了&#xff0c;类似于一个假死界面&#xff0c;也挺有意思。 友情提醒 输入 quit 可以退出程…

前端vue 数字 字符串 丢失精度问题

1.问题 后端返回的数据 是这样的 一个字符串类型的数据 前端要想显示这个 肯定需要使用Json.parse() 转换一下 但是 目前有一个问题 转换的确可以 showId:1206381711026823172 有一个这样的字段 转换了以后 发现 字段成了1206381711026823200 精度直接丢了 原本的数据…

SAP PP学习笔记- 豆知识01 - 怎么查询既存品目

SAP系统当中已经有哪些品目要怎么查询呢&#xff1f; 1&#xff0c;MM60 品目一览 这里可以输入Plant&#xff0c;然后可以查询该工厂的所有品目。 2&#xff0c;SE16 > MARA MARA 品目一般Data&#xff0c;存放的是品目基本信息。 如果要查询该品目属于哪个Plant&#x…

[计算机网络]---序列化和反序列化

前言 作者&#xff1a;小蜗牛向前冲 名言&#xff1a;我可以接受失败&#xff0c;但我不能接受放弃 如果觉的博主的文章还不错的话&#xff0c;还请点赞&#xff0c;收藏&#xff0c;关注&#x1f440;支持博主。如果发现有问题的地方欢迎❀大家在评论区指正 目录 一、再谈协议…

杨中科 ASP.NET DI综合案例

综合案例1 需求说明 1、目的:演示DI的能力; 2、有配置服务、日志服务&#xff0c;然后再开发一个邮件发送器服务。可以通过配置服务来从文件、环境变量、数据库等地方读取配置&#xff0c;可以通过日志服务来将程序运行过程中的日志信息写入文件、控制台、数据库等。 3、说明…

linux-firewalld防火墙端口转发

目的:通过统一地址实现对外同一地址暴露 1.系统配置文件开启 ipv4 端口转发 echo "net.ipv4.ip_forward 1" >> /etc/sysctl.confsysctl -p 2.查看防火墙配置端口转发之前的状态 firewall-cmd --statefirewall-cmd --list-all 3.开启 IP 伪装 firewall-cm…