【踩坑经历】Java Long 类型传给前端损失精度的问题

2023-11-15

最近在做一个 SpringBoot + Vue 的项目,持久层框架用的是 MyBatis-Plus,然后遇到了一个问题,一起来看下怎么回事。

这个项目就是一个文章收藏器,可以收藏一些技术文章,然后可以选择星标,以便查找这篇文章。
在这里插入图片描述
那么点击星标的按钮,实际上就是调了后端一个接口,更新了数据库中相应字段。每一个列表的字段如下:
在这里插入图片描述
可以看到,标记星标的字段就是 isFavorite ,没有星标的时候是 0 ,星标之后变成 1 。同时这边还有个 id 字段,这个字段是由 MyBatis-Plus 插入数据库时自动填充的,是一个 Long 类型的 uuid 。在调接口的时候,后台根据传过来的 id 进行条件更新。

但是在测试的时候,调接口发现 isFavorite 一直更新失败,但是SQL 语句正常执行,没有报错,看一下打印的 SQL 语句:
在这里插入图片描述
然后 Controller 层的逻辑是这样写的:

@PutMapping
public ResponseEntity<ServerResponse> changeFavoriteStatus(
        @NotBlank(message = "文章 ID 不能为空") @RequestParam("articleId") String articleId,
        @NotBlank(message = "星标状态不能为空") @RequestParam("isFavorite") String isFavorite
) {
	LambdaUpdateWrapper<ArticleModel> updateWrapper = new LambdaUpdateWrapper<>();
    updateWrapper
            .eq(ArticleModel::getId, Long.valueOf(articleId))
            .set(ArticleModel::getIsFavorite, Integer.valueOf(isFavorite));
	articleMapper.update(null, updateWrapper);
    return ServerResponse.ok(null);
}

这就有点头疼了,关键没有报错,无从下手啊。假如是 MyBatis-Plus 提供的 API 有问题,应该会有报错信息。然后网上查了很多资料,看别人的代码都是跟我一样写的,但是也没看他们遇到这样的问题。

后来尝试硬编码,直接在代码里面填写参数,发现可以更新成功:

@PutMapping
public ResponseEntity<ServerResponse> changeFavoriteStatus(
        @NotBlank(message = "文章 ID 不能为空") @RequestParam("articleId") String articleId,
        @NotBlank(message = "星标状态不能为空") @RequestParam("isFavorite") String isFavorite
) {
	LambdaUpdateWrapper<ArticleModel> updateWrapper = new LambdaUpdateWrapper<>();
    updateWrapper
            .eq(ArticleModel::getId, 1404788380714967042L)
            .set(ArticleModel::getIsFavorite, 1);
	articleMapper.update(null, updateWrapper);
    return ServerResponse.ok(null);
}

这样一下就有了思路,因为我这边的 id 是直接数据库复制过来的,那么基本可以确定是前端传过来的 id 有问题。看一下查询接口打印的 SQL 语句:
在这里插入图片描述
看到第一条记录的 id 字段是 1404788380714967042 。然后再来看一下传给前端的字段,发现确实不一样:
在这里插入图片描述
综上,本人再查询了相关资料,Java 中 long 数据类型是 64 位,最大值是 9,223,372,036,854,775,807 ,也就是 2^63 - 1 ,然后 JavaScript 中的 number 类型,最大的安全整数 2^53 - 1

也就是说,Java 中的 Long 类型能表示的范围比 JS 的 number 类型大,如果后端在序列化的时候直接用 Long 类型,前端在反序列化的时候,number 会损失精度,导致 id 字段不是实际的 id ,进而查询不到记录,无法更新。因此对于 Long 类型,后端需要转为字符串才能传给前端。

转为字符串的话,常规的做法是定义一个 DTO 类,将 Long 类型的字段都定义成 String ,那么这样的做法比较麻烦,而且如果接口有很多,需要大量的 DTO 。另一种做法是使用注解,在序列化的时候保留精度。

SpringBoot 默认使用的 JSON 解析框架是 jackson,可以直接使用。如果要用第三方框架,例如 fastjson 则需要进行配置。以 jackson 为例,对于在序列化时需要保留精度的字段,添加 @JsonSerialize 注解即可:

@Data
@TableName("tb_article")
public class ArticleModel {
    @JsonSerialize(using = ToStringSerializer.class)
    private Long id;

    private Integer isFavorite;
}

在添加注解之后,前端拿到的参数如下,可以看到 Long 类型 id 变成字符串了:
在这里插入图片描述
另外这边还有一个时间戳,没有转为字符串,本人认为没有必要,因为 JS 里面本身也有时间戳哈,而且这个范围没有超出 number 的最大安全整数,可以直接用。另外 jackson 还提供了注解可以在序列化时自定义格式化:

@Data
//序列化、反序列化忽略的属性,多个时用“,”隔开
@JsonIgnoreProperties({"captcha"})
//当属性的值为空(null或者"")时,不进行序列化,可以减少数据传输
@JsonInclude(JsonInclude.Include.NON_EMPTY)
public class UserVoByJson {

    // 序列化、反序列化时,属性的名称
    @JsonProperty("userName")
    private String username;

    // 为反序列化期间要接受的属性定义一个或多个替代名称,可以与@JsonProperty一起使用
    @JsonAlias({"pass_word", "passWord"})
    @JsonProperty("pwd")
    private String password;

    //序列化、反序列化时,格式化时间
    @JsonFormat(locale = "zh", timezone = "GMT+8", pattern = "yyyy-MM-dd HH:mm:ss")
    private Date createDate;

    //序列化、反序列化忽略属性
    @JsonIgnore
    private String captcha;

}

参考:
java long类型报错:error: integer number too large
java的long类型传输到前端损失精度
Springboot引入FastJson
修复Long类型太长,而Java序列化JSON丢失精度问题的方法
Number.MAX_SAFE_INTEGER - MDN
SpringBoot系列——Jackson序列化

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

【踩坑经历】Java Long 类型传给前端损失精度的问题 的相关文章

  • AES 加密 Java/plsql

    我需要在Java和plsql DBMS CRYPTO for Oracle 10g 上实现相同的加密 解密应用程序 两种实现都工作正常 但这里的问题是我对相同纯文本的加密得到了不同的输出 下面是用于加密 解密过程的代码 Java 和 PLS
  • Java程序中的数组奇怪的行为[重复]

    这个问题在这里已经有答案了 我遇到了这个 Java 程序及其以意想不到的方式运行 以下程序计算 int 数组中元素对之间的差异 import java util public class SetTest public static void
  • 如何查找 Android 设备中的所有文件并将它们放入列表中?

    我正在寻求帮助来列出 Android 外部存储设备中的所有文件 我想查找所有文件夹 包括主文件夹的子文件夹 有办法吗 我已经做了一个基本的工作 但我仍然没有得到想要的结果 这不起作用 这是我的代码 File files array file
  • 如何在 Java 中禁用 System.out 以提高速度

    我正在用 Java 编写一个模拟重力的程序 其中有一堆日志语句 到 System out 我的程序运行速度非常慢 我认为日志记录可能是部分原因 有什么方法可以禁用 System out 以便我的程序在打印时不会变慢 或者我是否必须手动检查并
  • HDFS:使用 Java / Scala API 移动多个文件

    我需要使用 Java Scala 程序移动 HDFS 中对应于给定正则表达式的多个文件 例如 我必须移动所有名称为 xml从文件夹a到文件夹b 使用 shell 命令我可以使用以下命令 bin hdfs dfs mv a xml b 我可以
  • Microsoft Graph 身份验证 - 委派权限

    我可以使用 Microsoft Graph 访问资源无需用户即可访问 https developer microsoft com en us graph docs concepts auth v2 service 但是 此方法不允许我访问需
  • Prim 的迷宫生成算法:获取相邻单元格

    我基于 Prim 算法编写了一个迷宫生成器程序 该算法是 Prim 算法的随机版本 从充满墙壁的网格开始 选择一个单元格 将其标记为迷宫的一部分 将单元格的墙壁添加到墙壁列表中 While there are walls in the li
  • 无法理解 Java 地图条目集

    我正在看一个 java 刽子手游戏 https github com leleah EvilHangman blob master EvilHangman java https github com leleah EvilHangman b
  • 从 android 简单上传到 S3

    我在网上搜索了从 android 上传简单文件到 s3 的方法 但找不到任何有效的方法 我认为这是因为缺乏具体步骤 1 https mobile awsblog com post Tx1V588RKX5XPQB TransferManage
  • 制作java包

    我的 Java 类组织变得有点混乱 所以我要回顾一下我在 Java 学习中跳过的东西 类路径 我无法安静地将心爱的类编译到我为它们创建的包中 这是我的文件夹层次结构 com david Greet java greeter SayHello
  • 将 Long 转换为 DateTime 从 C# 日期到 Java 日期

    我一直尝试用Java读取二进制文件 而二进制文件是用C 编写的 其中一些数据包含日期时间数据 当 DateTime 数据写入文件 以二进制形式 时 它使用DateTime ToBinary on C 为了读取 DateTime 数据 它将首
  • org.jdesktop.application 包不存在

    几天以来我一直在构建一个 Java 桌面应用程序 一切都很顺利 但是今天 当我打开Netbeans并编译文件时 出现以下编译错误 Compiling 9 source files to C Documents and Settings Ad
  • Java中未绑定通配符泛型的用途和要点是什么?

    我不明白未绑定通配符泛型有什么用 具有上限的绑定通配符泛型 stuff for Object item stuff System out println item Since PrintStream println 可以处理所有引用类型 通
  • Tomcat 6找不到mysql驱动

    这里有一个类似的问题 但关于类路径 ClassNotFoundException com mysql jdbc Driver https stackoverflow com questions 1585811 classnotfoundex
  • 如何在 Maven 中显示消息

    如何在 Maven 中显示消息 在ant中 我们确实有 echo 来显示消息 但是在maven中 我该怎么做呢 您可以使用 antrun 插件
  • Windows 上的 Nifi 命令

    在我当前的项目中 我一直在Windows操作系统上使用apache nifi 我已经提取了nifi 0 7 0 bin zip文件输入C 现在 当我跑步时 bin run nifi bat as 管理员我在命令行上看到以下消息 但无法运行
  • 如何配置eclipse以保持这种代码格式?

    以下代码来自 playframework 2 0 的示例 Display the dashboard public static Result index return ok dashboard render Project findInv
  • 如何测试 spring-security-oauth2 资源服务器安全性?

    随着 Spring Security 4 的发布改进了对测试的支持 http docs spring io spring security site docs 4 0 x reference htmlsingle test我想更新我当前的
  • android Accessibility-service 突然停止触发事件

    我有一个 AccessibilityService 工作正常 但由于开发过程中的某些原因它停止工作 我似乎找不到这个原因 请看一下我的代码并告诉我为什么它不起作用 public class MyServicee extends Access
  • 将2-3-4树转换为红黑树

    我正在尝试将 2 3 4 树转换为 java 中的红黑树 但我无法弄清楚它 我将这两个基本类编写如下 以使问题简单明了 但不知道从这里到哪里去 public class TwoThreeFour

随机推荐

  • C语言【二分查找】详解

    提示 文章写完后 目录可以自动生成 如何生成可参考右边的帮助文档 二分 文章目录 前言 一 二分查找 的步骤 二 二分查找 的注意事项 三 举例详解 二分查找 前言 一 二分查找 步骤 二 二分查找 的注意事项 三 举例详细讲解 二分查找
  • redis使用rightPushAll保存List,List的size为1

    redis使用rightPushAll保存List List的size为1 问题描述 问题定位 解决方案 问题疑惑 问题描述 使用RedisTemplate保存List 期望是直接将Java的List直接转换redis的list 但是rig
  • C++智能指针简介

    C 智能指针简介 一 简介 前言 从C到C C malloc 和 free C new 和 delete 从C 到Morden C C new 和 delete morden C shared ptr unique ptr 和 weak p
  • IDEA创建Zookeeper客户端

    IDEA创建Zookeeper客户端 解决单机Zookeeper的错误 不是Zookeeper集群问题 但是集群问题也可以参考 网上看到的教程基本上都是服务器端口号 防火墙的问题 但是个人测试之后发现没有解决问题 于是自己尝试一下方法 最后
  • 梯度下降法的推导(非常详细、易懂的推导)

    原作者 红色石头 来自 AI有道 梯度下降 算法的公式非常简单 沿着梯度的反方向 坡度最陡 是我们日常经验得到的 其本质的原因到底是什么呢 为什么局部下降最快的方向就是梯度的负方向呢 也许很多朋友还不太清楚 没关系 接下来我将以通俗的语言来
  • WSL2开启后Vmware虚拟机性能下降

    开启WSL2 需要开启 虚拟机平台 功能 实际就是HyperV 开启后 Vmware虚拟机里面性能下降大约10 再次断绝了我的WSL2直接挂在ext分区的念想 见图 开启前 CPU Z跑分464 开启后414 按照Vmware建议关闭侧信道
  • C++学习(六十六)IoT和AIoT

    loT是物联网的英文名称 是Internet of Things的缩写 AloT AI IoT 是人工智能 物联网的英文名称 智能物联网
  • 修复ROS2使用zsh无法用tab补全 ros2 指令的相关问题

    安装完ROS2 后 改用zsh发现无法使用tab补全 ros2相关指令 现记录下修复办法 首先 安装python3 argcomplete sudo apt install python3 argcomplete 然后 在 zshrc里面添
  • STM32 CAN通信的学习笔记总结(从小白开始)

    知识来源于互联网 回馈于互联网 目录 1 总体概述 1 1 基本概念 1 2 通讯方式 1 3 为什么使用CAN 1 4 CAN的协议及组成 2 上帝视角看CAN的通讯过程 2 1 数据传输原理实现 2 2 通信的整个过程 2 2 1 空闲
  • Python读写excel文件

    1 使用pandas库读取Excel 最常用 pandas可以读取各种各样格式的数据文件 一般输出dataframe格式 如 txt csv excel json 剪切板 数据库 html hdf parquet pickled文件 sas
  • FSCapture注册码

    企业版序列号 name bluman serial 序列号 注册码 VPISCJULXUFGDDXYAUYF 转载于 https www cnblogs com wshsdlau p 4396184 html
  • HTML<DIV>常用标签

    目录 1 什么是DIV 1 1 div是什么意思 1 2 div标签怎么用 1 3 div布局优势 1 4 DIV作用是什么 1 5 有哪些DIV方式 1 5 1行内样式 1 5 2内嵌样式 1 5 3外部样式 1 6 样式使用规则 2 D
  • 使用队列实现stack

    两个队列实现一个stack q1只保持一个元素即可 多余的转换到q2当中 出队列元素 有两种情况 q1不为空 直接出队列 如果连续出队列 q1可能为空 需要q2的部分元素放到q1当中去 说白了就是元素捣鼓来捣鼓去的问题即可 class My
  • Linux-安装redis6.2.1及主备复制模式(replication)

    Linux 安装redis6 2 1 下载redis6 2 1资源 上传至安装目录 解压及编译 解压 修改名称 编译 修改配置文件 主节点 从节点 启动及测试 启动 主节点 从节点 测试 下载redis6 2 1资源 地址 https re
  • 华为数通方向HCIP-DataCom H12-821题库(单选题:101-120)

    第101题 可用于多种路由协议 由 if match 和 apply 子句组成的路由选择工具是 A route policy B IP Prefix C commnityfilter D as path filter 答案 A 解析 Rou
  • QT对话框去掉帮助和关闭按钮 拦截QT关闭窗口的CloseEvent

    建了一个对话框 我不想把边框去掉 只想去掉关闭按钮 setWindowFlags windowFlags Qt WindowCloseButtonHint Qt WindowContextHelpButtonHint 结果那个问号的按钮去掉
  • c++序列化以及反序列化实现

    1 什么是序列化和反序列化 当我们在写程序时 比如说我们自定义了一个实体类Person 然后在程序中创建一个该实体类对象 并给对象赋了一些值 但是我们想将这些数据发给我们的其他的程序员朋友 让他们也可以调用我们创建的这个实体类并使用我们的数
  • 数据库实时同步利器——CDC(变化数据捕获技术)

    在进行数据ETL过程中 我们经常需要通过周期性的定时调度将业务数据按照T 1的方式同步到数据仓库中 进行数据分析处理 最终通过BI报表展示给最终用户 但这种方式实时性较差 用户往往只能看到昨天的数据 会影响用户决策的及时性 而如果用户要近实
  • 更换持续集成工具,从 Travis 到 Github Actions

    我真傻 真的 单单受文档的推荐就选择了 Travis 作为部分项目的持续集成工具 没有料到它早已于 2020 年 12 月更换了免费政策 不再为开源项目提供免费的用于持续集成使用的 Credits 了 当赠送的 10000 个点数用完 就需
  • 【踩坑经历】Java Long 类型传给前端损失精度的问题

    最近在做一个 SpringBoot Vue 的项目 持久层框架用的是 MyBatis Plus 然后遇到了一个问题 一起来看下怎么回事 这个项目就是一个文章收藏器 可以收藏一些技术文章 然后可以选择星标 以便查找这篇文章 那么点击星标的按钮