系列一、SpringBoot统一格式化日期 & 将long型数字转换成字符串

当数据库中的bigint类型数据通过接口传递给Vue前端时,出现了精度丢失。解决方案是通过在SpringBoot后端配置Jackson转换器,将Long类型序列化为String,以及对日期进行格式化处理,确保数据正确传输。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一、现象描述

        数据库中的字段保存的是bigint类型的数据,前端vue通过接口获取数据时,该字段的值却丢失精度了,如下图所示:

二、解决方案

2.1、导入依赖

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-json</artifactId>
</dependency>

2.2、Constants.java

public class Constants {
    public static String DEFAULT_DATE_TIME_FORMAT = "yyyy-MM-dd HH:mm:ss";
    public static String DEFAULT_DATE_FORMAT = "yyyy-MM-dd";
    public static String DEFAULT_TIME_FORMAT = "HH:mm:ss";
}

2.3、配置转换器

package org.star.config;

import com.fasterxml.jackson.databind.DeserializationFeature;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.SerializationFeature;
import com.fasterxml.jackson.databind.module.SimpleModule;
import com.fasterxml.jackson.databind.ser.std.ToStringSerializer;
import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule;
import com.fasterxml.jackson.datatype.jsr310.deser.LocalDateDeserializer;
import com.fasterxml.jackson.datatype.jsr310.deser.LocalDateTimeDeserializer;
import com.fasterxml.jackson.datatype.jsr310.deser.LocalTimeDeserializer;
import com.fasterxml.jackson.datatype.jsr310.ser.LocalDateSerializer;
import com.fasterxml.jackson.datatype.jsr310.ser.LocalDateTimeSerializer;
import com.fasterxml.jackson.datatype.jsr310.ser.LocalTimeSerializer;
import com.fasterxml.jackson.module.paramnames.ParameterNamesModule;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.converter.HttpMessageConverter;
import org.springframework.http.converter.json.MappingJackson2HttpMessageConverter;
import org.springframework.web.servlet.config.annotation.EnableWebMvc;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;
import org.star.constant.Constants;

import java.time.LocalDate;
import java.time.LocalDateTime;
import java.time.LocalTime;
import java.time.format.DateTimeFormatter;
import java.util.List;


@Configuration
@EnableWebMvc
public class MyWebMvcConfigurer implements WebMvcConfigurer {

    @Override
    public void configureMessageConverters(List<HttpMessageConverter<?>> converters) {
        MappingJackson2HttpMessageConverter jackson2HttpMessageConverter =new MappingJackson2HttpMessageConverter();

        /**
         * 序列换成json时,将所有的long变成String,因为js中得数字类型不能包含所有的java long值
         */
        ObjectMapper objectMapper =objectMapper();
        SimpleModule simpleModule =new SimpleModule();
        simpleModule.addSerializer(Long.class, ToStringSerializer.instance);
        simpleModule.addSerializer(Long.TYPE, ToStringSerializer.instance);
        objectMapper.registerModule(simpleModule);
        jackson2HttpMessageConverter.setObjectMapper(objectMapper);
        converters.add(jackson2HttpMessageConverter);

    }

    /**
     * 格式化日期
     * @return
     */
    @Bean
    public ObjectMapper objectMapper() {
        ObjectMapper objectMapper = new ObjectMapper();
        objectMapper.disable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS);
        objectMapper.disable(DeserializationFeature.ADJUST_DATES_TO_CONTEXT_TIME_ZONE);
        JavaTimeModule javaTimeModule = new JavaTimeModule();
        javaTimeModule.addSerializer(LocalDateTime.class,new LocalDateTimeSerializer(DateTimeFormatter.ofPattern(Constants.DEFAULT_DATE_TIME_FORMAT)));
        javaTimeModule.addSerializer(LocalDate.class,new LocalDateSerializer(DateTimeFormatter.ofPattern(Constants.DEFAULT_DATE_FORMAT)));
        javaTimeModule.addSerializer(LocalTime.class,new LocalTimeSerializer(DateTimeFormatter.ofPattern(Constants.DEFAULT_TIME_FORMAT)));
        javaTimeModule.addDeserializer(LocalDateTime.class,new LocalDateTimeDeserializer(DateTimeFormatter.ofPattern(Constants.DEFAULT_DATE_TIME_FORMAT)));
        javaTimeModule.addDeserializer(LocalDate.class,new LocalDateDeserializer(DateTimeFormatter.ofPattern(Constants.DEFAULT_DATE_FORMAT)));
        javaTimeModule.addDeserializer(LocalTime.class,new LocalTimeDeserializer(DateTimeFormatter.ofPattern(Constants.DEFAULT_TIME_FORMAT)));
        objectMapper.registerModule(javaTimeModule).registerModule(new ParameterNamesModule());
        return objectMapper;
    }

}

2.4、效果

2.5、参考

https://www.bilibili.com/read/cv22828897/
https://www.jianshu.com/p/cf50c4bc71d3
https://www.cnblogs.com/exmyth/p/14305299.html(靠谱[推荐])
http://www.shouxicto.com/article/2858.html(靠谱[不推荐])

### 实现全局去除字符串前后空格 在Spring Boot应用程序中,可以通过自定义注解和`ConstraintValidator`接口来实现对字符串参数自动去除前后空格的功能。这不仅提高了代码的致性和可维护性,还减少了开发者手动处理空白字符的工作量。 #### 定义自定义注解 `Trimmed` 创建个新的Java类用于表示该注解: ```java import javax.validation.Constraint; import java.lang.annotation.ElementType; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target; @Target({ ElementType.FIELD, ElementType.PARAMETER }) @Retention(RetentionPolicy.RUNTIME) @Constraint(validatedBy = TrimmedValidator.class) public @interface Trimmed { String message() default &quot;Invalid string format&quot;; Class&lt;?&gt;[] groups() default {}; Class&lt;? extends Payload&gt;[] payload() default {}; } ``` 此部分定义了个名为`Trimmed`的约束条件,可以应用于字段或方法参数上[^1]。 #### 创建验证器逻辑 接着编写具体的验证逻辑,在这里主要是调用String对象自带的trim()函数完成去空操作并返回原值: ```java import javax.validation.ConstraintValidator; import javax.validation.ConstraintValidatorContext; public class TrimmedValidator implements ConstraintValidator&lt;Trimmed, String&gt; { @Override public void initialize(Trimmed constraintAnnotation) {} @Override public boolean isValid(String value, ConstraintValidatorContext context) { if (value == null || &quot;&quot;.equals(value.trim())) { return true; // Allow empty strings or only whitespace. } ((ConstraintValidatorContextImpl)context).addMessageParameter(&quot;trimmedValue&quot;, value.trim()); return true; } } ``` 注意这里的特殊之处在于当输入为空串或是全由空白组成时不作任何修改直接放行;而对于其他情况,则会利用反射机制设置经过修剪后的字符串作为额外的消息参数传递给后续流程使用。 #### 修改控制器中的DTO实体类 假设有个简单的数据传输对象(DTO),其中包含需要被自动清理空白符的目标属性: ```java public class UserRequestDto { private Long id; @NotNull(message = &quot;Name cannot be null&quot;) @Size(min=2, max=30, message=&quot;Name must between 2 and 30 characters&quot;) @Trimmed private String name; // Getters and Setters... } ``` 此时只要请求体内的name字段带有前导或尾随空间,它们都会被自动移除后再参与进步业务逻辑判断。 #### 自定义全局异常处理器 最后步是为了能够友好地展示错误信息,建议引入统一的异常捕获机制。这样即使某些地方忘记加上必要的校验也不会让整个服务崩溃掉而是给出合理的提示: ```java @ControllerAdvice @ResponseBody class GlobalExceptionHandler extends ResponseEntityExceptionHandler { @ExceptionHandler(MethodArgumentNotValidException.class) protected ResponseEntity&lt;Object&gt; handleMethodArgumentNotValid( MethodArgumentNotValidException ex) { List&lt;String&gt; errors = new ArrayList&lt;&gt;(); for (FieldError error : ex.getBindingResult().getFieldErrors()) { errors.add(error.getField() + &quot;: &quot; + error.getDefaultMessage() .replace(&quot;{trimmedValue}&quot;, &quot;\&quot;&quot; + error.getRejectedValue() + &quot;\&quot;&quot;)); } ApiError apiError = new ApiError(HttpStatus.BAD_REQUEST, &quot;Validation Error&quot;, errors); return new ResponseEntity&lt;&gt;(apiError, HttpStatus.BAD_REQUEST); } } ``` 上述代码片段展示了如何捕捉到所有因违反Bean Validation规则而抛出的方法参数不合法异常,并将其转换成结构化的JSON响应格式返回给客户端查看。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值