项目中遇到了数值运算,如网上所写的,一般有这几个方法:

  1. /**
  2. * 提供精确的加法运算。
  3. * @param v1 被加数
  4. * @param v2 加数
  5. * @return 两个参数的和
  6. */
  7. public static double add(double v1, double v2) {
  8. BigDecimal b1 = new BigDecimal(Double.toString(v1));
  9. BigDecimal b2 = new BigDecimal(Double.toString(v2));
  10. return (b1.add(b2)).doubleValue();
  11. }
  12. /**
  13. * 提供精确的减法运算。
  14. * @param v1 被减数
  15. * @param v2 减数
  16. * @return 两个参数的差
  17. */
  18. public static double sub(double v1, double v2) {
  19. BigDecimal b1 = new BigDecimal(Double.toString(v1));
  20. BigDecimal b2 = new BigDecimal(Double.toString(v2));
  21.  
  22. return (b1.subtract(b2)).doubleValue();
  23. }
  24. /**
  25. * 提供精确的乘法运算。
  26. * @param v1 被乘数
  27. * @param v2 乘数
  28. * @return 两个参数的积
  29. */
  30. public static double mul(double v1, double v2) {
  31. BigDecimal b1 = new BigDecimal(Double.toString(v1));
  32. BigDecimal b2 = new BigDecimal(Double.toString(v2));
  33. return (b1.multiply(b2)).doubleValue();
  34. }
  35. /**
  36. * 提供(相对)精确的除法运算。当发生除不尽的情况时,由scale参数指
  37. * 定精度,以后的数字四舍五入。
  38. * @param v1 被除数
  39. * @param v2 除数
  40. * @param scale 表示需要精确到小数点以后几位。
  41. * @return 两个参数的商
  42. */
  43. public static double div(double v1, double v2, int scale) {
  44. if (scale < 0) {
  45. System.err.println("除法精度必须大于0!");
  46. return 0;
  47. }
  48. BigDecimal b1 = new BigDecimal(Double.toString(v1));
  49. BigDecimal b2 = new BigDecimal(Double.toString(v2));
  50. return (b1.divide(b2, scale, BigDecimal.ROUND_HALF_UP)).doubleValue();
  51. }

但是这里,我还有两个地方需要做一些变化:

1、乘法运算需要做精度的四舍五入

2、double会自动进行科学计数法

对于第一点新增mul方法

  1. public static double mul(double v1, double v2,int scale) {
  2. BigDecimal b1 = new BigDecimal(Double.toString(v1));
  3. BigDecimal b2 = new BigDecimal(Double.toString(v2));
  4. return (b1.multiply(b2).setScale(scale, RoundingMode.HALF_UP)).doubleValue();
  5. }

这样就可以对所做的运算进行需要的精度的四舍五入

第二点新增double2Str方法

  1. private static String double2Str(double number)
  2. {
  3. return DecimalFormat.getInstance().format(number).replace(",","");
  4. }

不需要进行格式化,因为项目中把科学计数法的数字传给客户端.Net接收,无法正确转换成数值(除非客户端特殊处理这个科学计数法数值)

当时,对第一点做了一个错误的精度处理:MathContext

  1. public static double mul(double v1, double v2,int scale) {
  2. BigDecimal b1 = new BigDecimal(Double.toString(v1));
  3. BigDecimal b2 = new BigDecimal(Double.toString(v2));

return b1.multiply(b2, new MathContext(scale,RoundingMode.HALF_UP)).doubleValue();

  1. }

这个MathContext是从左边第一个数开始算起,而不是从小数点后开始算

补充一下舍入RoundingMode

在RoundingMode中HALF_DOWN 没有按照我预期的对数值进行向下取数值有点疑惑

  1. double d3 = 102207767.555;
  2. d3=new BigDecimal(d3).setScale(2,RoundingMode.HALF_DOWN).doubleValue();

打印后d3显示102207767.56 不应该是102207767.55?

  1.  

ROUND_CEILING
Rounding mode to round towards positive infinity.
向正无穷方向舍入
ROUND_DOWN
Rounding mode to round towards zero.
向零方向舍入
ROUND_FLOOR
Rounding mode to round towards negative infinity.
向负无穷方向舍入

ROUND_HALF_DOWN

Rounding mode to round towards "nearest neighbor" unless both neighbors are equidistant, in which case round down.
向(距离)最近的一边舍入,除非两边(的距离)是相等,如果是这样,向下舍入, 例如1.55 保留一位小数结果为1.5
ROUND_HALF_EVEN
Rounding mode to round towards the "nearest neighbor" unless both neighbors are equidistant, in which case, round towards the even neighbor.
向(距离)最近的一边舍入,除非两边(的距离)是相等,如果是这样,如果保留位数是奇数,使用ROUND_HALF_UP ,如果是偶数,使用ROUND_HALF_DOWN
ROUND_HALF_UP
Rounding mode to round towards "nearest neighbor" unless both neighbors are equidistant, in which case round up.
向(距离)最近的一边舍入,除非两边(的距离)是相等,如果是这样,向上舍入, 1.55保留一位小数结果为1.6

ROUND_UNNECESSARY

Rounding mode to assert that the requested operation has an exact result, hence no rounding is necessary.
计算结果是精确的,不需要舍入模式
ROUND_UP
Rounding mode to round away from zero.
向远离0的方向舍入

简单BigDecimal运算精度的更多相关文章

  1. BigDecimal带精度的运算的两篇文章

    转自:http://guoliangqi.iteye.com/blog/670908 之前提到过在商业运算中要使用BigDecimal来进行相关的钱的运算(java中关于浮点运算需要注意的 ),可是实 ...

  2. 关于java中Double类型的运算精度问题

    标题     在Java中实现浮点数的精确计算    AYellow(原作) 修改    关键字     Java 浮点数 精确计算   问题的提出:如果我们编译运行下面这个程序会看到什么?publi ...

  3. 160918、BigDecimal运算

    java.math.BigDecimal.BigDecimal一共有4个够造方法,让我先来看看其中的两种用法: 第一种:BigDecimal(double val)Translates a doubl ...

  4. 关于java中Double类型的运算精度问题(转)

    Java Java double:浮点数:精确计算  public class Test{    public static void main(String args[]){        Syst ...

  5. #define与运算精度问题探究

    #include <stdio.h> #define SQR(X) X*X int main(int argc, char* argv[]) { ; ; ; printf("SQ ...

  6. 搞懂js中小数运算精度问题原因及解决办法

    js小数运算会出现精度问题 js number类型 JS 数字类型只有number类型,number类型相当于其他强类型语言中的double类型(双精度浮点型),不区分浮点型和整数型. number类 ...

  7. JavaScript 浮点数及运算精度调整总结

    JavaScript 浮点数及运算精度调整总结 JavaScript 只有一种数字类型 Number,而且在Javascript中所有的数字都是以IEEE-754标准格式表示的.浮点数的精度问题不是J ...

  8. PHP浮点数运算精度造成的,订单金额支付经常少1分的问题

    最近碰见一个奇怪的问题,商城通过微信支付的订单经常少一分钱,经过排查是PHP浮点运算精度问题造成的 由PHP浮点数运算精度造成的,鸟哥的Bolg有详细的说明.http://www.laruence.c ...

  9. Java 避免精度丢失之BigDecimal 运算

    * 由于Java的简单类型不能够精确的对浮点数进行运算,这个工具类提供精确的浮点数运算,包括加减乘除和四舍五入 import java.math.BigDecimal; /** 计算工具类 */ pu ...

随机推荐

  1. PTA Strongly Connected Components

    Write a program to find the strongly connected components in a digraph. Format of functions: void St ...

  2. Android多线程编程之AsyncTask

    进程?线程? 进程是并发执行的程序在执行过程中分配和管理资源的基本单位,是一个动态的概念.每个进程都有自己的地址空间(进程空间).进程空间的大小与处理机位数有关.进程至少有5种基本状态:初始态,执行态 ...

  3. RHEL6.7 x64双节点安装Oracle 11g r2 RAC

    基础环境 使用两台HP DL580服务器作为RAC节点,存储使用IBM V7000.具体环境如下: 设备 用途 IP地址 磁盘空间 HP DL580 RAC节点01 RAC01-pub:116.1.1 ...

  4. Hibernate控制台显示创建数据库表语句

    package cqvie.yjq.View; import org.hibernate.Session; import org.hibernate.Transaction; import org.h ...

  5. SQL 计算两个地理坐标相差的距离的函数

    Create function getGreatCircleDistance(@lat1 decimal(18,11),@lng1 decimal(18,11),@lat2 decimal(18,11 ...

  6. 淌水 UE4的shootergame 案例 准备

    从毕业到现在,从GIS到游戏. 先记录一下cesium源码研究停止了一个多月了,还是有点放不下,等有机会一定研究透彻.感谢一下法克鸡丝博主. 好,研究了近两个月的游戏整体制作,熟悉了maya\unfl ...

  7. centos7引导项修复

    每次装了双系统,都会发现原来的windows引导项不见了,这让我这个windows重度依赖者情何以堪,所以,必须要把我挚爱的windows给找回来. 翻看了一些网上的教程,看来这并不是一个困难的问题. ...

  8. Oracle 中的作业队列和队列调度

    一,启动执行作业的进程       在 Oracle 中,是使用 “作业队列协调进程(CJQ0)” 这个协调数据库实例的作业队列的后台进程,来监视作业队列中的作业表(JOB$),并启动作业队列进程(J ...

  9. 浅谈php设计模式(1)---工厂模式

    一.接口继承直接调用 先看看这样一段代码: <?php interface db{ function conn(); } class dbmysql implements db { public ...

  10. CentOS6系统openssl生成证书和自签证书

    CentOS6系统openssl生成证书和自签证书的过程,记录一下,本文基于CentOS 6 64bit.$ yum install openssl openssl-devel 1,生成服务器端的私钥 ...