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

结果

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

问题复现

场景描述

实际项目中前端会这样处理:调用后端接口获得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

结果:

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

法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

结果:

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

方案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

结果:

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

法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

结果:

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

来源:https://knife.blog.csdn.net/article/details/125095018

标签:SpringBoot,BigDecimal,精度丢失
0
投稿

猜你喜欢

  • 实现Android键盘的中英文适配

    2022-03-27 20:40:33
  • Hibernate持久化对象生命周期原理解析

    2022-08-14 19:44:27
  • java参数传递之值传递和引用传递

    2021-09-26 10:28:46
  • 详解如何在SpringBoot中自定义参数解析器

    2023-07-24 16:06:51
  • 全面详解Spring Bean生命周期教程示例

    2023-08-09 11:41:15
  • MyBatis超详细讲解如何实现分页功能

    2023-08-22 23:06:51
  • Java如何读取jar包中的resource资源文件

    2022-10-12 22:51:43
  • 详解SpringMVC常用注解功能及属性

    2021-12-29 02:49:23
  • java实现获取安卓设备里已安装的软件包

    2022-10-21 00:53:04
  • Java+Swing实现五子棋游戏的示例代码

    2022-06-27 11:56:22
  • Java 图片与byte数组互相转换实例

    2023-06-24 03:28:39
  • C#实现调用本机摄像头实例

    2022-07-01 19:54:49
  • Android CountDownTimer案例总结

    2023-12-21 05:40:36
  • Android异步方法以同步方式实现

    2022-11-05 15:13:50
  • Android实现蓝牙聊天功能

    2023-06-10 01:19:48
  • hadoop实现grep示例分享

    2023-12-24 05:31:04
  • java多线程之铁路售票系统

    2022-01-31 12:25:14
  • ANDROID 完美退出APP的实例代码

    2023-04-27 13:03:06
  • 图文详解SpringBoot中Log日志的集成

    2023-05-27 09:55:13
  • android获取屏幕宽高与获取控件宽高(三种方法)

    2022-07-11 11:25:42
  • asp之家 软件编程 m.aspxhome.com