【源码分析系列】number-precision和bignumber.js

news2025/1/4 19:25:06

01_JS精度

好久前在公司分享的文章,现在才发…本文阐述了为什么0.1 + 0.2 != 0.3,并分析了number-precision和bignumber.js的解决原理

被JS精度问题小坑了一把,所以系统来 复习 学习一波~

背景

在实际业务开发中,可能会遇到一下问题:

// 加法
0.1 + 0.2      // 0.30000000000000004

// 减法
1.5 - 1.2      // 0.30000000000000004

// 乘法
19.9 * 100     // 1989.9999999999998

// 除法
0.3 / 0.1      // 2.9999999999999996

toFixed()toPrecision() 在必要时进行四舍五入

有时候我们会用toFixed() 来解决这个问题,但是其实这个方法有时候会出现不希望的结果:

2.54.toFixed(1)         // 2.5
2.56.toFixed(1)         // 2.6

2.55.toFixed(1)             // error: 2.5
2.55.toPrecision(1)         // error: 2.5

业界内也诞生了一道经典的面试题:0.1 + 0.2 为什么不等于0.3?

因为 JS 采用IEEE 754 双精度版本(64位),并且只要采用 IEEE 754 的语言都有该问题。

IEEE754

前置知识

  • 计算机内部都是采用二进制进行表示,即0 1编码组成;

  • 十进制数转为二进制:

    • 正整数转二进制:将正整数除以2,得到的商继续再除以2,直到商为0或1为止,然后将余数倒着链接起来即可;
      在这里插入图片描述
      然后高位补0,如果是8位,那么在前面补2个0,所以最后结果是 :

    00100110 0010 0110 00100110

    • 负数转二进制:先将正整数转为二进制之后,对二进制取反,然后对结果再加1;

      -38 为例子,38的二进制是0010 0110 ,则取反后的结果是1101 1001 ,加1之后结果为:1101 1010

    • 小数转二进制:对小数点以后的数乘以2,取整数部分,再用小数部分乘以2,依次来推,直到小数部分为0或者位数已经OK了,再把整数部分依次排列就是小数的二进制结果了:

      以0.125为例子:

      0.125 * 2 = 0.25 --------------- 取整数 0,小数 0.25
      0.25 * 2 = 0.5 ----------------- 取整数 0,小数 0.5
      0.5 * 2 = 1 -------------------- 取整数 1
      
      

    所以结果是 0.001,可以按需低位补0。

    • 小数的整数部分大于0时,将整数、小数部分依次转为二进制,然后加在一起就OK。所以 38.125的二进制就是:0010 0110.001
  • 科学计数法,首先以10进制科学计数法为例子:

    • 23.32 => 0.2332 => 小数点向左移动了2位置,所以最终的结果是

      0.2332 ∗ 1 0 2 0.2332 * 10^2 0.2332102
      二进制在存储的时候是以二进制的科学计数法来存储的,如果是二进制科学计数法,则:

    • 10111=> 1.0111=> 小数点向左移动了4位,4转为2进制是100,所以最终的结果是
      1.0111 ∗ 2 ( 100 ) 1.0111 * 2^(100) 1.01112(100)

根据二进制科学技术法,小数点前必须有一个非0

什么是IEEE754

IEEE754标准中规定:

  • float单精度浮点数在机器中表示用 1 位表示数字的符号,用 8 位来表示指数,用23 位来表示尾数,即小数部分。
  • 对于double双精度浮点数,用 1 位表示符号,用 11 位表示指数,52 位表示尾数,其中指数域称为阶码。所有数值的计算和比较,都是这样以64个bit的形式来进行的

在这里插入图片描述

JS中,所有Number都是以 64bit的双精度浮点数存储的。

符号S

由于计算机万物都是以二进制表示,为了理解符号,一般将最高位当作符号位来理解,0代表+,1代表-。

指数E

它占了11位,所以取值范围是0~2的11次方,即0~1024位,即可以代表1024个数字。但是IEEE 754 标准规定指数偏移值的固定值 2 e − 1 − 1 2^{e-1}-1 2e11,以双精度浮点数为例: 2 11 − 1 − 1 = 1023 2^{11-1}-1=1023 21111=1023

为什么 IEE754浮点数标准中64位浮点数的指数偏移量是1023?

在这里插入图片描述

以32位浮点数为例子,指数占8位,即0-2的8次方,也就是256。由于指数也有正负,所以从中间劈开,-128 ~ +128,但是中间有个0,所以表示-128到127这256个数字。

怎么记录正负?一种作法是把高位置1,这样我们只要看到高位是1的就知道是负数了,所谓高位置1就是说把0到255这么多个数字劈成两半,从0到127表示正数,从128到255表示负数。但是这种作法会带来一个问题:当你比较两个数的时候,比如130和30,谁更大呢?机器会觉得130更大,但实际上130是个负数,它应该比30小才对啊。

所以后来有人提出了,将所有数字加上128,这样-128 + 128 = 0, 127 + 128 = 255这样比较,就不存在负数比正数大的情况。

所以如果你读到0,就减去128,则得到负指数-128,读到255,减去128,就得到127。

那为什么最终指数偏移量是127,不是128,因为不允许使用0和255两个数字代表指数。少了2个数字,所以只能采用127。

同理,64位,指数11位,即2^11 = 2048,对半1024,去掉0和2048,所以偏移量用1023

尾数M

对于尾数M,只保存后面的小数部分。这是由于1≤M<2,在计算机内部保存M时,默认这个数的第一位总是1,因此可以被舍去,这样做的好处是可以节省一位有效数字。对于双精度 64 位浮点数,M 为 52 位,将第一位的 1 舍去,可以保存的有效数字为 52 + 1 = 53 位。

根据二进制科学技术法,小数点前必须有一个非0,那么有效域即1.xxxx,小数点前的1默认存在,但是默认不占坑,尾数部分就存储小数点后的部分

十进制转为IEEE754

当我们使用 Number的时候,计算机底层会自动将我们输入的十进制数字自动转为 IEEE754标准的浮点数。

以0.1为例子,它转为二进制科学计数法是这个:

0.1001100110011001100110011001100110011001100110011001 ∗ 2 − 4 0.1001100110011001100110011001100110011001100110011001*2^{-4} 0.100110011001100110011001100110011001100110011001100124

  • 由于0.1是正数,所以符号位是0;
  • 指数是-4,则-4 +1023 = 1019,转为二进制为1111111011,共10位。由于指数E为11位,所以高位补0.最终得出 01111111011;
  • 尾数最多存储52位,所以会采取进1舍0的情况:
    11001100110011001100110011001100110011001100110011001 // M 舍去首位的 1,得到如下
    1001100110011001100110011001100110011001100110011001  // 01 入,得到如下:
    1001100110011001100110011001100110011001100110011010  // 最终存储
    

0舍1入法:尾数右移时,被移去的最高位数值为0,则舍去;被移去的最高位数值为1,则在末位加1

所以0.1的最终转换结果为:

S  E            M
0  01111111011  1001100110011001100110011001100110011001100110011010 

那么同理,0.2的最终转换结果为:

S  E            M
0  01111111100  1001100110011001100110011001100110011001100110011010 // 0.2

浮点数的运算

对阶

在进行结算前,需要判断两个数的指数位是否相同,即小数点位置是否一致。0.1的阶码是-4,而0.2的阶码是-3,根据 小阶向大阶看齐原则,需要将0.1进行移码操作:尾数向右移动1位,指数位+1:

// 0.1 移动之前
0  01111111011  1001100110011001100110011001100110011001100110011010 

// 0.1 右移 1 位之后尾数最高位空出一位,(0 舍 1 入,此处舍去末尾 00  01111111100   100110011001100110011001100110011001100110011001101(0) 

// 0.1 右移 1 位完成
0  01111111100  1100110011001100110011001100110011001100110011001101

p.s. 不改变最高位值,是 1 补 1,是 0 补 0。尾数部分我们是有隐藏掉最高位是 1 的

尾数求和

  0  01111111100   1100110011001100110011001100110011001100110011001101 // 0.1 
+ 0  01111111100   1001100110011001100110011001100110011001100110011010 // 0.2
= 0  01111111100 100110011001100110011001100110011001100110011001100111 // 产生进位,待处理

规格化和舍入

由于产生进位,阶码需要 + 1, 所以为 01111111101,对应的十进制为 1021,1021 - 1023 = -2,所以:

  S  E
= 0  01111111101

尾部进位 2 位,去除最高位默认的 1,因最低位为 1 需进行舍入操作(在二进制中是以 0 结尾的),舍入的方法就是在最低有效位上加 1,若为 0 则直接舍去,若为 1 继续加 1:

  100110011001100110011001100110011001100110011001100111 // + 1
=  00110011001100110011001100110011001100110011001101000 // 去除最高位默认的 1
=  00110011001100110011001100110011001100110011001101000 // 最后一位 0 舍去
=  0011001100110011001100110011001100110011001100110100  // 尾数最后结果

IEEE 754 中最终存储如下:

S  E           M
0  01111111101 0011001100110011001100110011001100110011001100110100

IEEE754转为十进制

根据公式:

n = ( − 1 ) s ∗ 2 ( e − 1023 ) ∗ ( 1 + f ) n = (-1)^s * 2^(e-1023)*(1+f) n=(1)s2(e1023)(1+f)

( − 1 ) 0 ∗ 2 ( − 2 ) ∗ ( 1 + 0011001100110011001100110011001100110011001100110100 ) (-1)^0 * 2(-2) * (1 + 0011001100110011001100110011001100110011001100110100) (1)02(2)(1+0011001100110011001100110011001100110011001100110100)

最终答案为:

0.30000000000000004

当你打印的时候,其实发生了二进制转为十进制,十进制转为字符串,最后输出的。而十进制转为二进制会发生近似,那么二进制转为十进制也会发生近似,打印出来的值其实是近似过的值,并不是对浮点数存储内容的精确反映。

How does javascript print 0.1 with such accuracy?

精度丢失点

  • 十进制转二进制,如果遇到小数是无限循环,超过52位,那么就会被舍入;
  • 浮点数参与计算的时候需要对阶,以加法为例,要把小的指数域转化为大的指数域,也就是左移小指数浮点数的小数点,一旦小数点左移,必然会把52位有效域的最右边的位给挤出去,这个时候挤出去的部分也会发生“舍入”。这就又会发生一次精度丢失。

解决思路

  • 借助 parseFloat 对结果进行指定精度的四舍五入,但是并不保守
    210000 * 10000  * 1000 * 8.2                   // 17219999999999.998
    parseFloat(17219999999999.998.toFixed(12));    // 17219999999999.998
    parseFloat(17219999999999.998.toFixed(2));     // 而正确结果为 17220000000000
    
    
  • 将浮点数转为整数运算,再对结果做除法,目前足够应付大多数场景的思路就是,将小数转化为整数,在整数范围内计算结果,再把结果转化为小数,因为存在一个范围,这个范围内的整数是可以被IEEE754浮点形式精确表示的
    0.1 + 0.2                        // 0.30000000000000004
    (0.1 * 100 + 0.2 * 100) / 100    // 0.3
    
  • 将浮点数转为字符串,模拟实际运算的过程。

常见的轮子

number-precision

https://github.com/nefe/number-precision

用法

import NP from 'number-precision'
NP.strip(0.09999999999999998); // = 0.1
NP.plus(0.1, 0.2);             // = 0.3, not 0.30000000000000004
NP.plus(2.3, 2.4);             // = 4.7, not 4.699999999999999
NP.minus(1.0, 0.9);            // = 0.1, not 0.09999999999999998
NP.times(3, 0.3);              // = 0.9, not 0.8999999999999999
NP.times(0.362, 100);          // = 36.2, not 36.199999999999996
NP.divide(1.21, 1.1);          // = 1.1, not 1.0999999999999999
NP.round(0.105, 2);            // = 0.11, not 0.1

原理

主要就是结合了 parseFloat() 将小数转为了整数。以加法为例子:

function plus(...nums: numType[]): number {
  // 如果是多个参数,则递归相加
  if (nums.length > 2) {
    return iteratorOperation(nums, plus);
  }

  const [num1, num2] = nums;
  // 取两个数当中,小数位长度最大的值的长度
  const baseNum = Math.pow(10, Math.max(digitLength(num1), digitLength(num2)));
  // 把小数都转为整数然后再计算
  return (times(num1, baseNum) + times(num2, baseNum)) / baseNum;
}
  • 取两数之中小数最大的小数长度作为基数;
  • 将两个数转为整数相加之后然后除以基数。

其中:

function times(...nums: numType[]): number {
  // 如果是多个参数,则递归相乘
  if (nums.length > 2) {
    return iteratorOperation(nums, times);
  }
  
  // 将每个变量转为整数并相乘
  const [num1, num2] = nums;
  const num1Changed = float2Fixed(num1);
  const num2Changed = float2Fixed(num2);
  const leftValue = num1Changed * num2Changed;
  
  // 检查是否越界,如果越界就报错
  checkBoundary(leftValue);
  
  // 获得分母,即Math.pow(10,小数长度的数量)
  const baseNum = digitLength(num1) + digitLength(num2);
  return leftValue / Math.pow(10, baseNum);
}

float2Fixed 将小数转为整数:

function float2Fixed(num: numType): number {
  // 如果不是科学计数法,直接去掉小数点
  if (num.toString().indexOf('e') === -1) {
    return Number(num.toString().replace('.', ''));
  }
  
  // 如果是科学计数法,获得小数的长度
  const dLen = digitLength(num);
  return dLen > 0 ? strip(Number(num) * Math.pow(10, dLen)) : Number(num);
}

digitLength 计算小数的长度:

// 常见的数字:1、0.1、2.2e-7
// 其中 2.2e-7 实际上就是指 0.00000022
            
function digitLength(num: numType): number {
  // 获取指数前后的数字
  const eSplit = num.toString().split(/[eE]/);
  // 如果 e 之前是小数,获取小数的数量 + e之后的数量
  const len = (eSplit[0].split('.')[1] || '').length - +(eSplit[1] || 0);
  // 返回小数的长度
  return len > 0 ? len : 0;
}

借助 parseFloat

function strip(num: numType, precision = 15): number {
  return +parseFloat(Number(num).toPrecision(precision));
} 

console.log(strip(0.1 + 0.2));    // 0.3

bignumber.js

https://github.com/MikeMcl/bignumber.js

这位大佬同时还写了 big.jsdecimal.js等跟计算有关的库。

在这里插入图片描述

第一感觉:为啥这么多???

在这里插入图片描述

用法

0.3 - 0.1                           // 0.19999999999999998
x = new BigNumber(0.3)
x.minus(0.1)                        // "0.2"
x                                   // "0.3"

原理

先看一下构造函数,emmm…看源码的时候,我其实是这样的:

在这里插入图片描述

在这里插入图片描述

let x = new BigNumber(123.4567);
console.log(x);
// { c: (2) [123, 45670000000000], e: 2, s: 1 }

let y = BigNumber('123456.7e-3');
console.log(y);
// { c: (2) [123, 45670000000000], e: 2, s: 1 }

加法的实现:

  • 先将两个数都转为 BigNumber 类型;以 0.1 和 1.1为例子:
    { c: [10000000000000], e: -1, 1 }     // 0.1
    { c: [1, 25000000000000], e: 0, 1 }   // 1.1
    
  • 判断两个数之中是否是 NaN的,有的话直接返回new BigNumber(NaN);
  • 如果有一方是负数,则调用减法的计算结果;
  • 记录 x.e、y.e、x.c、y.c
    var xe = x.e / LOG_BASE,
            ye = y.e / LOG_BASE,
            xc = x.c,
            yc = y.c;
    
    console.log(xe, ye, xc, yc);      
    // -0.07142857142857142 0 [10000000000000] (2) [1, 25000000000000]
    // 其中LOG_BASE = 14;
    
  • 判断 xe、ye中是否有一个为0的时候,根据条件返回不同的值:
    if (!xe || !ye) {
    
      // ±Infinity
      if (!xc || !yc) return new BigNumber(a / 0);
    
      // Either zero?
      // Return y if y is non-zero, x if x is non-zero, or zero if both are zero.
      if (!xc[0] || !yc[0]) return yc[0] ? y : new BigNumber(xc[0] ? x : a * 0);
    }
    
  • 处理一下 xeye, 浅拷贝xc :
    xe = bitFloor(xe);    // -0.07142857142857142 => -1
    ye = bitFloor(ye);    // 0 => 0
    xc = xc.slice();
    
    // n | 0 有省略小数的作用
    function bitFloor(n) {
        var i = n | 0;
        return n > 0 || n === i ? i : i - 1;
     }
     
    console.log( 104.249834 | 0 ); //104
    console.log( 9.999999 | 0 );   // 9
    
  • 根据yexe,对xcyc中比较短的一方进行补0操作,所以此时变为:
    // [10000000000000]
    xc: [0, 10000000000000] 
    // [1, 25000000000000]
    yc: [1, 25000000000000]
    
  • 比较 xc yc 的长度,确保长度较长的数值放在 xc中;
  • 遍历相加:
    // Only start adding at yc.length - 1 as the further digits of xc can be ignored.
    for (a = 0; b;) {
       a = (xc[--b] = xc[b] + yc[b] + a) / BASE | 0;
       xc[b] = BASE === xc[b] ? 0 : xc[b] % BASE;
    }
    
  • 最后通过 normalise 整合一下最后的结果,返回一个新的 BigNumber 对象。

学习到的几行感觉很有用的代码:

// 将v转为整数比较的最快方法(当v < 2**31 时,比较是否是整数)
v === ~~v

// |0 直接取整数部分
function bitFloor(n) {
  var i = n | 0;
  return n > 0 || n === i ? i : i - 1;
}
console.log(0.6 | 0);     // 0
console.log(1.1 | 0);     // 1
console.log(3.6555 | 0);   // 3
console.log(-3.6555 | 0);   // -3

这个库跟 big.js的差别在于,后者的API没有前者多,而且不支持十进制以外的计算。

总结

  • 当我们在使用 Number 类型时,计算机底层会自动将我们输入的十进制数字自动转为 IEEE754标准的浮点数;
  • 转浮点数的时候会出现精度丢失的情况,一般是发生在十进制转二进制的时候,或者是浮点数参与计算的时候需要对阶;
  • 解决这个问题可以考虑使用parseFloat、浮点转整数、浮点转字符串;
  • 业界出名的轮子有 number-precision、bignumber.js等,前者主要借助了parseFloat和浮点转整数的思想,后者是先将数值转为特定的对象,然后进行整数计算。

参考

  • 该死的IEEE-754浮点数,说「约」就「约」,你的底线呢?以JS的名义来好好查查你
  • IEEE 754浮点数标准中64位浮点数为什么指数偏移量是1023?
  • 在线转IEEE-754工具
  • How does javascript print 0.1 with such accuracy?
  • MDN-toFixed

如果错误,欢迎指出,感谢阅读~

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

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

相关文章

Redis_简介(1)

目录 Redis简介 Redis特性 Redis 优势 Redis应用场景 源码等资料获取方法 Redis简介 Redis是一个开源的使用ANSI C语言编写、支持网络、可基于内存亦可持久化的日志型、Key-Value数据库&#xff0c;并提供多种语言的API。从2010年3月15日起&#xff0c;Redis的开发工作由…

Git命令-状态与版本查看

Git是一个强大的分布式管理系统,版本控制主要在本地端完成,在团队合作和协助开发中,都在广泛使用Git工具,因为他免费哈哈哈 目录 git status git reflog git reset --hard 版本号 git status 可以查看当前git状态 On branch master 是你当前的分支状态,当前我是在master主…

[分布式] zookeeper集群与kafka集群

目录 一、Zookeeper 概述1.1 Zookeeper定义1.2 Zookeeper 工作机制1.3 Zookeeper 特点1.4 Zookeeper 数据结构1.5 Zookeeper 应用场景1.6 Zookeeper 选举机制 二、 部署 Zookeeper 集群2.1 环境部署2.2 关闭防火墙2.3 安装 JDK 下载安装包 安装 Zookeeper2.4 修改配置文件2.5 将…

[NGINX] NGINX下载、安装编译、启动检查停止命令

一、NGINX 下载 mkdir -p /soft/nginx cd /soft/nginx wget https://nginx.org/download/nginx-1.21.6.tar.gz二、下载相关依赖 ①在线安装依赖&#xff1a; yum -y install gcc zlib zlib-devel pcre-devel openssl openssl-devel ②下载依赖到本地安装依赖&#xff1a; y…

2023年7月23日 星期三 Linux驱动作业

1.使用驱动代码实现如下要求 a.应用程序通过阻塞的io模型来读取number变量的值 b.number是内核驱动中的一个变量 c.number的值随着按键按下而改变(按键中断) 例如number0 按下按键number1再次按下按键number0 d.在按下按键的时候需要同时将1ed1的状态取反 e.驱动中需要编写字符…

FlinkCDC第四部分-同步mysql到mysql,ctrl就完事~(flink版本1.17.1)

本文介绍了不同源单表-单表同步&#xff0c;不同源多表-单表同步。 注&#xff1a;此版本支持火焰图 Flink版本&#xff1a;1.17.1 环境&#xff1a;Linux CentOS 7.0、jdk1.8 基础文件&#xff1a; flink-1.17.1-bin-scala_2.12.tgz、 flink-connector-jdbc-3.0.0-1.16.…

LeetCode·每日一题·2544. 交替数字和·模拟

作者&#xff1a;小迅 链接&#xff1a;https://leetcode.cn/problems/alternating-digit-sum/solutions/2341276/mo-ni-zhu-shi-chao-ji-xiang-xi-by-xun-ge-7fjq/ 来源&#xff1a;力扣&#xff08;LeetCode&#xff09; 著作权归作者所有。商业转载请联系作者获得授权&#…

产品经理学习画原型(三)

1.布尔运算的使用 合并必须选择两个及以上元件 合并后&#xff1a; 组合&#xff1a; 去除顶层&#xff1a;

【基于FPGA的芯片设计】32位RISC-V存储器

实验板卡&#xff1a;xc7a100tlc sg324-2L&#xff0c;共20个开关 实验要求

述往思来,砥砺前行—易我文化系列课《企业发展历程》讲座圆满举行

2023年6月9日&#xff0c;易我文化系列课《企业发展历程》讲座如期举行并取得圆满成功。 此次讲座邀请了易我客服总监——喻涛为大家授课&#xff0c;旨在让员工对易我的历史、发展以及未来有更全面的认识&#xff0c;引导员工从易我发展史中汲取奋进力量&#xff0c;增强集体…

【kubernetes系列】Kubernetes之ServiceAccount

概述 Service Account是什么呢&#xff0c;顾名思义&#xff0c;服务账号&#xff0c;一种给服务使用的账号&#xff0c;它不是给Kubernetes的集群的用户&#xff08;系统管理员、运维人员、租户用户等&#xff09;使用&#xff0c;而是给运行在Pod里的进程用的&#xff0c;它…

【雕爷学编程】Arduino动手做(113)---5110液晶屏模块2

37款传感器与执行器的提法&#xff0c;在网络上广泛流传&#xff0c;其实Arduino能够兼容的传感器模块肯定是不止这37种的。鉴于本人手头积累了一些传感器和执行器模块&#xff0c;依照实践出真知&#xff08;一定要动手做&#xff09;的理念&#xff0c;以学习和交流为目的&am…

基于springboot+Redis的前后端分离项目(八)-【黑马点评】

&#x1f381;&#x1f381;资源文件分享 链接&#xff1a;https://pan.baidu.com/s/1189u6u4icQYHg_9_7ovWmA?pwdeh11 提取码&#xff1a;eh11 好友关注&Feed流 &#xff08;一&#xff09;好友关注-关注和取消关注(二)好友关注-共同关注&#xff08;三&#xff09; 好友…

飞行动力学 - 第7节-part1-起飞性能 之 基础点摘要

飞行动力学 - 第7节-part1-起飞性能 之 基础点摘要 1. 气动特性2. 起飞性能3. 性能指标3.1 地面滑跑阶段3.2 起飞滑跑距离估算 4. 跑道4.1 编号4.2 等级 5. 参考资料 1. 气动特性 起飞不仅需要考虑升力&#xff0c;还需要在有限跑道长度上加速&#xff0c;因此襟翼放出的角度不…

3Ds max入门教程:创建尼亚加拉大瀑布模型

推荐&#xff1a; NSDT场景编辑器助你快速搭建可二次开发的3D应用场景 初学者在3ds Max中为尼亚加拉大瀑布建模 这次您将学习通过几个简单的步骤在3ds max中对尼亚加拉大瀑布&#xff08;从远处看起来很逼真&#xff09;进行建模。所以&#xff0c;让我们开始吧&#xff01; …

JavaScript定时器(setTimeout和setInterval)的用法技巧

1.满足特定条件时才执行后续步骤。 核心代码&#xff1a; //初始化let isPass false;//只有满足特定条件&#xff08;在这里为&#xff1a;isPass为true时&#xff09;时&#xff0c;才执行下一步操作let timeout setInterval(() > {if (isPass) {//这里写要执行的后续方…

Kubernetes 入门篇上

❖ Kubernetes核心概念 ❖ 快速部署一个Kubernetes集群 ❖ Kubernetes将弃用Docker&#xff01; ❖ kubectl管理工具 ❖ Kubernetes基本资源概念 ❖ 资源编排&#xff08;YAML&#xff09; ❖ 深入理解Pod对象&#xff1a;基本管理 ❖ 深入理解Pod对象&#xff1a;调度 …

Chapter 6: Loops and Iterations | Python for Everybody 讲义笔记_En

文章目录 Python for Everybody课程简介Loops and IterationsUpdating variablesThe while statementInfinite loopsFinishing iterations with continueDefinite loops using forLoop patternsCounting and summing loopsMaximum and minimum loopsDebuggingGlossary Python f…

Chapter 7: Strings | Python for Everybody 讲义笔记_En

文章目录 Python for Everybody课程简介StringsA string is a sequenceGetting the length of a string using lenTraversal through a string with a loopString slicesStrings are immutableLooping and countingThe in operatorString comparisonString methodsParsing stri…

x86平台实时Windows机器视觉EtherCAT运动控制器VPLC711

一、市场背景 简单易用&#xff0c;功能强大&#xff0c;正运动技术持续专注智能制造核心控制器的产品与平台的突破&#xff01; 随着智能制造的兴起&#xff0c;制造型企业正面临着日益激烈的市场竞争和对生产效率与产品品质提升的迫切需求&#xff0c;以满足市场的要求。同…