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

news2024/10/6 18:27:18

一、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.coloradmin.cn/o/1446308.html

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

相关文章

一周学会Django5 Python Web开发-Django5操作命令

锋哥原创的Python Web开发 Django5视频教程&#xff1a; 2024版 Django5 Python web开发 视频教程(无废话版) 玩命更新中~_哔哩哔哩_bilibili2024版 Django5 Python web开发 视频教程(无废话版) 玩命更新中~共计11条视频&#xff0c;包括&#xff1a;2024版 Django5 Python we…

2024年P气瓶充装证模拟考试题库及P气瓶充装理论考试试题

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 2024年P气瓶充装证模拟考试题库及P气瓶充装理论考试试题是由安全生产模拟考试一点通提供&#xff0c;P气瓶充装证模拟考试题库是根据P气瓶充装最新版教材&#xff0c;P气瓶充装大纲整理而成&#xff08;含2024年P气瓶…

单例模式:懒汉饿汉线程安全问题

在我们前几篇文章中都了解了一些关于线程的知识&#xff0c;那么在多线程的情况下如何创建单例模式&#xff0c;其中的线程安全问题如何解决&#xff1f; 目录 1.什么是单例模式&#xff1f; (饿汉模式) 2.单例模式(懒汉模式) *懒汉模式与懒汉模式的对比 *如何解决懒汉模式…

VTK 正交投影 透视投影

1.VTK默认透视投影&#xff08;近大远小&#xff09;&#xff1b; 1.调用vtkCamera的ParallelProjectionOn函数开启 2.通过vtkCamera的SetParallelScale缩放 3.通过vtkCamera的SetClippingRange设置前后裁剪平面 2.正交投影&#xff08;平行投影&#xff0c;远近一样&#xf…

Netty Review - NioEventLoopGroup源码解析

文章目录 概述类继承关系源码分析小结 概述 EventLoopGroup bossGroup new NioEventLoopGroup(1); EventLoopGroup workerGroup new NioEventLoopGroup();这段代码是在使用Netty框架时常见的用法&#xff0c;用于创建两个不同的EventLoopGroup实例&#xff0c;一个用于处理连…

Linux第51步_移植ST公司的linux内核第3步_添加修改设备树

1、设备树文件的路径 1)、创建linux中的设备树头文件 在“my_linux/linux-5.4.31/arch/arm/boot/dts/”目录中&#xff0c;以“stm32mp15xx-edx.dtsi”为蓝本&#xff0c;复制一份&#xff0c;并命名为 “stm32mp157d-atk.dtsi”&#xff0c;这就是我们开发板的设备树头文件。…

【DDD】学习笔记-四色建模法

或许正是认识到彩色 UML 在建模过程的不足之处&#xff0c;ThoughtWorks 的徐昊才在彩色 UML 基础之上提出了自己的“四色建模法”。可考的四色建模法资料仅见于徐昊在 InfoQ 上发表的文章运用四色建模法进行领域分析。在这篇文章中&#xff0c;徐昊回答了建模活动的一个关键问…

【开源】基于JAVA+Vue+SpringBoot的房屋出售出租系统

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 房屋销售模块2.2 房屋出租模块2.3 预定意向模块2.4 交易订单模块 三、系统展示四、核心代码4.1 查询房屋求租单4.2 查询卖家的房屋求购单4.3 出租意向预定4.4 出租单支付4.5 查询买家房屋销售交易单 五、免责说明 一、摘…

Kafka 之生产者(Producer)

目录 一. 前言 二. 生产消息 三. 幂等和事务 四. send() 发送消息 五. 原理解析 一. 前言 Kafka生产者是一个应用程序&#xff0c;它负责向 Kafka 主题发送消息。这些消息可以用于多种目的&#xff0c;如记录用户活动、收集物联网设备数据、保存日志消息或缓存即将写入数据…

《乱弹篇(十四)香火旺》

连日来&#xff0c;“大年初一烧香祈福&#xff0c;北京雍和宫人山人海”这一词条登上社交网站热搜&#xff0c;对这一现象的描述多为“初一凌晨 民众在雍和宫前排大队”&#xff0c;“大年初一&#xff0c;雍和宫内人山人海&#xff0c;烟雾缭绕”&#xff0c;“雍和宫迎来6万…

Asp .Net Core 系列:Asp .Net Core 集成 NLog

简介 NLog是一个基于.NET平台编写的日志记录类库&#xff0c;它可以在应用程序中添加跟踪调试代码&#xff0c;以便在开发、测试和生产环境中对程序进行监控和故障排除。NLog具有简单、灵活和易于配置的特点&#xff0c;支持在任何一种.NET语言中输出带有上下文的调试诊断信息…

三、案例 - MySQL数据迁移至ClickHouse

MySQL数据迁移至ClickHouse 一、生成测试数据表和数据1.在MySQL创建数据表和数据2.在ClickHouse创建数据表 二、生成模板文件1.模板文件内容2.模板文件参数详解2.1 全局设置2.2 数据读取&#xff08;Reader&#xff09;2.3 数据写入&#xff08;Writer&#xff09;2.4 性能设置…

ctfshow-文件上传(web151-web161)

目录 web151 web152 web153 web154 web155 web156 web157 web158 web159 web160 web161 web151 提示前台验证不可靠 那限制条件估计就是在前端设置的 上传php小马后 弹出了窗口说不支持的格式 查看源码 这一条很关键 这种不懂直接ai搜 意思就是限制了上传类型 允许…

计算机组成原理 1 概论

主要内容 介绍运算器、控制器、存储器结构、工作原理、设计方法及互连构成整机的技术。 主要内容&#xff1a; ◼ 数值表示与运算方法 ◼ 运算器的功能、组成和基本运行原理 ◼ 存储器及层次存储系统 ◼ 指令系统 ◼ CPU功能、组成和运行原理 ◼ 流水线 ◼ 系统总线 ◼ 输入输出…

UE5 播放本地MP3、MP4

1.创建一个媒体播放器 2.如创建视频&#xff0c;勾选。 它会多一个媒体纹理给你 3.1 设置音频 在一个actor上添加“媒体音频组件” “音频媒体播放器”赋值给它 3.2播放音频 添加一个音频媒体播放器变量&#xff0c; 赋值 地址使用绝对地址 4.1设置视频 UI上创建一个imag…

快速学习Spring

Spring 简介 Spring 是一个开源的轻量级、非侵入式的 JavaEE 框架&#xff0c;它为企业级 Java 应用提供了全面的基础设施支持。Spring 的设计目标是简化企业应用的开发&#xff0c;并解决 Java 开发中常见的复杂性和低效率问题。 Spring常用依赖 <dependencies><!-…

Linux:信号的处理

文章目录 信号处理 本篇总结的是关于信号的处理 信号处理 在之前有这样的观点&#xff1a;信号在合适的时候被处理好&#xff0c;当进程收到信号后&#xff0c;当前进程可能在做优先级更高的事&#xff0c;所以它来不及处理这个信号&#xff0c;那么就会把这个信号暂时保存起…

spring aop @annotation的用法

直接看原文: spring aop annotation的用法-CSDN博客 -------------------------------------------------------------------------------------------------------------------------------- annotation用在定义连接点时&#xff0c;对连接点进行限制。比如我们想对标注了…

双非本科准备秋招(18.2)—— 图解Monitor

对象头 普通对象&#xff1a; 数组对象&#xff1a; java中对象存储结构分为对象头&#xff08;Header&#xff09;、实例数据&#xff08;Instance Date&#xff09;和对齐填充&#xff08;Padding&#xff09;。 对象头存储着Mark Word和Klass Word&#xff0c;通过Klass Wo…

【MySQL】操作库 —— 库的操作 -- 详解

一、增删数据库 1、创建数据库 create database db_name; 本质就是在 /var/lib/mysql 创建一个目录。 说明&#xff1a; 大写的表示关键字。[ ] 是可选项。CHARACTER SET&#xff1a;指定数据库采用的字符集。COLLATE&#xff1a;指定数据库字符集的校验规则。 2、数据库删除…