SpringBoot--解决BigDecimal传到前端后精度丢失的问题

article/2025/10/1 5:44:20

原文网址:SpringBoot--解决BigDecimal传到前端后精度丢失的问题_IT利刃出鞘的博客-CSDN博客

简介

本文用示例介绍SpringBoot如何解决BigDecimal传到前端后精度丢失问题。

解决方案是:转成字符串格式返回。

问题描述

实例

Controller

package com.knife.controller;import com.knife.entity.UserVO;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;import java.math.BigDecimal;@RestController
@RequestMapping("user")
public class UserController {@GetMapping("save")public UserVO save(BigDecimal amount) {UserVO userVO = new UserVO();userVO.setId(1L);userVO.setUsername("Tony");userVO.setAmount(amount);return userVO;}
}

Entity

package com.knife.entity;import lombok.Data;import java.math.BigDecimal;@Data
public class UserVO {private Long id;private String username;private BigDecimal amount;
}

测试

访问:http://localhost:8080/user/save?amount=12345671234567.1234

结果

问题复现

场景描述

        实际项目中前端会这样处理:调用后端接口获得JSON格式的响应字符串,然后将JSON字符串解析为JavaScript对象(用于展示到对应的位置、方便计算等)。

        前端调后端的写接口(增删改)时,会将JavaScript对象序列化为JSON格式的字符串,然后将其作为参数请求后端接口。

 实例1:精度丢失

const json = '{"id": 1, "name": "Tony", "amount": 12345671234567.12345}';
const obj = JSON.parse(json);
console.log(obj.amount);  // 12345671234567.123
console.log(JSON.stringify(obj));  // {"id":1,"name":"Tony","amount":12345671234567.123}

可以看到,在将json字符串转为JavaScript对象后,“amount” 丢失了精度。

实例2:丢失小数位

const json = '{"id": 1, "name": "Tony", "amount": 12345671234567.00000}';
const obj = JSON.parse(json);
console.log(obj.amount);  // 12345671234567
console.log(JSON.stringify(obj));  // {"id":1,"name":"Tony","amount":12345671234567}

可以看到,在将json字符串转为JavaScript对象后,“amount” 丢失了小数。

其他示例

const json = '{"id": 1, "name": "Tony", "amount": 12345671234567.12345}';
const obj = JSON.parse(json);
console.log(obj.amount);  // 12345671234567.123const json = '{"id": 1, "name": "Tony", "amount": 123456712345678.12345}';
const obj = JSON.parse(json);
console.log(obj.amount);  // 123456712345678.12const json = '{"id": 1, "name": "Tony", "amount": 98765432198765.12345}';
const obj = JSON.parse(json);
console.log(obj.amount);  // 98765432198765.12const json = '{"id": 1, "name": "Tony", "amount": 987654321987654321.12345}';
const obj = JSON.parse(json);
console.log(obj.amount);  // 987654321987654300

Java后端BigDecimal的范围

  1. 范围没有限制,可以认为无限大、无限小
  2. 可以通过如下代码验证:
    1. package com.example.a;import java.math.BigDecimal;public class Demo {public static void main(String[] args) {BigDecimal bigDecimal = new BigDecimal("1234567890123456789012345678901234567890"+ "1234567890123456789012345678901234567890"+ ".123456789");System.out.println(bigDecimal);}
      }

      执行结果:

      12345678901234567890123456789012345678901234567890123456789012345678901234567890.123456789

解决方案

        把BigDecimal的序列化值改成字符串类型即可。

方案1:全局处理

法1:ToStringSerializer

配置类

package com.knife.config;import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.module.SimpleModule;
import com.fasterxml.jackson.databind.ser.std.ToStringSerializer;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.converter.json.Jackson2ObjectMapperBuilder;import java.math.BigDecimal;@Configuration
public class JacksonConfig {@Beanpublic ObjectMapper jacksonObjectMapper(Jackson2ObjectMapperBuilder builder) {ObjectMapper objectMapper = builder.createXmlMapper(false).build();// 全局配置序列化返回 JSON 处理SimpleModule simpleModule = new SimpleModule();// 将使用String来序列化BigDecimal类型simpleModule.addSerializer(BigDecimal.class, ToStringSerializer.instance);objectMapper.registerModule(simpleModule);return objectMapper;}
}

测试

访问:http://localhost:8080/user/save?amount=12345671234567.1234

结果:

法2:自定义序列化

自定义序列化器

package com.knife.config;import com.fasterxml.jackson.core.JsonGenerator;
import com.fasterxml.jackson.databind.SerializerProvider;
import com.fasterxml.jackson.databind.annotation.JacksonStdImpl;
import com.fasterxml.jackson.databind.jsontype.TypeSerializer;
import com.fasterxml.jackson.databind.ser.std.ToStringSerializer;import java.io.IOException;
import java.math.BigDecimal;
import java.math.RoundingMode;@JacksonStdImpl
class BigDecimalToStringSerializer extends ToStringSerializer {public final static BigDecimalToStringSerializer instance = new BigDecimalToStringSerializer();public BigDecimalToStringSerializer() {super(Object.class);}public BigDecimalToStringSerializer(Class<?> handledType) {super(handledType);}@Overridepublic boolean isEmpty(SerializerProvider prov, Object value) {if (value == null) {return true;}String str = ((BigDecimal) value).stripTrailingZeros().toPlainString();return str.isEmpty();}@Overridepublic void serialize(Object value, JsonGenerator gen, SerializerProvider provider)throws IOException {gen.writeString(((BigDecimal) value).stripTrailingZeros().toPlainString());// 如果要求所有BigDecimal保留两位小数,可以这么写:// gen.writeString(((BigDecimal) value).setScale(2, RoundingMode.HALF_UP)//         .stripTrailingZeros().toPlainString());}@Overridepublic void serializeWithType(Object value, JsonGenerator gen,SerializerProvider provider, TypeSerializer typeSer)throws IOException {// no type info, just regular serializationserialize(value, gen, provider);}
}

配置类

package com.knife.config;import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.module.SimpleModule;
import com.fasterxml.jackson.databind.ser.std.ToStringSerializer;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.converter.json.Jackson2ObjectMapperBuilder;import java.math.BigDecimal;@Configuration
public class JacksonConfig {@Beanpublic ObjectMapper jacksonObjectMapper(Jackson2ObjectMapperBuilder builder) {ObjectMapper objectMapper = builder.createXmlMapper(false).build();// 全局配置序列化返回 JSON 处理SimpleModule simpleModule = new SimpleModule();// 将使用String来序列化BigDecimal类型simpleModule.addSerializer(BigDecimal.class, BigDecimalToStringSerializer.instance);objectMapper.registerModule(simpleModule);return objectMapper;}
}

测试

访问:http://localhost:8080/user/save?amount=12345671234567.1234

结果:

方案2:局部处理

法1:@JsonSerialize

在相应字段上加此注解:

@JsonSerialize(using= ToStringSerializer.class)

示例

package com.knife.entity;import com.fasterxml.jackson.databind.annotation.JsonSerialize;
import com.fasterxml.jackson.databind.ser.std.ToStringSerializer;
import lombok.Data;import java.math.BigDecimal;@Data
public class UserVO {private Long id;private String username;@JsonSerialize(using= ToStringSerializer.class)private BigDecimal amount;
}

测试

访问:http://localhost:8080/user/save?amount=12345671234567.1234

结果:

法2:@JsonFormat

 在相应字段上加此注解:

@JsonFormat(shape = JsonFormat.Shape.STRING)

示例

package com.knife.entity;import com.fasterxml.jackson.annotation.JsonFormat;
import lombok.Data;import java.math.BigDecimal;@Data
public class UserVO {private Long id;private String username;@JsonFormat(shape = JsonFormat.Shape.STRING)private BigDecimal amount;
}

测试

访问:http://localhost:8080/user/save?amount=12345671234567.1234

结果:

法3:自定义序列化(推荐)

上边两种方法都存在问题,比如:用户输入了23,然后存入数据库。数据库实际存的是:23.00,用户查看的时候,发现数据变成了:23.00。

解决方法:自定义序列化,去掉末尾无用的0。

自定义的序列化类

package com.knife.common.format.serializer;import com.fasterxml.jackson.databind.annotation.JacksonStdImpl;
import com.fasterxml.jackson.databind.ser.std.ToStringSerializerBase;import java.math.BigDecimal;/*** BigDecimal序列化,去掉末尾的0,并返回String类型* 参考:{@link com.fasterxml.jackson.databind.ser.std.ToStringSerializer}*/
@JacksonStdImpl
@SuppressWarnings("serial")
public class BigDecimalStripAndToStringSerializerextends ToStringSerializerBase
{public final static BigDecimalStripAndToStringSerializer instance =new BigDecimalStripAndToStringSerializer();public BigDecimalStripAndToStringSerializer() { super(Object.class); }public BigDecimalStripAndToStringSerializer(Class<?> handledType) {super(handledType);}@Overridepublic final String valueToString(Object value) {if (!(value instanceof BigDecimal)) {String message = String.format("仅用于BigDecimal类型,不可用于%s",value.getClass().getName());throw new RuntimeException(message);}// 这里不需要判断null。如果是null,框架不会再调用此方法。// 官方的ToStringSerializer也是没有判断nullBigDecimal bigDecimal = (BigDecimal) value;return bigDecimal.stripTrailingZeros().toPlainString();}
}

用法

​
package com.knife.entity;import com.fasterxml.jackson.databind.annotation.JsonSerialize;
import com.knife.common.format.serializer;
import lombok.Data;import java.math.BigDecimal;@Data
public class UserVO {private Long id;private String username;@JsonSerialize(using= BigDecimalStripAndToStringSerializer.class)private BigDecimal amount;
}


http://chatgpt.dhexx.cn/article/jTSQXbhx.shtml

相关文章

BigDecimal 简单使用

目录 为什么使用BigDecimal 解决方案 构造方法 类型转换 double 转 BigDecimal BigDecimal 转 String BigDecimal 转 double/int/long等 加减乘除取余 divide 舍入模式 比较大小 格式化(DecimalFormat) 小结 为什么使用BigDecimal 1.float和double类型的主要设计目…

BigDecimal用法

文章目录 1、创建一个BigDecimal对象1.1、常用构造函数1.1、常用计算方法 2、BigDecimal很方便的几个用法2.1、用BigDecimal去除小数点后多余的0&#xff1a;stripTrailingZeros()2.2、BigDecimal的原值和科学计数值2.3、用BigDecimal比较大小&#xff1a;compareTo()2.4、BigD…

mysql bigdecimal查询_mysql bigdecimal

java学习:Java中的其它类 568x573 - 55KB - JPEG Java中BigDecimal类你了解多少?! 720x480 - 30KB - JPEG TypeHandlers 640x396 - 25KB - JPEG BigDecimal equals方法可能不相等 678x260 - 9KB - JPEG

BigDecimal 详解

一&#xff0c;BigDecimal的简介 Java在java.math包中提供的API类BigDecimal&#xff0c;用来对超过16位有效位的数进行精确的运算。 双精度浮点型变量double可以处理16位有效数。在实际应用中&#xff0c;需要对更大或者更小的数进行运算和处理。float和double只能用来做科学…

BigDecimal加减乘除计算

BigDecimal的运算——加减乘除 首先是bigdecimal的初始化 这里对比了两种形式&#xff0c;第一种直接value写数字的值&#xff0c;第二种用string来表示 我们对其进行加减乘除绝对值的运算,其实就是Bigdecimal的类的一些调用 BigDecimal num1 new BigDecimal(0.005);BigDecim…

java 中 BigDecimal 详解

首先&#xff0c;学习一个东西&#xff0c;我们都必须要带着问题去学&#xff0c;这边我分为 【为什么&#xff1f;】【是什么&#xff1f;】【怎么用&#xff1f;】 【为什么要用BigDecimal&#xff1f;】 首先&#xff0c;我们先看一下&#xff0c;下面这个现象 那为什么会…

Java —— JDBC关闭Statement后是否还需要关闭ResultSet?

一、问题描述 下面的代码使用了try-with-resource语法&#xff0c;会自动关闭Connection和Statement,是否还需要关闭ResultSet&#xff1f; Statement关闭后ResultSet会被回收么&#xff1f; 二、JDBC规范 JDBC规范4.3中有对Statement关闭后是否需要关闭ResultSet进行说明。…

ResultSet(结果集)、Statement

ResultSet 基本介绍&#xff1a; 1.表示数据库结果集的数据表&#xff0c;通常通过执行查询数据库的语句生成 2.ResultSet对象保持一个光标指向其当前的数据行&#xff0c;光标最初在第一行之前 3.next()方法是将光标移动到下一行&#xff0c;并且由于在ResultSet对象中没有…

【JDBC】------ResultSet(结果集)和常见异常

分享第二十条励志语句 宁可自信&#xff0c;也不要盲目悲观。因为自信是一种力量&#xff0c;即使你的自信有些盲目&#xff0c;也无关大局&#xff0c;你可以在实践中调整心态&#xff0c;找到自己的恰当的位置。如果盲目自卑&#xff0c;你就必然失去一切。 目录 分享第二十…

JDBC的ResultSet

一、ResultSet[结果集] 1.表示数据库结果集的数据表&#xff0c;通常通过执行查询数据库的语句生成。 2.ResultSet对象保持一个光标指向其当前的数据行&#xff0c;最开始光标在第一行。 3.next方法将光标移动到下一行&#xff0c;由于在ResultSet对象中没有更多行时返回fal…

Java中,ResultSet 的用法

•在Java中&#xff0c;获得ResultSet的总行数的方法有以下几种。 第一种&#xff1a;利用ResultSet的getRow方法来获得ResultSet的总行数 Java代码 Statement stmt con.createStatement(ResultSet.TYPE_SCROLL_INSENSITIVE,ResultSet.CONCUR_UPDATABLE); ResultSet r…

ResultSet结果集的使用

一、ResultSet结果集的简介 查询数据库时&#xff0c;返回的是一个二维的结果集&#xff0c;我们需要用到ResultSet来遍历结果集&#xff0c;获取每一行的数据。 二、使用ResultSet遍历查询结果 boolean next() 将光标从当前位置向前移一行。 String  getString(int column…

ResultSet用法集锦

ResultSet用法集锦 添加链接描述 结果集(ResultSet)是数据中查询结果返回的一种对象&#xff0c;可以说结果集是一个存储查询结果的对象&#xff0c;但是结果集并不仅仅具有存储的功能&#xff0c;他同时还具有操纵数据的功能&#xff0c;可能完成对数据的更新等.结果集读取数据…

ResultSet的介绍与使用

相关文章 jdbc连接以及出现的异常处理Jdbc工具类ResultSet的介绍与使用jdbc中的sql注入PreparedStatement的介绍与解决sql注入jdbc管理事务 ResultSet ResultSet是我们使用jdbc连接时&#xff0c;查询的一个返回结果集,ResultSet resultSet stmt.executeQuery(sql),下面就使用…

JDBC之ResultSet接口

ResultSet介绍 对数据库的查询操作&#xff0c;一般需要返回查询结果。在程序中&#xff0c;JDBC为我们提供了ResultSet接口来专门处理查询结果集。 由于ResultSet是JDBC的一个接口&#xff0c;先来简单了解一下JDBC&#xff1a; JDBC&#xff1a;Java Database Connection&am…

android studio BindService

一.输入&#xff1a; 1.全部代码&#xff1a; 主界面代码&#xff1a; public class BindServiceActivity extends AppCompatActivity implements View.OnClickListener {private static final String TAG "BindServiceActivity";private Button mBtBindService;…

Service中bindService

最近有用到Activity需要不断的从Service中获取数据&#xff0c;第一个想法肯定就是通过bind回调机制了&#xff0c;有几点概念模糊特此记录下&#xff1a; 单独使用bindService()&#xff0c;unbindService()会经历&#xff1a;->onCreate()->onBind()->Service runn…

bindService不调用onServiceConnected的问题

bindService不调用onServiceConnected的问题 昨天做一个项目时&#xff0c;Activity需要bindService获取到service实例与service进行交互&#xff0c;创建好service如下&#xff08;例&#xff09;&#xff1a; class MyService{private IBinder mBinder new MyServicerBind…

bind服务

一、域名 www.baidu.com www.baidu.com. &#xff08;.&#xff09;根域 &#xff08;com&#xff09;一级域名 &#xff08;baidu&#xff09;二级域名 二、DNS解析记录分类 A记录、CNAME、MX记录、NS记录 1、A记录 通过域名直接查询到IP 例如&…

bindService启动流程

通过bindService启动的Service,会执行Service的onCreate、onBind、onUnbind、onDestroy方法&#xff0c;可以通过onBind方法返回的Binder对象和调用端进行通信&#xff0c;并且Service的生命周期和调用端同步。 如下是启动bindService的代码 var stu: Student? null val con…