当前位置: 首页 > news >正文

Rust 学习笔记:比较数值

Rust 学习笔记:比较数值

  • Rust 学习笔记:比较数值
    • 整数类型
    • 浮点类型
    • NAN

Rust 学习笔记:比较数值

整数类型

在 Rust 中,可以用以下运算符比较数值:

>、<、==、!=、>=、<=

但是, Rust 不允许比较不同类型的值。

解决方法 1:使用 as 进行类型转换

fn main() {let a: i32 = 10;let b: u16 = 100;if a < (b as i32) {println!("{a} is less than {}", b);}
}

从范围小的类型转换成范围大的类型是安全的,比如 u16 -> i32。

但要注意从范围大的类型转换成范围小的类型,编译不会报错,但结果可能不对。

解决方法 2:使用 try_into() 进行类型转换

try_into() 方法:

  • 导入 std::convert::TryInto trait。
  • 该方法返回 Result 类型。
use std::convert::TryInto;fn main() {let a: i32 = 10;let b: u16 = 100;let b = b.try_into().unwrap();if a < b {println!("{a} is less than {}", b);}
}

浮点类型

Rust 中的浮点类型(如 f32、f64)是基于二进制实现的,但我们通常用十进制来计算数值。

浮点类型的某些值不能很好地结合在一起。例如 f32、f64 只实现了 std::cmp::PartialEq,而其他数值类型还实现了 std::cmp::Eq。

针对浮点类型的指导方针:

  • 避免测试浮点类型的相等性
  • 如果结果在数学上是未定义的,这时候要小心

示例 1:

fn main() {assert!(0.1 + 0.2 == 0.3);
}

该程序执行会出错:

在这里插入图片描述

浮点数的底层表示使得 0.1 + 0.2 和 0.3 存在极其细微的差别。

示例 2:

fn main() {let abc: (f32, f32, f32) = (0.1, 0.2, 0.3);let xyz: (f64, f64, f64) = (0.1, 0.2, 0.3);println!("abc (f32)");println!("   0.1 + 0.2: {:x}", (abc.0 + abc.1).to_bits());println!("         0.3: {:x}", (abc.2).to_bits());println!();println!("xyz (f64)");println!("   0.1 + 0.2: {:x}", (xyz.0 + xyz.1).to_bits());println!("         0.3: {:x}", (xyz.2).to_bits());println!();assert!(abc.0 + abc.1 == abc.2);assert!(xyz.0 + xyz.1 == xyz.2);
}

程序在最后的 assert! 还是出错了:

在这里插入图片描述

在 f64 类型中,我们看到 0.1 + 0.2 为 3fd3333333333334,而 0.3 为 3fd3333333333333,它们并不是完全相等。

那么,我们怎么比较浮点类型呢?

一般来说,测试数学运算是否在其真实数学结果的可接受范围内更安全。这个边界通常被称为 ε。

Rust 提供了一些可容忍的误差值:f32::EPSILON 和 f64::EPSILON。

示例 3:

fn main() {let result: f32 = 0.1 + 0.2;let desired: f32 = 0.3;let abs_diff = (desired - result).abs();assert!(abs_diff <= f32::EPSILON);
}

程序正常执行。

示例 4:

fn main() {let result: f64 = 0.1 + 0.2;let desired: f64 = 0.3;println!("{}", desired - result);let abs_diff = (desired - result).abs();assert!(abs_diff <= f64::EPSILON);
}

程序正常执行。输出:-0.00000000000000005551115123125783。可以看出误差很小,且小于 f64::EPSILON。

Rust 编译器实际上将比较的工作交给了 CPU,浮点运算是使用芯片内的定制硬件实现的。

NAN

NAN 表示“不是一个数”,例如负数的平方根就是 NAN。

NAN 会影响其它数值:

  • 几乎所有与 NAN 交互的操作都返回 NAN
  • NAN 值永远不相等

相关方法:

  • is_nan():判断一个数是不是 NAN
  • is_finite():判断一个数是不是有限的

示例:

fn main() {let x: f32 = 1.0 / 0.0;assert!(x.is_finite());
}

程序出错:

在这里插入图片描述

http://www.dtcms.com/a/263970.html

相关文章:

  • Prompt生成指南
  • 数据结构与算法--蛇行矩阵问题
  • WPF学习笔记(17)样式Style
  • 【机器学习2】正则化regularizaiton(降低模型过拟合)
  • Http、Ftp、Dns和Dhcp服务器搭建
  • Go 服务如何“主动”通知用户?SSE广播与断线重连实战
  • 从docker-compose快速入门Docker
  • VCenter SSL过期,登录提示HTTP 500错误解决办法
  • Linux驱动学习day13(同步与互斥)
  • 记录一次生产环境ActiveMQ无法启动的问题
  • 变幻莫测:CoreData 中 Transformable 类型面面俱到(八)
  • Raspberry Pi 4边缘智能PLC:OpenPLC赋能物联网
  • 25-7-1 论文学习(1)- Fractal Generative Models 何恺明大佬的论文
  • 半导体和PN结
  • 遥感影像岩性分类:基于CNN与CNN-EL集成学习的深度学习方法
  • 胖喵安初 (azi) Android 应用初始化库 (类似 Termux)
  • Adobe AI高效设计技巧与创新思维指南
  • day41简单CNN
  • 注意力得分矩阵求解例子
  • 网站崩溃的幕后黑手:GPTBot爬虫的流量冲击
  • 第七讲~~测试工具(禅道项目管理系统)
  • 【记录】Word|Word创建自动编号的多级列表标题样式
  • poi java 删除word的空白页
  • 【docker】docker save和docker load
  • 通达信【极弱强势指标与股道波段交易系统】幅图
  • Gin 中间件详解与实践
  • 发布/订阅模式:解耦系统的强大设计模式
  • Python Flask 容器化应用链路可观测
  • 基于SSM万华城市货运服务系统的设计与实现
  • 开源模型与商用模型协同开发机制设计