• php浮点数比较


    本文实例讲述了PHP中两个float(浮点数)比较方法。分享给大家供大家参考。具体如下:

    最近在开发一个合同管理系统的时候,涉及到两个浮点数比较,算是把我郁闷惨了。
    在N久以前,就不晓得从哪里听来的一个“不要用等号去比较浮点数”的“真理”,自己平时也在用,好像没有出现啥问题,可这次问题总算是来了。

    1
    2
    3
    4
    5
    6
    7
    8
    9
    <?php
    $sum = "12300.00";
    $a  = "10000.30";
    $b  = "2000.30";
    $c  "299.40";
    $sum = (float) $sum;
    $s = (float) ($a+$b+$c);
    var_dump($sum, $s);
    var_dump($sum==$s);

    结果是:

    float(12300)
    float(12300)
    bool(false)

    后来才知道在PHP中,要比较两个浮点数的大小,可以用bccomp(参数1,参数2,小数位)来比较。

    1
    2
    3
    4
    5
    6
    7
    8
    9
    <?php
    $sum = "12300.00";
    $a  = "10000.30";
    $b  = "2000.30";
    $c  "299.40";
    $sum = (float) $sum;
    $s = (float) ($a+$b+$c);
    var_dump($sum, $s);
    var_dump(bccomp($sum,$s,2));

    结果:

    float(12300)
    float(12300)
    int(0) // 0表示两个浮点数值相等

    <?php

    echo bccomp('1', '2') . " ";   // -1 小于
    echo bccomp('1.00001', '1', 3); // 0 等于
    echo bccomp('1.00001', '1', 5); // 1 大于

    ?>

    如果用php的+-*/计算浮点数的时候,可能会遇到一些计算结果错误的问题,比如echo intval( 0.58*100 );会打印57,而不是58

    这个其实是计算机底层二进制无法精确表示浮点数的一个bug,是跨语言的

    可以用精度函数库解决问题

      

      bcadd — 将两个高精度数字相加

      bccomp — 比较两个高精度数字,返回-1, 0, 1

      bcdiv — 将两个高精度数字相除

      bcmod — 求高精度数字余数

      bcmul — 将两个高精度数字相乘

      bcpow — 求高精度数字乘方

      bcpowmod — 求高精度数字乘方求模,数论里非常常用

      bcscale — 配置默认小数点位数,相当于就是Linux bc中的”scale=”

      bcsqrt — 求高精度数字平方根

      bcsub — 将两个高精度数字相减

    复制代码
     1 /**
     2   * 两个高精度数比较
     3   * 
     4   * @access global
     5   * @param float $left
     6   * @param float $right
     7   * @param int $scale 精确到的小数点位数
     8   * 
     9   * @return int $left==$right 返回 0 | $left<$right 返回 -1 | $left>$right 返回 1
    10   */
    11 var_dump(bccomp($left=4.45, $right=5.54, 2));
    12 // -1
    13   
    14  /**
    15   * 两个高精度数相加
    16   * 
    17   * @access global
    18   * @param float $left
    19   * @param float $right
    20   * @param int $scale 精确到的小数点位数
    21   * 
    22   * @return string 
    23   */
    24 var_dump(bcadd($left=1.0321456, $right=0.0243456, 2));
    25 //1.04
    26  
    27   /**
    28   * 两个高精度数相减
    29   * 
    30   * @access global
    31   * @param float $left
    32   * @param float $right
    33   * @param int $scale 精确到的小数点位数
    34   * 
    35   * @return string 
    36   */
    37 var_dump(bcsub($left=1.0321456, $right=3.0123456, 2));
    38 //-1.98
    39   
    40  /**
    41   * 两个高精度数相除
    42   * 
    43   * @access global
    44   * @param float $left
    45   * @param float $right
    46   * @param int $scale 精确到的小数点位数
    47   * 
    48   * @return string 
    49   */
    50 var_dump(bcdiv($left=6, $right=5, 2));
    51 //1.20
    52  
    53  /**
    54   * 两个高精度数相乘
    55   * 
    56   * @access global
    57   * @param float $left
    58   * @param float $right
    59   * @param int $scale 精确到的小数点位数
    60   * 
    61   * @return string 
    62   */
    63 var_dump(bcmul($left=3.1415926, $right=2.4569874566, 2));
    64 //7.71
    65  
    66  /**
    67   * 设置bc函数的小数点位数
    68   * 
    69   * @access global
    70   * @param int $scale 精确到的小数点位数
    71   * 
    72   * @return void 
    73   */ 
    74 bcscale(3);
    75 var_dump(bcdiv('105', '6.55957')); 
    76 // 16.007
  • 相关阅读:
    在线客服系统前端多国语言实现方案和代码
    索引下推,这个点你肯定不知道!
    拿捏!隔离级别、幻读、Gap Lock、Next-Key Lock
    现在已经卷到需要问三色标记了吗?
    听说你对explain 很懂?
    面试官:你说说一条更新SQL的执行过程?
    面试官:你说说一条查询SQL的执行过程
    别再纠结线程池大小线程数量了,没有固定公式的
    记一次慢SQL优化
    缓存热点,缓存穿透,终极解决方案看过来
  • 原文地址:https://www.cnblogs.com/as3lib/p/6623943.html
Copyright © 2020-2023  润新知