SpringBoot解决BigDecimal传到前端后精度丢失问题
作者:IT利刃出鞘 发布时间:2021-12-13 14:44:39
简介
本文用示例介绍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.123
const json = '{"id": 1, "name": "Tony", "amount": 123456712345678.12345}';
const obj = JSON.parse(json);
console.log(obj.amount); // 123456712345678.12
const json = '{"id": 1, "name": "Tony", "amount": 98765432198765.12345}';
const obj = JSON.parse(json);
console.log(obj.amount); // 98765432198765.12
const json = '{"id": 1, "name": "Tony", "amount": 987654321987654321.12345}';
const obj = JSON.parse(json);
console.log(obj.amount); // 987654321987654300
Java后端BigDecimal的范围
1.范围没有限制,可以认为无限大、无限小
2.可以通过如下代码验证:
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 {
@Bean
public 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);
}
@Override
public boolean isEmpty(SerializerProvider prov, Object value) {
if (value == null) {
return true;
}
String str = ((BigDecimal) value).stripTrailingZeros().toPlainString();
return str.isEmpty();
}
@Override
public 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());
}
@Override
public void serializeWithType(Object value, JsonGenerator gen,
SerializerProvider provider, TypeSerializer typeSer)
throws IOException {
// no type info, just regular serialization
serialize(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 {
@Bean
public 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
结果:
来源:https://knife.blog.csdn.net/article/details/125095018
猜你喜欢
- java 根据经纬度获取地址实现代码实现代码:public class GetLocation { public
- shiro是一个权限框架,具体的使用可以查看其官网 http://shiro.apache.org/ 它提供了很方便的权限认证和
- 本文实例讲述了Java排序算法总结之希尔排序。分享给大家供大家参考。具体分析如下:前言:希尔排序(Shell Sort)是插入排序的一种。是
- 目录1.概览2.自定义枚举方法3.使用 == 比较枚举类型4.在 switch 语句中使用枚举类型5.枚举类型的属性,方法和构造函数6.En
- 装箱和拆箱是值类型和引用类型之间相互转换是要执行的操作。 1. 装箱在值类型向引用类型转换时发生2. 拆箱在引用类型向值
- 在本文中,我们将介绍一个非常有用的Spring功能,该功能允许我们基于一个或多个Spring注释创建自己的注释。假设我们有一组经常一起使用的
- 以下共有4个函数分别是:1.从剪切板获得文字。2.将字符串复制到剪切板。3.从剪切板获得图片。4.复制图片到剪切板。/** * 从剪切板获得
- 本文核心为分层领域模型(VO , PO , BO, DAO ,POJO等)概念的个人理解。1.为什么出现分层领域模型这个东西?(1)解决MV
- 在软件开发过程中经常需要知道程序运行的大概时间,或者需要在规定时间内取数据,这是可以使用下面的方法获取时间段,还可以用在限时循环方法一:/*
- 下面给大家分享一小段代码给大家介绍C# 输出字符串到文本文件中,具体代码如下所示: public class WriteHelp
- 原理分析:迅雷的thunder://地址就是将普通url地址加前缀‘AA'、后缀‘ZZ',再base64编码后得到的字符串实
- 笔者最近需要上位机与下位机进行数据交互,在广泛参考大佬的资料后,较为完善地使用Textbox控件进行数据输入的功能。程序段主要功能:实现输入
- 简介本文用示例介绍java的Period的用法。Duration和Period说明Duration类通过秒和纳秒相结合来描述一个时间量,最高
- 项目里一直用的是 spring-security ,不得不说,spring-security 真是东西太多了,学习难度太大(可能我比较菜),
- Object 类位于 java.lang 包中,是所有 Java 类的祖先,Java 中的每个类都由它扩展而来。定义Java类时如果没有显示
- Redis 简介Redis 是完全开源的,遵守 BSD 协议,是一个高性能的 key-value 数据库。Redis 与其他 key - v
- 在style中如下面那样定义:<style name="mystyle"> <item name=&
- 本文实例讲述了c#用for语句输出一个三角形的方法。分享给大家供大家参考。具体分析如下:这是一道面试题,要求是这样的:只使用一个for循环输
- @Validated和BindingResult 使用遇到的坑@Validated 与BindingResult 需要相邻,否则 变量res
- 官网文档背景项目A中需要多数据源的实现,比如UserDao.getAllUserList() 需要从readonly库中读取,但是UserD