Rust 基础教程

Rust 编程语言教程

Rust是一门注重安全、并发和性能的系统编程语言。本文将从Rust的基本语法、常用功能到高级特性,详细介绍Rust的使用方法。

目录

  1. 简介
  2. 环境配置
  3. 基础语法
    • 变量和常量
    • 数据类型
    • 函数
    • 控制流
  4. 所有权和借用
    • 所有权
    • 借用
  5. 结构体和枚举
    • 结构体
    • 枚举
  6. 模块和包
  7. 错误处理
  8. 并发编程
  9. 常用库和工具
  10. 示例项目

简介

Rust是一门由Mozilla开发的系统编程语言,它的设计目标是提供内存安全、并发编程和高性能。Rust借鉴了许多现代编程语言的优点,同时引入了独特的所有权系统,以确保内存安全和线程安全。

环境配置

要开始使用Rust,首先需要安装Rust编译器和包管理工具Cargo。

安装Rust

可以通过Rust的官方安装工具 rustup 安装Rust。打开终端并运行以下命令:

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

安装完成后,可以通过以下命令检查Rust是否安装成功:

rustc --version
cargo --version

创建第一个Rust项目

使用Cargo创建一个新的Rust项目:

cargo new hello_rust
cd hello_rust

项目目录结构如下:

hello_rust
├── Cargo.toml
└── src
    └── main.rs

Cargo.toml 文件是项目的配置文件,main.rs 文件是项目的入口点。

基础语法

变量和常量

Rust中的变量默认是不可变的,可以使用 let 关键字声明变量,使用 mut 关键字声明可变变量。常量使用 const 关键字声明。

fn main() {
    let x = 5; // 不可变变量
    let mut y = 10; // 可变变量
    const PI: f64 = 3.14159; // 常量
    println!("x: {}, y: {}, PI: {}", x, y, PI);
    y = 15;
    println!("Updated y: {}", y);
}

数据类型

Rust具有多种基本数据类型,包括标量类型和复合类型。

fn main() {
    // 标量类型
    let int_var: i32 = 10; // 整数
    let float_var: f64 = 3.14; // 浮点数
    let bool_var: bool = true; // 布尔值
    let char_var: char = 'R'; // 字符

    // 复合类型
    let tuple_var: (i32, f64, char) = (10, 3.14, 'R'); // 元组
    let array_var: [i32; 3] = [1, 2, 3]; // 数组

    println!("int_var: {}, float_var: {}, bool_var: {}, char_var: {}", int_var, float_var, bool_var, char_var);
    println!("tuple_var: {:?}", tuple_var);
    println!("array_var: {:?}", array_var);
}

函数

函数是Rust中的基本代码单元,使用 fn 关键字定义。

fn main() {
    let result = add(5, 10);
    println!("Result: {}", result);
}

fn add(a: i32, b: i32) -> i32 {
    a + b
}

控制流

Rust支持常见的控制流结构,如 ifloopwhilefor

fn main() {
    let number = 5;

    // if 表达式
    if number < 10 {
        println!("number is less than 10");
    } else {
        println!("number is 10 or greater");
    }

    // loop 循环
    let mut count = 0;
    loop {
        count += 1;
        if count == 3 {
            break;
        }
    }

    // while 循环
    while count < 5 {
        count += 1;
    }

    // for 循环
    for i in 0..5 {
        println!("i: {}", i);
    }
}

所有权和借用

所有权

Rust的所有权系统是其核心特性,确保内存安全。

fn main() {
    let s1 = String::from("hello");
    let s2 = s1; // s1的所有权被移动到s2
    // println!("{}", s1); // 这行代码将导致编译错误
    println!("{}", s2);
}

借用

借用可以让我们在不转移所有权的情况下使用变量。

fn main() {
    let s1 = String::from("hello");
    let len = calculate_length(&s1); // 借用s1
    println!("The length of '{}' is {}.", s1, len);
}

fn calculate_length(s: &String) -> usize {
    s.len()
}

结构体和枚举

结构体

结构体用于定义复杂的数据类型。

struct User {
    username: String,
    email: String,
    sign_in_count: u64,
    active: bool,
}

fn main() {
    let user1 = User {
        username: String::from("someone"),
        email: String::from("someone@example.com"),
        sign_in_count: 1,
        active: true,
    };

    println!("Username: {}", user1.username);
}

枚举

枚举用于定义一组可能的值。

enum Message {
    Quit,
    Move { x: i32, y: i32 },
    Write(String),
    ChangeColor(i32, i32, i32),
}

fn main() {
    let msg = Message::Write(String::from("Hello"));
    match msg {
        Message::Quit => println!("Quit"),
        Message::Move { x, y } => println!("Move to ({}, {})", x, y),
        Message::Write(text) => println!("Text: {}", text),
        Message::ChangeColor(r, g, b) => println!("Change color to ({}, {}, {})", r, g, b),
    }
}

模块和包

模块和包用于组织代码。

mod network {
    pub fn connect() {
        println!("Network connected");
    }

    pub mod server {
        pub fn start() {
            println!("Server started");
        }
    }
}

fn main() {
    network::connect();
    network::server::start();
}

错误处理

Rust使用 ResultOption 类型进行错误处理。

fn main() {
    let result = divide(4.0, 2.0);
    match result {
        Ok(v) => println!("Result: {}", v),
        Err(e) => println!("Error: {}", e),
    }
}

fn divide(a: f64, b: f64) -> Result<f64, String> {
    if b == 0.0 {
        Err(String::from("Division by zero"))
    } else {
        Ok(a / b)
    }
}

并发编程

Rust通过线程和消息传递模型支持并发编程。

use std::thread;
use std::sync::mpsc;

fn main() {
    let (tx, rx) = mpsc::channel();

    thread::spawn(move || {
        tx.send(String::from("Hello from thread")).unwrap();
    });

    let received = rx.recv().unwrap();
    println!("Received: {}", received);
}

常用库和工具

Rust生态系统中有许多常用库和工具,例如 serde 用于序列化和反序列化,tokio 用于异步编程。

serde

# Cargo.toml
[dependencies]
serde = { version = "1.0", features = ["derive"] }
serde_json = "1.0"
use serde::{Serialize, Deserialize};

#[derive(Serialize, Deserialize)]
struct Person {
    name: String,
    age: u8,
}

fn main() {
    let p = Person {
        name: String::from("Alice"),
        age: 30,
    };

    let json = serde_json::to_string(&p).unwrap();
    println!("JSON: {}", json);

    let p2: Person = serde_json::from_str(&json

).unwrap();
    println!("Name: {}, Age: {}", p2.name, p2.age);
}

示例项目

最后,我们通过一个简单的示例项目来巩固学到的知识。

项目描述

创建一个命令行程序,它接受用户输入,并计算输入字符串的单词数量。

项目结构

word_count
├── Cargo.toml
└── src
    └── main.rs

代码实现

use std::env;
use std::fs;

fn main() {
    let args: Vec<String> = env::args().collect();
    let filename = &args[1];

    let contents = fs::read_to_string(filename)
        .expect("Something went wrong reading the file");

    let word_count = contents.split_whitespace().count();

    println!("The file '{}' contains {} words", filename, word_count);
}

运行项目

cargo run text.txt

以上就是Rust编程语言的详细教程,从基础语法到高级特性,希望对您学习Rust有所帮助。Rust是一门强大而高效的语言,掌握它将使您在系统编程和并发编程领域游刃有余。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Java8新特性之Optional、Lambda表达式、Stream、新日期Api使用总结

标题 Optional类常用Api使用说明Optional API 使用建议 Lambda表达式Lambda 表达式的局限性与解决方案 Stream案例实操优化 Stream 操作 新的日期和时间APILocalDateTime在SpringBoot中的应用 函数式接口&#xff08;Functional&#xff09; Optional博客参考地址1 Stream博客参…

【Linux从入门到放弃】探究进程如何退出以进程等待的前因后果

&#x1f9d1;‍&#x1f4bb;作者&#xff1a; 情话0.0 &#x1f4dd;专栏&#xff1a;《Linux从入门到放弃》 &#x1f466;个人简介&#xff1a;一名双非编程菜鸟&#xff0c;在这里分享自己的编程学习笔记&#xff0c;欢迎大家的指正与点赞&#xff0c;谢谢&#xff01; 进…

【C++】STL-priority_queue

目录 1、priority_queue的使用 2、实现没有仿函数的优先级队列 3、实现有仿函数的优先级队列 3.1 仿函数 3.2 真正的优先级队列 3.3 优先级队列放自定义类型 1、priority_queue的使用 priority_queue是优先级队列&#xff0c;是一个容器适配器&#xff0c;不满足先进先出…

pdf拆分,pdf拆分在线使用,pdf拆分多个pdf

在数字化的时代&#xff0c;pdf文件已经成为我们日常办公、学习不可或缺的文档格式。然而&#xff0c;有时候我们可能需要对一个大的pdf文件进行拆分&#xff0c;以方便管理和分享。那么&#xff0c;如何将一个pdf文件拆分成多个pdf呢&#xff1f;本文将为你推荐一种好用的拆分…

监控平台zabbix介绍与部署

目录 1.为什么要做监控 2.zabbix是什么&#xff1f; 3.zabbix 监控原理 4.Zabbix 6.0 新特性 5.Zabbix 6.0 功能组件 6.部署zabbix 6.1 部署 Nginx PHP 环境并测试 6.2 部署数据库 6.3 向数据库导入 zabbix 数据 6.4 编译安装 zabbix Server 服务端 6.5 修改 zabbix…

中小企业如何防止被查盗

在当前的商业环境中&#xff0c;小企业面临诸多挑战&#xff0c;其中之一便是如何在有限的预算内满足日常运营的技术需求。由于正版软件的高昂成本&#xff0c;一些小企业可能会选择使用盗版软件来降低成本。 我们联网之后存在很多风险&#xff0c;你可以打开自己的可以联网的电…

【面试系列】机器学习工程师高频面试题及详细解答

欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;欢迎订阅相关专栏&#xff1a; ⭐️ 全网最全IT互联网公司面试宝典&#xff1a;收集整理全网各大IT互联网公司技术、项目、HR面试真题. ⭐️ AIGC时代的创新与未来&#xff1a;详细讲解AIGC的概念、核心技术、…

Java--创建对象内存分析

1.如图所示&#xff0c;左边为一个主程序&#xff0c;拥有main方法&#xff0c;右边定义了一个Pet类&#xff0c;通过debug不难看出&#xff0c;当启动该方法时&#xff0c;有以下该步骤 1.运行左边的实例化Pet类对象 2.跳转至右边定义Pet类的语句 3.跳回至左边获取Pet类基本属…

代码生成器使用指南,JeecgBoot低代码平台

JeecgBoot 提供强大的代码生成器&#xff0c;让前后端代码一键生成&#xff0c;实现低代码开发。支持单表、树列表、一对多、一对一等数据模型&#xff0c;增删改查功能一键生成&#xff0c;菜单配置直接使用。 同时提供强大模板机制&#xff0c;支持自定义模板&#xff0c;目…

【bug报错已解决】ERROR: Could not find a version that satisfies the requirement

&#x1f3ac; 鸽芷咕&#xff1a;个人主页 &#x1f525; 个人专栏: 《C干货基地》《粉丝福利》 ⛺️生活的理想&#xff0c;就是为了理想的生活! 文章目录 引言一、问题描述1.1 报错示例1.2 报错分析 二、解决方法2.1 方法一2.2 方法二 三、总结 引言 有没有遇到过那种让人…

JSON JOLT常用示例整理

JSON JOLT常用示例整理 1、什么是jolt Jolt是用Java编写的JSON到JSON转换库&#xff0c;其中指示如何转换的"specification"本身就是一个JSON文档。以下文档中&#xff0c;我统一以 Spec 代替如何转换的"specification"json文档。以LHS(left hand side)代…

kaggle量化赛金牌方案(第七名解决方案)

获奖文章(第七名解决方案) 致谢 我要感谢 Optiver 和 Kaggle 组织了这次比赛。这个挑战提出了一个在金融市场时间序列预测领域中具有重大和复杂性的问题。 方法论 我的方法结合了 LightGBM 和神经网络模型,对神经网络进行了最少的特征工程。目标是结合这些模型以降低最终…

arco disign vue 日期组件的样式穿透

问题描述: 对日期组件进行样式穿透. 原因分析: 如图,日期组件被展开时它默认将dom元素挂载到body下, 我们的页面在idroot的div 里层, 里层想要穿透外层是万万行不通的. 解决问题: 其实官网提供了参数,但是并没有提供例子, 只能自己摸索着过河. 对于日期组件穿透样式,我们能…

收集了很久的全网好用的磁力搜索站列表分享

之前找资源的时候&#xff0c;收集了一波国内外大部分主流的磁力链接搜索站点。每一个站可能都有对应的优缺点&#xff0c;多试试&#xff0c;就能知道自己要哪个了。 全网好用的磁力链接 大部分的时候&#xff0c;我们用国内的就可以了&#xff0c;速度块&#xff0c;而且不…

Snappy使用

Snappy使用 Snappy是谷歌开源的压缩和解压的开发包&#xff0c;目标在于实现高速的压缩而不是最大的压缩 项目地址&#xff1a;GitHub - google/snappy&#xff1a;快速压缩器/解压缩器 Cmake版本升级 该项目需要比较新的cmake&#xff0c;CMake 3.16.3 or higher is requi…

51单片机第23步_定时器1工作在模式0(13位定时器)

重点学习51单片机定时器1工作在模式0的应用。 在51单片机中&#xff0c;定时器1工作在模式0&#xff0c;它和定时器0一样&#xff0c;TL1占低5位&#xff0c;TH1占高8位&#xff0c;合计13位&#xff0c;也是向上计数。 1、定时器1工作在模式0 1)、定时器1工作在模式0的框图…

8619 公约公倍

这个问题可以通过计算最大公约数 (GCD) 和最小公倍数 (LCM) 来解决。我们需要找到一个整数&#xff0c;它是 a, b, c 的 GCD 的倍数&#xff0c;同时也是 d, e, f 的 LCM 的约数。 以下是解决这个问题的步骤&#xff1a; 1. 计算 a, b, c 的最大公约数。 2. 计算 d, e, f 的最…

流处理系统对比:RisingWave vs ksqlDB

本文将从架构、部署与可扩展性、Source 和 Sink、生态系统与开发者工具几个方面比较 ksqlDB 和 RisingWave 这两款领先的流处理系统。 1. 架构 ksqlDB 是由 Confluent 开发和维护的流处理 SQL 引擎&#xff0c;专为 Apache Kafka 设计。它基于 Kafka Streams 构建&#xff0c;…

鸿蒙:路由Router原理

页面路由&#xff1a;在应用程序中实现不同页面之间的跳转和数据传递 典型应用&#xff1a;商品信息返回、订单等多页面跳转 页面栈最大容量为32个页面&#xff0c;当页面需要销毁可以使用router.clear()方法清空页面栈 router有两种页面跳转模式&#xff1a; router.pushUrl…

Golang 开发实战day15 - Input info

&#x1f3c6;个人专栏 &#x1f93a; leetcode &#x1f9d7; Leetcode Prime &#x1f3c7; Golang20天教程 &#x1f6b4;‍♂️ Java问题收集园地 &#x1f334; 成长感悟 欢迎大家观看&#xff0c;不执着于追求顶峰&#xff0c;只享受探索过程 Golang 开发实战day15 - 用户…