解决double转 BigDecimal 时出现的精度失真问题

解决double转 BigDecimal 时出现的精度失真问题解决double转BigDecimal时出现的精度问题比如,doubledd=344999.03d;转成BigDecimal类型,BigDecimalss=newBigDecimal(dd);最后,ss的值是344999.03000000002793967723846435546875失真了。解决方法是先将dd转换字符串,然后转换成BigDecimal。/…

大家好,又见面了,我是你们的朋友全栈君。

比如,double dd=344999.03d;
转成 BigDecimal 类型,BigDecimal ss=new BigDecimal(dd);
打印 ss 的值是344999.03000000002793967723846435546875
精度失真啦!

解决方法是
先将 double 转换 字符串,
然后转换成 BigDecimal 。

代码:

/** * 解决double转bigdecimal时出现的精度问题 * @param v1 * @return */
public static BigDecimal doubleToBig(double v1) { 
   
	return new BigDecimal(String.valueOf(v1));
}

查询BigDecimal 源码,得出结论:
所有的 基本数字类型(float、double)转换 BigDecimal 或 BigInteger时,

先将 数字类型 先转成字符串,然后再转换BigDecimal 或 BigInteger。

最后,附上 double转 BigDecimal 的工具类

package com.delongra.nback.system.util;

import java.math.BigDecimal;
import java.text.DecimalFormat;
import java.text.NumberFormat;


/** * 由于Java的简单类型不能够精确的对浮点数进行运算,这个工具类提供精 确的浮点数运算,包括加减乘除和四舍五入。 */
public class DoubleUtil { 
   

	// 默认除法运算精度
	private static final int DEF_DIV_SCALE = 10;
	/**#.## */
	public static final String REG_1 = "#.##";
	/** #,###.## */
	public static final String REG_2 = "#,###.##";

	// 这个类不能实例化
	private DoubleUtil() { 
   
	}

	/** * 解决double转bigdecimal时出现的精度问题 * @param v1 * @return */
	public static BigDecimal doubleToBig(double v1) { 
   
		return new BigDecimal(String.valueOf(v1));
	}
	
	
	/** * 科学记数法转换成字符串 */
	public static String doubleToStr(double v1) { 
   
		BigDecimal b1 = new BigDecimal(Double.toString(v1));
		return b1.toPlainString();
	}
	/** * 科学记数法转换成字符串 */
	public static String stringToStr(String v1) { 
   
		BigDecimal b1 = new BigDecimal(v1);
		return b1.toPlainString();
	}
	
	/** * 主要用于格式化金额 * @param v1 * @return */
	public static String format(double v1) { 
   
		NumberFormat numberFormat = new DecimalFormat(REG_2);
		String str = numberFormat.format(v1);
		return str;
	}
	
	/** * 主要用于格式化小数点 * @param v1 * @param reg * @return */
	public static String format(double v1,String reg) { 
   
		NumberFormat numberFormat = new DecimalFormat(reg);
		String str = numberFormat.format(v1);
		return str;
	}
	
	
	/** * 加法 */
	public static double add(double v1, double ...v2) { 
   
		BigDecimal b1 = new BigDecimal(Double.toString(v1));
		for (double vv : v2){ 
   
			BigDecimal b2 = new BigDecimal(Double.toString(vv));
			b1=b1.add(b2);
		}
		return b1.doubleValue();
	}

	/** * 减法 */
	public static double sub(double v1, double ...v2) { 
   
		BigDecimal b1 = new BigDecimal(Double.toString(v1));
		for (double vv : v2){ 
   
			BigDecimal b2 = new BigDecimal(Double.toString(vv));
			b1=b1.subtract(b2);
        }
		
		return b1.doubleValue();
	}

	
	/** * 乘法 */
	public static double mul(double v1,double ...v2) { 
   
		BigDecimal b1=new BigDecimal(Double.toString(v1));
		for (double vv : v2){ 
   
			BigDecimal b2 = new BigDecimal(Double.toString(vv));
			b1=b1.multiply(b2);
        }
		return b1.doubleValue();
	}

	/** * 除法 */

	public static double div(double v1, double ...v2) { 
   
		BigDecimal b1 = new BigDecimal(Double.toString(v1));
		for (double vv : v2){ 
   
			BigDecimal b2 = new BigDecimal(Double.toString(vv));
			b1=b1.divide(b2, DEF_DIV_SCALE, BigDecimal.ROUND_HALF_UP);
        }
		return b1.doubleValue();
	}

	/** * 小数的四舍五入 */

	public static double round(double v, int scale) { 
   
		if (scale < 0) { 
   
			throw new IllegalArgumentException(
					"The scale must be a positive integer or zero");
		}
		BigDecimal b = new BigDecimal(Double.toString(v));
		BigDecimal one = new BigDecimal("1");
		return b.divide(one, scale, BigDecimal.ROUND_HALF_UP).doubleValue();
	}
	
	/** * 两个数比较,获取比较值相对于基准值的大小百分比 * @param base 基准值 * @param compare 比较值 * @return */
	public static String getComparePercent(int base,int compare){ 
   
		if (base==compare) { 
   
			return "0%";
		}
		if (compare==0) { 
   
			return "0%";
		}
		if (base==0) { 
   
			return compare*100+"%";
		}
        NumberFormat numberFormat = NumberFormat.getInstance();  
        numberFormat.setMaximumFractionDigits(2); // 设置精确到小数点后2位 
        int margin = compare-base;
        String result = numberFormat.format((float) margin / (float) base * 100);  
		return result + "%";
	}	
}

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/137391.html原文链接:https://javaforall.net

(0)
上一篇 2022年6月2日 上午7:00
下一篇 2022年6月2日 上午7:00


相关推荐

  • Trae IDE 新手使用教程:从安装到项目开发全攻略

    Trae IDE 新手使用教程:从安装到项目开发全攻略

    2026年3月16日
    3
  • msm8953 uart配置

    msm8953 uart配置参考链接:MSM8937-MSM8953UART配置调试指南uart驱动是使用内核驱动,无需自己编写,一般只需修改设备树。一、修改设备树设备树的配置有三部分:1.msm8953.dtsi中添加code blsp2_uart2:serial@7af0000{ compatible=”qcom,msm-lsuart-v14″; reg=<0x7af00000x…

    2022年8月31日
    6
  • 2020idea安装教程_3dmax2020安装失败

    2020idea安装教程_3dmax2020安装失败IDEA安装教程1、下载IDEA首先在官网下载IDEA,官网下载地址为:IDEA下载地址我这里选择的是Ultimate版本2、安装IDEA双击刚才下载好的IDEA安装包进行安装。我把安装路径改为了我在D盘选择的路径点击Next,然后点击Install进行安装安装过程安装成功3、IDEA配置启动IDEA此时需要激活,我之前申请的教育账号还可以用就直接使用的教育账…

    2026年4月17日
    5
  • MSVCR110.dll缺失问题

    MSVCR110.dll缺失问题电脑出现无法启动此程序 因为计算机中丢失 MSVCR110 dll 尝试重新安装该程序以解决此问题 是此程序的成功运行需要 msvcr110 dll 文件来支持 1 下载 msvcr110 dll 文件 2 将 msvcr110 dll 文件直接拷贝到 c Windows System32 文件夹目录下 例如图

    2025年10月2日
    3
  • win10安装ubuntu「建议收藏」

    win10上安装ubuntu为了能更好地学python,本来打算装个双系统,用Linux写python,不过发现双系统切换起来麻烦了点,然后就发现有虚拟机这东西。花费了一些时间,最后成功通过VMwareWorkstationPro14虚拟机安装了ubuntu,在此将安装教程整合一下,供需要者参考。 1.安装VMwareWorkstationPro14虚拟机要安装ubun…

    2022年4月13日
    53
  • 公网IP和内网IP的区别[通俗易懂]

    公网IP和内网IP的区别[通俗易懂]最近在学习shell反弹这块的时候,在向源主机发送报文的时候,由于不了解公网IP和内网IP的区别,导致在监听端口这块一直没有捕获到信息,后来才知道是因为我用的公司的局域网是192开头的,属于内网,因此只能在内部进行通信,而不能与其他网络互连。因为本网络中的保留地址同样也可能被其他网络使用,如果进行网络互连,那么寻找路由时就会因为地址的不唯一而出现问题。因此总结下内网和公网的区别。内网,一般来说,也就是局域网,我们可以把局域网理解成一个小家庭,然后给我们每个家庭的成员都编上号,比如张三是192.168.1.

    2022年4月27日
    58

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号