2


我有一个 MySQL 表estimate_item,其中包含几列保存具有固定精度的十进制值。这是我的表结构。

表名:estimate_item

在此处输入图像描述

在使用 PHP 检索记录时,我使用 MySQL 的函数进行了一些基本计算,我使用的 SQL 查询就是这个。

SELECT 
    COUNT(ei.item_id) as total_item,
    SUM(ei.quantity) as total_quantity,
    SUM(ei.number_of_carton) as total_carton,
    SUM(ei.number_of_carton * ei.cbm_per_carton) as total_cbm,
    SUM(ei.quantity * ei.cost) as total_cost,
    SUM(ei.quantity * ei.rate) as total_rate,
    e.commission_amount,
    SUM(ei.quantity * ei.rate) + e.commission_amount as total_amount
FROM
    ai_estimate e
LEFT JOIN
    ai_estimate_item ei ON ei.estimate_id = e.id
WHERE
    ei.estimate_id = ?

例如上面的查询返回结果。

+------------+----------------+--------------+-----------+------------+------------+-------------------+--------------+
| total_item | total_quantity | total_carton | total_cbm | total_cost | total_rate | commission_amount | total_amount |
+------------+----------------+--------------+-----------+------------+------------+-------------------+--------------+
|          2 |            120 |          807 | 1122.6440 |     2.7500 |   137.5000 |           1500.00 |    1637.5000 |
+------------+----------------+--------------+-----------+------------+------------+-------------------+--------------+

由于以下列包含 4 个精度值total_cbm_total, total_cost, total_rate, total_amount,我想以这种方式对所有四列的值进行四舍五入。

a)如果值是1122.6440然后将其四舍五入到1122.644

b)如果值是2.7500然后将其四舍五入到2.75

c)如果值是137.5000然后将其四舍五入到137.50

d) 如果值是1200.0000然后将其四舍五入到1200.00

即我希望这些值包含至少两个精度,根据值可以达到 3 或 4 精度。

有什么办法可以直接在 MySQL 中做到这一点?还是 PHP 方式?

谢谢并恭祝安康。

4

3 回答 3

1

如果你在 MySQL 中这样做,我的猜测是你需要过度复杂化查询添加一个 CASE WHEN...END。

所以我会在 PHP 中做到这一点。

function reprecise($value)
{
    $two_digit = number_format($value, 2);
    $three_digit = number_format($value, 3);
    return (float)$two_digit == (float)$three_digit ? $two_digit : $three_digit;
}

这使用了一个技巧 - 首先它计算两位和三位数字的表示,然后检查它们是否是相同的浮点数。如果是,则三位版本的最后一位必须为零,而使用两位版本。

要扩展到 2、3 或 4 位数字,您可以在一个循环中执行此操作:

function reprecise($value)
{
    $digits = 4;
    for ($rep = number_format($value, $digits--); $digits >= 2; $digits--)
    {
            $new = number_format($value, $digits);
            if ($new != $rep)
                    break;
            $rep = $new;
    }
    return $rep;
}

或者,因为字符串函数很可能比 number_format 更快:

function reprecise($value)
{
    $digits = 9;
    $base   = strrev(number_format($value, $digits));
    for ($i = 0; $i < $digits-2; $i++)
            if ($base[$i] != '0')
                    break;
    return strrev(substr($base, $i));
}
于 2012-07-20T17:30:34.963 回答
1

一种方法是使用正则表达式删除0小数点后数字末尾的所有内容,然后格式化具有两个小数位的数字并比较两个结果 - 返回最长的一个(按字符串)。虽然这可以在 mysql 中完成,但在 php 中会更容易:

//$input - number to format
//$num - minimum number of decimal places to keep
function reformat($input, $num) {
    $parse = preg_replace('/(\.[1-9]*)0+$/', "$1", $input);
    $withmin = number_format($parse, 2); 
    return strlen($withmin) < strlen($parse) ? $parse : $withmin;
}
于 2012-07-20T17:45:46.193 回答
1

数据库方面,请注意列 cost、rate 和 cmb_per_carton 的尾随 0 的数量是由表结构预定义的,在您的情况下是 4 个 0:

cost DECIMAL 19,4
rate DECIMAL 19,4
cmb_per_carton DECIMAL 19,4

如果您对这些字段进行操作,结果自然会有相同的尾随 0。

PHP 方面是我刚刚破解的一个简单函数,它将完成您正在寻找的内容:

function getDecimalFormatted($value, $minPrecision) {
    $value = (float) $value;
    return ((round((($value - floor($value))*pow(10, 5)))%1000) > 0) ? $value : number_format($value, $minPrecision, '.', '');
}

使用示例:

$value = "1234.71200";
echo getDecimalFormatted($value, 2)."\n<br>";

$value = "1234.70000";
echo getDecimalFormatted($value, 2)."\n<br>";

这输出:

1234.712
1234.70
于 2012-07-20T18:22:34.400 回答