Rust中saturating_sub的使用

alt

为了防止整数溢出,开发人员通常使用checked_add、checked_sub、saturating_add、saturating_sub等函数,而不是简单的加法和减法(+、-)


关于饱和减法


saturating使饱和之意

饱和减法(Saturating Subtraction)是一种数学运算,常用于计算机编程中,特别是在处理固定大小的数值类型(如整数)时。在饱和减法中,如果减法运算的结果超出了数值类型的表示范围,结果会被“饱和”到该类型的最小(或最大)可表示值,而不是产生溢出。

例如:

fn main() {
    let a: u8 = 50;
    let b: u8 = 100;
    let result = a.saturating_sub(b); // 结果为 0
    println!("结果为:{}", result);
}

u8即 无符号 8 位整数,可表示的范围为0 到 255.

对于a-b,如果不使用saturating_sub, 会因为a-b的结果超出u8的表示范围而溢出,发生报错.

使用saturating_sub, 则不会产生溢出, 会是u8类型的最小值,即0


使用饱和减法可以避免由于溢出导致的不期望的行为,确保结果始终在有效的数值范围内。在 Rust 语言中,标准库提供了 saturating_sub 方法来执行饱和减法运算。


关于saturating_sub


在 Rust 中,saturating_sub 方法是数值类型(如 u32, i32, u64, i64 等)的一个方法,用于执行饱和减法运算。

其作用可以:

  • 防止溢出:在减法运算中防止整数溢出,确保结果始终在有效范围内。
  • 提高安全性:避免因溢出导致的不可预测行为,增加代码的健壮性。

如何使用:

saturating_sub 方法可以直接调用在任何数值类型的实例上。它接收一个参数,即要减去的值,并返回计算结果。

示例:

fn main() {
    let a: u32 = 100;
    let b: u32 = 200;

    // 正常减法可能导致无符号整数下溢
    let normal_sub = a.wrapping_sub(b); // 结果可能是一个非常大的数.
    // 饱和减法避免了下溢
    let saturating_sub = a.saturating_sub(b); // 结果为 0

    println!("Normal Subtraction: {}", normal_sub); //  Normal Subtraction: 4294967196
    println!("Saturating Subtraction: {}", saturating_sub); // Saturating Subtraction: 0
}

在这个例子中,a.saturating_sub(b) 防止了因为 a - b100 - 200)而导致的下溢。对于无符号整数 u32 来说,这种下溢会导致结果变成一个非常大的数,因为 u32 不能表示负数。使用 saturating_sub 后,当减法结果为负时,它会返回 0(即 u32 能表示的最小值)。


关于 wrapping_sub

上例中用到了wrapping_sub方法, 即“包裹减法”(wrapping subtraction)。

和普通的减法操作相比,当发生溢出时,它会“包裹”到类型的最大或最小值,而不是引发溢出错误或产生未定义行为。 和saturating_sub正好相对.

wrapping_sub 的作用

  • 处理溢出:在减法运算中,如果结果超出了类型的表示范围,wrapping_sub 会使结果在类型的有效范围内循环(或“包裹”)。对于无符号类型,如果结果是负数,它会包裹到类型的最大值;对于有符号类型,它会在最大值和最小值之间循环。

  • 避免溢出错误:在“调试”模式下,Rust 默认会检查算术溢出。使用 wrapping_sub 可以避免因溢出而导致的运行时错误。


与直接减法相比:

  • 直接减法:在 Rust 中,直接使用 - 运算符进行减法时,如果在“调试”模式下发生溢出,程序会触发 panic(崩溃)。但在“发布”模式下,Rust 默认不检查溢出,此时溢出的行为类似于 wrapping_sub,但这并不是明确的行为保证。

  • wrapping_sub 方法:无论是在调试模式还是发布模式下,wrapping_sub 都会明确地进行包裹减法,确保结果是可预测的。


包裹减法的示例:

fn main() {
    let a: u8 = 0;
    let b: u8 = 1;

    // 直接减法,在调试模式下将会 panic
    // let result = a - b;

    // 包裹减法
    let result1 = a.wrapping_sub(b); // 结果为 255,因为 u8 类型下溢
    let result2 = a.saturating_sub(b);

    println!("结果1为:{}", result1); // 结果为:255
    println!("结果2为:{}", result2); // 结果为:0
}

在这个例子中,0 - 1 对于 u8 类型来说是下溢。使用 wrapping_sub 后,结果会从 0 下溢到 255,而不是引发 panic。



总结一下, saturating_加减乘除wrapping_加减乘除 都是提供了一种明确和安全的方式来处理可能的溢出情况,确保即使在溢出发生时,程序行为也是可预测和一致的。


本文由 mdnice 多平台发布

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

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

相关文章

Ai企业系统源码 Ai企联系统源码 商用去授权 支持文心 星火 GPT4等等20多种接口

智思Ai系统2.4.9版本去授权(可商用)支持市面上所有版本的接口例如:文心、星火、GPT4等等20多种接口!代过审AI小程序类目!!! 安装步骤: 1、在宝塔新建个站点,php版本使用…

Matlab编程技巧:导入MDF文件

文章目录 1 问题引入2 导入MDF文件示例3 将信号导入Simulink4 总结 1 问题引入 在汽车电控开发中,测试标定工程师可以通过CANape将控制器运行时的观测量实时地录制下来,保存成MDF文件。MDF文件中的数据有助于工程师排查控制器的问题,从而优化…

详细讲解Java8中的forEach()用法

目录 前言1. forEach()函数2. demo 前言 在实战中学习新用法,特此记录下每个笔记,感兴趣也可收藏也可补充细节 代码例子: import java.util.Arrays; import java.util.List;public class ForEachExample {public static void main(String[…

03-C++ 类和对象

类和对象 1. 概述 1.1 对象 真实存在的事物1.2 类 多个对象抽取其共同特点形成的概念静态特征提取出来的概念称为成员变量,又名属性 动态特征提取出来的概念称为成员函数,又名方法1.3 类与对象的关系 在代码中先有类后有对象 一个类可以有多个对象 …

Electron自定义通知Notification

Notification是什么? 对于渲染进程,Electron 允许开发者使用通知中API,来运行系统的原生通知进行显示。 如何实现系统Notification? const { Notification } require(electron);const isAllowed Notification.isSupported();…

AI 引擎系列 8 - 运行时比率参数简介

简介 在 Versal AI 引擎 2 一文 中,我们注意到计算图 (graph) 文件中有一行内容用于为每个内核实例定义运行时比率参数。 在本文中,我们将讲解该参数如何影响 AI 引擎应用的资源使用率和性能。 要求 下文要求您通读前几篇 AI 引擎系列博文。 AI 引擎系…

cesium实现区域贴图及加载多个gif动图

1、cesium加载多个gif动图 Cesium的Billboard支持单帧纹理贴图,如果能够将gif动图进行解析,获得时间序列对应的每帧图片,然后按照时间序列动态更新Billboard的纹理,即可实现动图纹理效果。为此也找到了相对于好一点的第三方库libg…

亚线性空间算法1

流模型 问题:如果采用bit存储的话 可以是n或者是Logn 但是对于特别大的数据量 这也是不行的,所以我们思考是否有Loglogn的算法 来存储统计的数据。 问题1:近似计数Morris算法 多次实验 使结果更准确。 问题2:不重复元素/FM算法…

3分钟了解安全数据交换系统有什么用!

企业为了保护核心数据安全,都会采取一些措施,比如做网络隔离划分,分成了不同的安全级别网络,或者安全域,接下来就是需要建设跨网络、跨安全域的安全数据交换系统,将安全保障与数据交换功能有机整合在一起&a…

acwing linux docker教程

终章 听着名字还挺伤感的哈哈哈其实是Linux的终章,感谢大家这段时间的阅读,这段使时间我为了给大家清楚的讲解一下Linux自己也是很认真的学习了一遍,自己提升了也不少。其实最近学校里面是讲了Linux和windows server 2019搭载DNS、web、ftp服…

HarmonyOS 组件隔代双向数据通信Provide与Consume

今天 我们说一个场景 我们可以编写代码如下 Entry Component struct Index {State name:string "小猫猫";build() {Row() {Column() {Text(this.name)Button("改个name").onClick(() > {this.name this.name "小猫猫"?"小狗狗&quo…

论文阅读——X-Decoder

Generalized Decoding for Pixel, Image, and Language Towards a Generalized Multi-Modal Foundation Model 1、概述 X-Decoder没有为视觉和VL任务开发统一的接口,而是建立了一个通用的解码范式,该范式可以通过采用共同的(例如语义&#…