关于【BigDecimal】和Mysql中的decimal

> Mysql中的decimal的解释说明

decimal:数字型,不存在精度损失,常用于银行帐目计算

decimal(P,m)
P的取值范围是1–65,表示整数位数
M的取值是0–30.表示小数位
精度M最大为30,表示有30位小数
decimal整数和小数位均不写 默认为decimal(10,0)
若小数位不写默认的小数位为0

若存储的小数位比较多可以扩展整数位 decimal(22,6)整数位16个小数位6个。
在产品设计的时候可以约定。

decimal 还可以存储超过bigint的整数位,最大存储65位的整数类型。
还可以存储对应位数的负数

例:decimal(2,1),此时,插入数据“12.3”、“12”等会出现“数据溢出错误”的异常;插入“1.23”或“1.2345…”会自动四舍五入成“1.2”;插入“2”会自动补成“2.0”,以确保2位的有效长度,其中包含1位小数。
当用 int类型会有溢出时,可以用 decimal 类型进行处理,把结果可以用 convert 或是 cast 进行转换。

>关于java中BigDecimal的简介

1、BigDecimal属于大数据,精度极高,不属于基本数据类型,属于java对象(引用数据类型), 这是sun提供的一个类,专门用在财务软件中。
2、注意:财务软件中double是不够用的。
3、关于BigDecimal的方法

方法 说明
abs()函数 绝对值
toString(); 将BigDecimal对象的数值转换成字符串。
setScale(int newScale, RoundingMode roundingMode); 需要对BigDecimal进行截断和四舍五入可用setScale方法
compareTo(BigDecimal val); 和另一个BigDecimal对比 等于1表示大于、等于0表示等于、等于-1表示小于
add(BigDecimal augend);
subtract(BigDecimal subtrahend);
multiply(BigDecimal multiplicand);
divide(BigDecimal divisor, int scale, RoundingMode roundingMode); 除,第一参数表示被除数,第二个参数表示小数点后保留位数,第三个参数表示取舍规则(例如:四舍五入)

RoundingMode------说明(可以参考RoundingMode类)
ROUND_UP舍入远离零的舍入模式。1.1->2,-2.5->-3
ROUND_DOWN接近零的舍入模式。1.6->1,-2.5->-2
ROUND_CEILING接近正无穷大的舍入模式。
ROUND_FLOOR接近负无穷大的舍入模式。
ROUND_HALF_UP四舍五入
ROUND_HALF_DOWN
ROUND_HALF_EVEN
ROUND_UNNECESSARY

部分示例代码:

public class BigDecimalTest01 {
   
    public static void main(String[] args) {
   
//        这个100是精度极高的100
        BigDecimal v1 = new BigDecimal(100);
//        这个200是精度极高的200
        BigDecimal v2 = new BigDecimal(200);
//        求和// v1+v2 这样不行,v1和v2都是引用,不能直接使用+求和
  BigDecimal v3 = v1.add(v2);//调用方法求和
        System.out.println(v3);//300

        BigDecimal v4 = v2.divide(v1);//调用方法求商
        System.out.println(v4);//2
    }
}

其他方法调用类似

>>int与bigdecimal的相互转换

int转bigdecimal

BigDecimal number = new BigDecimal(0);
int value=score;
number=BigDecimal.valueOf((int)value);

bigdecimal转int

BigDecimal b=new BigDecimal(45.45);

int a = b.intValue();

>>String与bigdecimal的互转

double必须用作BigDecimal的源时,可以使用BigDecimal的静态方法valueOf

BigDecimal num = NumberUtils.createBigDecimal(“123”);
org.apache.commons.lang.math

或者直接用bigdecimal的构造方法转换

>>转换中的精度丢失问题

注意:使用构造方法只能转换 String 和 int 类型,如果转换float或者double等浮点类型,会出现精度丢失的问题。原因:
我们点开构造器方法看下源码:

public static long doubleToLongBits(double value) {
   
    long result = doubleToRawLongBits(value);
    // Check for NaN based on values of bit fields, maximum
    // exponent and nonzero significand.
    if ( ((result & DoubleConsts.EXP_BIT_MASK) ==
          DoubleConsts.EXP_BIT_MASK) &&
         (result & DoubleConsts.SIGNIF_BIT_MASK) != 0L)
        result = 0x7ff8000000000000L;
    return result;
}

问题就处在 doubleToRawLongBits 这个方法上,在 jdk 中 double 类(float 与 int 对应)中提供了 double 与 long 转换,doubleToRawLongBits 就是将 double 转换为 long,这个方法是原始方法(底层不是 java 实现,是 c++ 实现的)。

double 之所以会出问题,是因为小数点转二进制丢失精度。
在这里插入图片描述
所以,在涉及到精度计算的过程中,我们尽量使用 String 类型/或者转换为String再 来进行转换。

>>关于浮点类型的加减乘除 这里整理出一个 util 类供大家使用:

import java.math.BigDecimal;


public class BigDecimalUtils {
   
    public static BigDecimal doubleAdd(double v1, double v2) {
   
        BigDecimal b1 = new BigDecimal(Double.toString(v1));
        BigDecimal b2 = new BigDecimal(Double.toString(v2));
        return b1.add(b2);
    }
    public static BigDecimal floatAdd(float v1, float v2) {
   
        BigDecimal b1 = new BigDecimal(Float.toString(v1));
        BigDecimal b2 = new BigDecimal(Float.toString(v2));
        return b1.add(b2);
    }
    public static BigDecimal doubleSub(double v1, double v2) {
   
        BigDecimal b1 = new BigDecimal(Double.toString(v1));
        BigDecimal b2 = new BigDecimal(Double.toString(v2));
        return b1.subtract(b2);
    }
    public static BigDecimal floatSub(float v1, float v2) {
   
        BigDecimal b1 = new BigDecimal(Float.toString(v1));
        BigDecimal b2 = new BigDe
  • 1
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值