解决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)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 关于Cloneable接口和clone方法「建议收藏」

    关于Cloneable接口和clone方法「建议收藏」1、使用创建对象有两种方式:new和clone当一个对象创建过程复杂,我们是否可以根据已有的对象直接来克隆一份,而不必关系创建的细节呢(原型模式)。1.1JavaObject根类默认提

    2022年7月2日
    20
  • Unity3D安装_3D杀号APP

    Unity3D安装_3D杀号APPUnity3D是一个跨平台的游戏开发引擎,它能够帮助开发者快速构建游戏场景,进行游戏开发与发布。它支持将开发好的作品发布到Windows、MacOS、Android、iOS、Linux等多种平台。另外,Unity3D官方版本目前可以运行在Windows和MacOS上,当然还有可以运行在Linux上的非官方版本。

    2022年8月10日
    4
  • java集合系列——Map之TreeMap介绍(九)

    TreeMap是一个有序的key-value集合,基于红黑树(Red-Black tree)的 NavigableMap实现。该映射根据其键的自然顺序进行排序,或者根据创建映射时提供的 Comparator进行排序,具体取决于使用的构造方法。

    2022年2月26日
    116
  • 一致性哈希算法实现(一致性哈希与哈希的异同)

    1、使用哈希算法有什么问题?假设有一个由A、B、C三个节点组成的KV服务,每个节点存放不同的KV数据。通过哈希算法,每个key都可以寻址到对应的服务器,比如,查询key是key-01,计算公式为hash(key-01)%3,经过计算寻址到了编号为1的服务器节点A但如果服务器数量发生变化,基于新的服务器数量来执行哈希算法的时候,就会出现路由寻址失败的情况,Proxy无法找到之前寻址到的那个服务器节点假如3个节点不能满足业务需求了,这时增加了一个节点,节点的数量从3变化为4,那么之前的hash(key

    2022年4月14日
    41
  • 漂亮的个人博客模板

    漂亮的个人博客模板2021年了,个人博客还有人做吗?确实现在已经比较少人写博客了吧,那么2021年了个人博客还能不做?我的回答是只要搜索引擎存在,无论是个人博客还是其它网站就会一直长存。

    2022年7月1日
    27
  • CSS画猪

    效果查看:http://hovertree.com/texiao/css3/6/CSS3画猪头:http://hovertree.com/texiao/css3/6/1/代码如下:转自:htt

    2021年12月22日
    36

发表回复

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

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