Heim >Java >javaLernprogramm >So lösen Sie das Problem des Präzisionsverlusts, nachdem die Primärschlüssel-ID des SpringBoot-Snowflake-Algorithmus an das Frontend übertragen wurde
Der Bereich des Java-Backend-Long-Typs
-2^63~2^63, das heißt: -9223372036854775808~9223372036854775807, also 19 Bit.
Diese Zahl kann über die Methoden erhalten werden: Long.MAX_VALUE, Long_MIN_VALUE.
Der Bereich der numerischen Typen von Front-End-JS
-2^53~2^53, das heißt: -9007199254740991~9007199254740991, also 16-Bit.
Diese Zahl kann über folgende Methoden ermittelt werden: Number.MAX_SAFE_INTEGER, Number.MIN_SAFE_INTEGER.
Fazit
Es ist ersichtlich, dass die lange Breite des Java-Backends größer ist als die des Frontends. Der Snowflake-Algorithmus generiert im Allgemeinen Zahlen mit einer Breite von 18 oder 19 Bit, sodass zu diesem Zeitpunkt Probleme auftreten.
1. Tabellenstruktur
Der Primärschlüsseltyp ist BIGINT, der die vom Snowflake-Algorithmus generierte ID speichert.
CREATE TABLE `user` ( `id` BIGINT(32) NOT NULL COMMENT '用户id', ... PRIMARY KEY (`id`) USING BTREE ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='用户表';
2.Entity
Verwenden Sie den Long-Typ, um dem BIGINT-Typ der Datenbank-ID zu entsprechen.
Der Snowflake-Algorithmus von MybatisPlus wird hier verwendet, um automatisch eine 19-stellige reine Zahl als Primärschlüssel-ID zu generieren. (Natürlich können Sie die ID auch manuell mit dem Snowflake-Algorithmus generieren.)
import lombok.Data; @Data public class User { @TableId(type = IdType.ASSIGN_ID) private Long id; //其他成员 }
3. Reagieren Sie auf das Frontend.
Reagieren Sie auf das Frontend wie gewohnt mit JSON-Daten
{ "id": 1352166380631257089, ... }
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; @RestController @RequestMapping("user") public class UserController { @GetMapping("find") public UserVO find(Long id) { UserVO userVO = new UserVO(); userVO.setId(id); userVO.setUsername("Tony"); return userVO; } }
Test
Visit: http://localhost:8080/user/find?id=1352213368413982722
Result
Das Problem wird reproduziertWie Sie oben sehen können, gibt es ist kein Problem. Warum gibt es kein Problem?
Das Front-End wird an das Back-End übergeben: SpingMVC konvertiert die String-Typ-ID automatisch in den Long-Typ, und es gibt keine Probleme. Das Back-End antwortet dem Front-End: Es ist im JSON-Format und hat nichts mit JS zu tun, und es wird keine Probleme geben
Wann könnte was schief gehen?
Nachdem das Frontend den JSON empfangen hat, serialisiert es ihn in ein JS-Objekt und führt dann andere Vorgänge aus. Beim Konvertieren von JSON in JS-Objekte treten folgende Probleme auf:
Sie können sehen, dass die ursprüngliche ID 1352213368413982722 lautet und nach der Serialisierung in ein JS-Objekt zu 1352213368413982700 wird. Der Code lautet:
package com.knife.entity; import lombok.Data; @Data public class UserVO { private Long id; private String username; }Lösung
Es gibt die folgenden zwei Lösungen: Ändern Sie das ID-Feld des Datenbanktabellenentwurfs vom Typ „Lang“ in den Typ „String“.
Das Front-End verwendet den String-Typ, um die ID zu speichern und die Genauigkeit zu gewährleisten, und das Back-End und die Datenbank verwenden weiterhin den Long-Typ (BigINT).
Option 1 verwendet den String-Typ als Datenbank-ID , und die Abfrageleistung wird erheblich sinken. Daher sollte Option 2 übernommen werden. In diesem Artikel wird Option 2 vorgestellt.
ObjectMapper anpassen.
const json = '{"id": 1352213368413982722, "name": "Tony"}'; const obj = JSON.parse(json); console.log(obj.id); console.log(obj.name);
Test
Option 2: Benutzerdefinierter Serializer (nicht empfohlen)
Serialisierer
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; @Configuration public class JacksonConfig { @Bean public ObjectMapper jacksonObjectMapper(Jackson2ObjectMapperBuilder builder) { ObjectMapper objectMapper = builder.createXmlMapper(false).build(); // 全局配置序列化返回 JSON 处理 SimpleModule simpleModule = new SimpleModule(); // 将使用String来序列化Long类型 simpleModule.addSerializer(Long.class, ToStringSerializer.instance); simpleModule.addSerializer(Long.TYPE, ToStringSerializer.instance); objectMapper.registerModule(simpleModule); return objectMapper; } }ObjectMapper-Konfiguration
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.ser.std.NumberSerializer; import java.io.IOException; /** * 超出 JS 最大最小值 处理 */ @JacksonStdImpl public class BigNumberSerializer extends NumberSerializer { /** * 根据 JS Number.MAX_SAFE_INTEGER 与 Number.MIN_SAFE_INTEGER 得来 */ private static final long MAX_SAFE_INTEGER = 9007199254740991L; private static final long MIN_SAFE_INTEGER = -9007199254740991L; /** * 提供实例 */ public static final BigNumberSerializer instance = new BigNumberSerializer(Number.class); public BigNumberSerializer(Class<? extends Number> rawType) { super(rawType); } @Override public void serialize(Number value, JsonGenerator gen, SerializerProvider provider) throws IOException { // 超出范围 序列化位字符串 if (value.longValue() > MIN_SAFE_INTEGER && value.longValue() < MAX_SAFE_INTEGER) { super.serialize(value, gen, provider); } else { gen.writeString(value.toString()); } } }Test Besuch: http://localhost:8080/user/find?id=1352213368413982722 Methode 2: Lokale Verarbeitung
Anweisungen
at Hinzufügen: @JsonSerialize( using= ToStringSerializer.class) in das Feld ein.Instanz
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; @Configuration public class JacksonConfig { @Bean public ObjectMapper jacksonObjectMapper(Jackson2ObjectMapperBuilder builder) { ObjectMapper objectMapper = builder.createXmlMapper(false).build(); // 全局配置序列化返回 JSON 处理 SimpleModule simpleModule = new SimpleModule(); // 将使用自定义序列化器来序列化Long类型 simpleModule.addSerializer(Long.class, BigNumberSerializer.instance); simpleModule.addSerializer(Long.TYPE, BigNumberSerializer.instance); objectMapper.registerModule(simpleModule); return objectMapper; } }
Besuch: http://localhost:8080/user/find?id=1352213368413982722
Das obige ist der detaillierte Inhalt vonSo lösen Sie das Problem des Präzisionsverlusts, nachdem die Primärschlüssel-ID des SpringBoot-Snowflake-Algorithmus an das Frontend übertragen wurde. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!