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


相关推荐

  • PAT准备之2018.7.24

    昨天被我划水滑过去了,今天终于完成了救赎,基本没有划水,一直在认真的学习,今天也做了不少题,发现自己还是有很多知识点薄弱的地方,还是基础不太好吧,以前总觉得自己这些东西都会,结果发现真到自己用的时候,真的是不会。。。唉!这个暑假再把基础知识补一补吧。今天也是做了三道题。如下1007MaximumSubsequenceSum(25)(25分)Givenasequenceo…

    2022年4月9日
    51
  • smallint是sql的数据类型吗_char数据类型

    smallint是sql的数据类型吗_char数据类型环境:SQLServer2008R2 使用整数数据的精确数字数据类型。

    2025年12月2日
    6
  • 点菜宝基站信道设置_点菜宝怎么连接基站

    点菜宝基站信道设置_点菜宝怎么连接基站最近在跟菜鸟天地系统对接,业务中涉及到单点接入,感觉其实现思想很不错,现分享一下,供大家参考:从CP内部系统进入菜鸟天地,也就是从菜鸟的合作伙伴(物流或快递公司内部系统)单点跳转进入菜鸟天地系统红

    2022年8月4日
    40
  • ubuntu的ssh连不上_ubuntu网络连接没有显示出来

    ubuntu的ssh连不上_ubuntu网络连接没有显示出来之前发在其他的博客上的,现在移动以下位置之前的链接:http://blog.chinaunix.net/uid-69944074-id-5831708.html(原创文章)使用Ubuntu,经常需要需要SSH远程连接,但是有时候会出现问题,难以捉摸,下面参考别人的,在结合自己的尝试总结下吧。服务器配完ubuntu系统以及LNMP环境以后,想用WINSCP远程登录,就需要开启SSH服务才能支持。SSH服务分为客户端和服务器。顾名思义,我想用putty远程登录Ubuntu服务器,所以需要安装SSH s

    2022年8月8日
    8
  • keil MDK5搭建STM32开发环境

    keil MDK5搭建STM32开发环境1.安装keil到keil的官方网站http://www.keil.com/download/product/下载MDK-ARM并安装,注意可以更改安装路径,但是不能安在需要管理员权限的文件夹,例如不能在ProgramFiles,否则会出现一些问题。2.注册未注测有代码容量限制,需要破解。网上找KEIL_Lic。3.安装器件包打开PackInstaller左边选择STMicro

    2022年5月10日
    78
  • android碎片整理工具,Android的碎片整理「建议收藏」

    android碎片整理工具,Android的碎片整理「建议收藏」Android手机与平板高级历史橡皮擦和性能的助推器★★★★第一轮中的新工具支付Android上最流行的新的生产力应用程序,下载吧!清除历史搜索,杀敌多余的电池寿命,增加单键的任务!“碎片整理免费的Andr??oid现在是十大最热AppBrain.com应用之一,现在下载”“真棒应用程序”-编辑推荐奖AppEggs.com_________________________________我们现…

    2022年6月25日
    33

发表回复

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

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