猜数字游戏(Rust实现)

文章目录

  • 游戏说明
  • 游戏效果展示
  • 游戏代码
  • 游戏代码详解
    • 生成神秘数字
    • 读取用户输入
    • 解析用户输入
    • 进行猜测比较

游戏说明

游戏说明

游戏运行逻辑如下:

  1. 随机生成一个1-100的数字作为神秘数字,并提示玩家进行猜测。
  2. 如果玩家猜测的数字小于神秘数字,则提示用户“猜测的数字太小了”。
  3. 如果玩家猜测的数字大于神秘数字,则提示用户“猜测的数字太大了”。
  4. 让玩家不断进行猜测,直到最终猜出神秘数字,游戏结束。

游戏效果展示

游戏效果展示

在这里插入图片描述

游戏代码

游戏代码

游戏完整代码如下:

use rand::Rng;
use std::io;
use std::cmp::Ordering;fn main() {println!("欢迎来到猜数游戏!");//1、生成神秘数字let secret_number = rand::thread_rng().gen_range(1, 101);println!("神秘数字已经生成!");loop {//2、让用户进行猜测println!("请猜测:>");let mut guess = String::new();io::stdin().read_line(&mut guess).expect("无法读取行");//3、将用户输入的数字字符串转化为整型let guess: u32 = match guess.trim().parse() {Ok(num) => num,Err(_) => {println!("请您输入一个合法的整数!");continue;}};//4、将用户猜测的数与神秘数字进行比较match guess.cmp(&secret_number) {Ordering::Less => println!("您猜测的数字太小了"),Ordering::Greater => println!("您猜测的数字太大了"),Ordering::Equal => {println!("恭喜您猜对了, 神秘数字就是{}!", secret_number);break;}}}
}

游戏代码详解

下面对猜数字游戏中所用到的Rust语法和包(crate)进行讲解。

生成神秘数字

rand包

  • Rust团队没有把随机数字生成功能内置到标准库中,而是选择将它作为rand包(rand crate)提供给用户。
  • Rust中的包(crate)代表了一系列源代码文件的集合,我们当前正在构建的项目是一个用于生成可执行程序的二进制包(binary crate),而我们引用的rand包则是一个用于复用功能的库包(library crate)。
  • rand包中有一个名为Rng的trait,它定义了随机数生成器需要实现的方法集合,比如Rng中的gen_range方法可以根据指定的范围来生成随机数。

在rand包中有一个名为thread_rng的方法,该方法会返回一个特定的随机数生成器,通过调用该随机数生成器的gen_range方法即可在指定范围内生成随机数。如下:

use rand::Rng;fn main() {let secret_number = rand::thread_rng().gen_range(1, 101); //在[1,101)范围生成随机数println!("生成的神秘数字是: {}", secret_number);
}

说明一下:

  • thread_rng方法返回的随机数生成器(ThreadRng类型)位于本地线程空间,并通过操作系统获得随机种子。
  • ThreadRng类型实际并没有实现gen_range方法,我们调用的实际是Rng trait中默认实现的gen_range方法,因此需要通过use语句将rand包中的Rng trait引入到当前作用域。
  • Rust中的trait可以类比其他语言中的接口的概念,它定义了一组方法,比如可以类比Golang中的interface,但trait的职责远比interface更多。
  • 代码中生成的随机数位与1-100之间,而Rust中很多整数类型都能涵盖这个范围,比如i32、u32、i64等,除非在代码中增加更多的信息用于类型推断,否则这个类型将会被视为i32类型,代码中定义secret_number变量时就没有指定它的类型,因此其为i32类型。
  • 代码中通过fn关键字声明了一个无参无返回值的main函数,通过let关键字定义一个secret_number变量来存储生成的随机值。
  • 代码中的println!是一个宏,其功能是将字符串格式化后打印到标准输出,格式化字符串中包含占位符{},从第二个参数开始,各参数依次替换格式化字符串中的占位符。

添加依赖

Cargo最主要的功能就是帮助我们管理和使用第三方库,在使用rand包编写代码之前,需要在Cargo.toml文件的[dependencies]片段下将rand包声明为依赖,并指明它的版本号。如下:

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

执行cargo build命令后,Cargo就会从crates.io网站下载并编译指定的包,并基于这些依赖编译我们自己的项目。如下:

在这里插入图片描述

说明一下:

  • 在Cargo.toml文件中指明的0.3.14实际上是^0.3.14的简写,它表示任何与0.3.14版本公共API相兼容的版本,因此我们下载不一定是我们指定版本号的包。
  • Cargo可以从注册表(registry)中获取所有可用rand包的最新版本信息,这些信息通常是从crates.io上复制过来的。(crates.io是用于分享各种各样开源Rust项目的网站)
  • Cargo会在更新完注册表后逐条检查[dependencies]片段下的依赖,并下载当前缺失的依赖包,由于rand包依赖于libc,因此在下载rand包的时候额外下载了一份libc数据。

Cargo.lock

Cargo提供了一套机制来保证我们构建的结果是可重现的,任何人是任何时候编译我们的代码都会生成相同的产物,因为Cargo会一直使用某个特定版本的依赖,直到我们手动指定了其他版本。

  • 当第一次使用cargo build构建项目时,在项目目录下会生成Cargo.lock文件,Cargo会依次遍历我们声明的依赖及其对应的语义化版本,找到符合要求的具体版本号,并将其写入Cargo.lock文件中。
  • 后续再次构建项目时,Cargo会先检索Cargo.lock文件,如果文件中存在已经指明具体版本的依赖库,那么它就会跳过计算版本号的过程,并直接使用文件中指明的版本,这就使得我们构建的结果是可重现的。

如果你确实需要升级某个依赖包,那么可以使用cargo update命令,该命令会强制Cargo忽略Cargo.lock文件,并重新计算出所有依赖包中符合Cargo.toml声明的最新版本。如下:

在这里插入图片描述

说明一下:

  • 如果cargo update命令运行成功,Cargo会将各个依赖包更新后的版本号写入Cargo.lock文件。
  • 在当前示例中,Cargo在自动升级时只会寻找大于0.3.0版本,并且小于0.4.0版本的最新版本,如果要将rand包的升级到0.4.x,那么就需要在Cargo.toml文件中指明对应的版本。
  • 当前示例执行cargo update命令后,Cargo.lock中的内容实际不会改变,因为之前构建项目的时候Cargo.lock中写入的就是0.3开头的最新的版本0.3.23。

读取用户输入

io模块

  • 标准库(std)中的io模块包含了许多有用的功能,通过io模块可以获取到用户输入的数据。
  • io模块中有一个名为Stdin的struct,该struct中的read_line方法可以从标准输入中读取用户的一行输入。

在io模块中有一个关联函数叫做stdin,该函数会创建一个Stdin的实例并返回,通过这个实例来调用read_line方法即可读取用户的一行输入。如下:

use std::io;fn main() {let mut guess = String::new();io::stdin().read_line(&mut guess).expect("无法读取行");println!("输入的内容: {}", guess);
}

说明一下:

  • Rust会将预导入(prelude)模块自动引入每一段程序的作用域中,该模块包含了一小部分常用的类型,如果你要使用的类型不在预导入模块中,那么就需要使用use语句显示进行导入声明。
  • 调用read_line方法读取用户数据时,为了将读取到的数据传递出来,需要以引用的方式传入一个String类型的变量,因此需要定义一个String类型的guess变量。(String是标准库中的一个字符串类型,该类型的new方法会创建一个新的空白字符串。)
  • 由于在read_line函数内部会将读取到的数据写入到guess变量中,因此在定义guess变量时使用mut关键字将其定义成一个可变的String变量,并且在将guess传递给read_line函数时也需要使用mut关键字来传递这个可变的引用变量。(Rust中的变量默认是不可变的)
  • Rust有一个静态强类型系统,同时拥有自动类型推导的能力,因此当代码中定义guess变量时虽然没有指明guess的类型,Rust也会自动将其推导为String类型。
小贴士:Rust中很多类型都有new方法,因为这是创建类型实例的惯用函数名称。

Result枚举

  • read_line函数会将读取到的内容存储到我们传入的字符串中,该函数的返回值类型是io::Result。
  • 在Rust标准库中有很多以Result命名的类型,它们通常是各个子模块中的Result泛型的特定版本。
  • Result是一个枚举类型,枚举类型由一系列固定的值组合而成,这些值被称作枚举的变体。

对于Result枚举来说,它有Ok和Err两个变体:

  • Ok变体表示当前操作执行成功,此时代码的结果值会存储在Ok变体中。
  • Err变体表示当前操作执行失败,此时引发失败的具体原因会存储在Err变体中。

Result枚举的定义如下:

pub enum Result<T, E> {/// Contains the success valueOk(T),/// Contains the error valueErr(E),
}

Result类型中定义了一系列方法,其中有一个方法叫做expect,该方法接收一个字符串:

  • 如果Result枚举的值为Ok,那么expect会提取出Ok中附带的值,并将其作为结果返回给用户。
  • 如果Result枚举的值为Err,那么expect会中断当前的程序,并将传入的字符串显示出来。

expect方法的定义如下:

impl<T, E> Result<T, E> {//...pub fn expect(self, msg: &str) -> TwhereE: fmt::Debug,{match self {Ok(t) => t,Err(e) => unwrap_failed(msg, &e),}}//...
}

说明一下:

  • 如果不对read_line函数的返回值做处理,那么会产生告警,因为read_line函数的返回值可能是一个Err变体,这就意味着我们没有对潜在的错误进行处理,通过这点就可以看到Rust的安全性。

解析用户输入

解析用户输入

现在我们已经能够获得用户的输入了,但此时guess变量是String类型的,而我们生成的神秘数字是整型的,为了能够将它们进行比较,需要将guess变量转化为整型。如下:

use std::io;fn main() {println!("请输入一个整数:>");let mut guess = String::new();io::stdin().read_line(&mut guess).expect("无法读取行");let guess: u32 = guess.trim().parse().expect("请您输入一个合法的整数!");println!("解析成功, 输入的数字是: {}", guess);
}

说明一下:

  • 字符串的trim方法的作用是去掉字符串两端的空白字符,包括空白字符、Tab、回车(\n)等,该方法会将处理后的字符串进行返回。
  • 字符串的parse方法的作用是将字符串解析成某种数值类型,比如i32、u32、i64、f64等,该方法的返回值是一个Result枚举,当字符串无法解析成数值类型时便会返回Err变体。
  • 由于parse方法的解析结果可能是多种数值类型,可以是浮点数也可以是整数,因此需要指定接收解析结果的变量的类型,定义变量时在变量名后面加一个冒号,再在冒号后面指定变量的类型即可。
  • 代码中接收解析结果的变量的变量名仍为guess,在Rust中不会报错,因为Rust允许使用同名的新变量来隐藏(shadow)原来同名的旧变量的值,这一特性通常被用在需要转换值类型的场景中。

多次处理用户输入

为了猜中神秘数字,用户可能需要进行多次猜测,因此我们也需要多次对用户的输入进行解析,在Rust中使用loop关键字即可创建一个无限循环。如下:

use std::io;fn main() {loop {println!("请输入一个整数:>");let mut guess = String::new();io::stdin().read_line(&mut guess).expect("无法读取行");let guess: u32 = match guess.trim().parse().expect("请您输入一个合法的整数!");println!("解析成功, 输入的数字是: {}", guess);}
}

match表达式

当用户输入非数字字符串时,parse方法就会解析失败,这时最好让用户重新猜测,而不是终止程序,鉴于parse方法的返回值类型是Result类型,因此可以用match表达式来代替expect函数,这也是在Rust中处理错误的一个惯用手段。

  • 如果parse返回的是Ok变体,那么表示解析成功,并且解析结果存储在Ok变体中,此时将Ok变体中的值返回即可。
  • 如果parse返回的是Err变体,那么表示解析失败,并且引发失败的具体原因会存储在Err变体中,此时打印一句提示语句并让循环continue即可。

代码如下:

use std::io;fn main() {loop {println!("请输入一个整数:>");let mut guess = String::new();io::stdin().read_line(&mut guess).expect("无法读取行");let guess: u32 = match guess.trim().parse() {Ok(num) => num,Err(_) => {println!("请您输入一个合法的整数!");continue;}};println!("解析成功, 输入的数字是: {}", guess);}
}

说明一下:

  • match表达式由多个分支(arm)组成,这些分支必须涵盖所有可能的情况,每个分支都包含一个用于匹配的模式(pattern),以及匹配成功后要执行的相应的代码。
  • Rust会尝试用我们传入match表达式的值去依次匹配每个分支的模式,如果匹配成功,它就会执行当前分支中的代码。
  • 对于match中的每个分支来说,模式中不需要的信息可以通过_忽略,而如果匹配成功后要执行的代码有多条,可以将这些代码放到一个代码块中。

进行猜测比较

进行猜测比较

现在要做的就是将用户猜测的数字和神秘数字进行比较。

  • 如果比较成功,则通过break跳出循环,游戏结束。
  • 如果比较失败,则提示用户猜大了还是猜小了,游戏继续。

这里也可以使用match表达式来处理。如下:

use rand::Rng;
use std::io;
use std::cmp::Ordering;fn main() {println!("欢迎来到猜数游戏!");//1、生成神秘数字let secret_number = rand::thread_rng().gen_range(1, 101);println!("神秘数字已经生成!");loop {//2、让用户进行猜测println!("请猜测:>");let mut guess = String::new();io::stdin().read_line(&mut guess).expect("无法读取行");//3、将用户输入的数字字符串转化为整型let guess: u32 = match guess.trim().parse() {Ok(num) => num,Err(_) => {println!("请您输入一个合法的整数!");continue;}};//4、将用户猜测的数与神秘数字进行比较match guess.cmp(&secret_number) {Ordering::Less => println!("您猜测的数字太小了"),Ordering::Greater => println!("您猜测的数字太大了"),Ordering::Equal => {println!("恭喜您猜对了, 神秘数字就是{}!", secret_number);break;}}}
}

说明一下:

  • 通过调用guess变量的cmp方法,可以将guess变量与同类型变量进行比较,在比较时传入另一个变量的引用即可。
  • cmp方法的返回值的类型是Ordering枚举,该枚举有Less、Greater和Equal三个变体,分别表示比较结果为小于、大于和等于。

此外,也可以使用if else语句对用户猜测的数字和神秘数字进行比较。如下:

use rand::Rng;
use std::io;fn main() {println!("欢迎来到猜数游戏!");//1、生成神秘数字let secret_number = rand::thread_rng().gen_range(1, 101);println!("神秘数字已经生成!");loop {//2、让用户进行猜测println!("请猜测:>");let mut guess = String::new();io::stdin().read_line(&mut guess).expect("无法读取行");//3、将用户输入的数字字符串转化为整型let guess: u32 = match guess.trim().parse() {Ok(num) => num,Err(_) => {println!("请您输入一个合法的整数!");continue;}};//4、将用户猜测的数与神秘数字进行比较if guess < secret_number {println!("您猜测的数字太小了");} else if guess > secret_number {println!("您猜测的数字太大了")} else {println!("恭喜您猜对了, 神秘数字就是{}!", secret_number);break;}}
}

说明一下:

  • 虽然之前secret_number变量被默认推导为i32类型,但由于这里将guess和secret_number进行了比较,而guess变量是u32类型的,因此Rust会将secret_number也推导为相同的u32类型。

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

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

相关文章

初始web项目tomcat部署报错404

问题 简单地创建了一个web项目&#xff0c;结果一运行就404咧&#xff0c;真滴烦。。。接下来的项目也没法继续了 问题原因&#xff1a;缺少文件 其实造成这样问题的原因有不少&#xff0c;但在这里我是踩了一个坑。在出问题之前&#xff0c;我运行的其他项目都是可以跑的&…

2. 验证1101序列(Mealy)

题目要求&#xff1a; 用 M e a l y \rm Mealy Mealy型状态机验证 1101 1101 1101序列 题目描述&#xff1a; 使用状态机验证 1101 1101 1101序列&#xff0c;注意&#xff1a;允许重复子序列。 方法一&#xff1a; 去掉 M o o r e \rm Moore Moore的 s 4 s_4 s4​&#xff…

掌握.NET基础知识(一)

前言 本文将讲解一些.NET基础。NET基础是指在计算机编程中使用.NET框架所需要的基础知识..NET的认识: .NET是由微软开发的一个跨平台的应用程序开发框架。它包括一个运行时环境和一个面向对象的程序库&#xff0c;可以用于开发各种类型的应用程序&#xff0c;包括桌面应用程序、…

数据驱动娱乐的未来:揭秘爱奇艺如何利用Apache Spark实现个性化推荐和内容分析

Apache Spark 在爱奇艺的现状与应用 Apache Spark 是爱奇艺大数据平台的核心组件&#xff0c;被广泛应用于数据处理、数据同步和数据查询分析等关键场景。在数据处理方面&#xff0c;爱奇艺通过数据开发平台支持开发者提交 Spark Jar 包任务或 Spark SQL 任务&#xff0c;进行…

【JVM系列】- 类加载子系统与加载过程

类加载子系统与加载过程 &#x1f604;生命不息&#xff0c;写作不止 &#x1f525; 继续踏上学习之路&#xff0c;学之分享笔记 &#x1f44a; 总有一天我也能像各位大佬一样 &#x1f3c6; 博客首页 怒放吧德德 To记录领地 &#x1f31d;分享学习心得&#xff0c;欢迎指正…

25.1 MySQL SELECT语句

1. SQL概述 1.1 SQL背景知识 1946年, 世界上诞生了第一台电脑, 而今借由这台电脑的发展, 互联网已经成为一个独立的世界. 在过去几十年里, 许多技术和产业在互联网的舞台上兴衰交替. 然而, 有一门技术却从未消失, 甚至日益强大, 那就是SQL.SQL(Structured Query Language&…

进化策略算法

前言 进化策略 (Evolution Strategy) 后面都简称 ES&#xff0c;其本质就是&#xff1a;种群通过交叉产生后代&#xff0c;我们只保留较好的父代和子代&#xff0c;一直这样迭代下去&#xff0c; 我们的保留方式是&#xff1a; 父代产生后代&#xff0c;然后将后代DNA和原来的…

02_单片机及开发板介绍

单片机简介 单片机&#xff0c;又称为微控制器&#xff08;Microcontroller&#xff09;&#xff0c;是一种集成了微处理器核心、存储器、输入/输出接口及各种功能模块的集成电路芯片。它通常由中央处理器&#xff08;CPU&#xff09;、存储器、输入/输出接口以及各种外设组成&…

【Leetcode】 707. 设计链表

你可以选择使用单链表或者双链表&#xff0c;设计并实现自己的链表。 单链表中的节点应该具备两个属性&#xff1a;val 和 next 。val 是当前节点的值&#xff0c;next 是指向下一个节点的指针/引用。 如果是双向链表&#xff0c;则还需要属性 prev 以指示链表中的上一个节点…

保序回归与金融时序数据

保序回归在回归问题中的作用是通过拟合一个单调递增或递减的函数&#xff0c;来保持数据点的相对顺序特性。 一、保序回归的作用 主要用于以下情况&#xff1a; 1. 有序数据&#xff1a;当输入数据具有特定的顺序关系时&#xff0c;保序回归可以帮助保持这种顺序关系。例如&…

rust学习—— 控制流if 表达式

控制流 根据条件是否为真来决定是否执行某些代码&#xff0c;或根据条件是否为真来重复运行一段代码&#xff0c;是大部分编程语言的基本组成部分。Rust 代码中最常见的用来控制执行流的结构是 if 表达式和循环。 if 表达式 if 表达式允许根据条件执行不同的代码分支。你提供…

vue-cli脚手架创建项目时报错Error: command failed: npm install --loglevel error

项目背景 环境&#xff1a;vue-cli 5.x 在工程文件中&#xff0c;后端模块wms已经创建完成&#xff0c;现在想新建一个名为vue-web的前端模块 执行命令vue create vue-web时&#xff0c; 报错Error: command failed: npm install --loglevel error 问题分析及解决 排查过程…

Linux性能优化--使用性能工具发现问题

9.0 概述 本章主要介绍综合运用之前提出的性能工具来缩小性能问题产生原因的范围。阅读本章后&#xff0c;你将能够&#xff1a; 启动行为异常的系统&#xff0c;使用Linux性能工具追踪行为异常的内核函数或应用程序。启动行为异常的应用程序&#xff0c;使用Linux性能工具追…

centos 7.9 源码安装htop

1.下载源码 wget http://sourceforge.net/projects/htop/files/latest/download 2.上传到tmp目录&#xff0c;并解压 tar xvzf htop-1.0.2.tar.gz mv htop-1.0.2 /opt/ 进入到 cd /opt/htop-1.0.2/ 3.编译并安装 ./configure && make && make install 4.…

基于混沌博弈优化的BP神经网络(分类应用) - 附代码

基于混沌博弈优化的BP神经网络&#xff08;分类应用&#xff09; - 附代码 文章目录 基于混沌博弈优化的BP神经网络&#xff08;分类应用&#xff09; - 附代码1.鸢尾花iris数据介绍2.数据集整理3.混沌博弈优化BP神经网络3.1 BP神经网络参数设置3.2 混沌博弈算法应用 4.测试结果…

什么是CSGO大行动,2023年CSGO大行动时间预测

什么是CSGO大行动&#xff0c;2023年CSGO大行动时间预测 什么是CSGO大行动&#xff0c;2023年CSGO大行动时间预测 那天群里在提大行动&#xff0c;不明所以的新同学在问&#xff0c;什么是大行动&#xff0c;是不是官方红锁大行动要来了&#xff1f;当然不是&#xff0c;别自己…

Linux内核的.config 配置文件和defconfig 配置文件

Linux 内核可以通过输入“make menuconfig”来打开图形化配置界面&#xff0c;menuconfig 是一套图形化的配置工具 一、图形化界面的操作 menuconfig 图形化的配置工具需要 ncurses 库支持。ncurses 库提供了一系列的 API 函数供调用者生成基于文本的图形界面&#xff0c;因此…

c++_learning-基础部分

文章目录 基础认识&#xff1a;语言特性&#xff08;面向对象编程&#xff09;&#xff1a;c的类&#xff08;相当于c中的结构体&#xff09;&#xff1a;三大特性&#xff1a;c包含四种编程范式&#xff1a;优缺点&#xff1a; c程序编译的过程&#xff1a;预处理->编译&am…

《软件方法》第1章2023版连载(07)UML的历史和现状

DDD领域驱动设计批评文集 做强化自测题获得“软件方法建模师”称号 《软件方法》各章合集 1.3 统一建模语言UML 1.3.1 UML的历史和现状 上一节阐述了A→B→C→D的推导是不可避免的&#xff0c;但具体如何推导&#xff0c;有各种不同的做法&#xff0c;这些做法可以称为“方…

WebDAV之π-Disk派盘 + 密码键盘

密码键盘是一款密码管理器,可以存储和管理需要受保护的数据。为方便日常使用,同时也是一款安全输入法,帮您安全便捷地填写账号密码、通用内容、卡包信息。 密码键盘使用军事级的 PBKDF2 有损加密算法保护您的根密码,使用军事级的 AES 加密算法保护您的存储数据。云端再额外…