Spring Boot 当Sping Boot (3.2)和Hibernate(6.4.0.Final)升级时,postgres中存储的大小增加

fsi0uk1n  于 2024-01-06  发布在  Spring
关注(0)|答案(1)|浏览(276)

小故事

获取错误java.lang.ClassCastException: class java.time.LocalDateTime cannot be cast to class [Ljava.lang.Object; (java.time.LocalDateTime and [Ljava.lang.Object; are in module java.base of loader 'bootstrap')

说来话长

我正在将Sping Boot 从2.7.x升级到3.2。我已经通过在线搜索进行了升级。但是我对下面的实体字段之一进行了更改,因为我的应用程序没有启动。

  1. // old
  2. @Column(name = "last_modified")
  3. @Convert(converter = LocalDateTimeConverter.class)
  4. private LocalDateTime lastModified;
  5. // new
  6. @Column(name = "last_modified")
  7. @JdbcType(BinaryJdbcType.class)
  8. @Convert(converter = LocalDateTimeConverter.class)
  9. private LocalDateTime lastModified;

字符串
以前存储在Postgres中的数据的last_modified列大小为51字节,但在此更改后,新行的大小为430字节。
使用下面的转换器。这里的大小是51字节之前,放入数据库和也走出数据库。但旧的数据是造成问题。

  1. @Override
  2. public Byte[] convertToDatabaseColumn(final LocalDateTime localDateTime) {
  3. log.info("Inside convertToDatabaseColumn");
  4. if (localDateTime == null) {
  5. return null;
  6. }
  7. try (final ByteArrayOutputStream buffer = new ByteArrayOutputStream();
  8. final ObjectOutputStream oos = new ObjectOutputStream(buffer)) {
  9. oos.writeObject(localDateTime);
  10. log.info("buffer size: {}", buffer.size());
  11. return ArrayUtils.toObject(buffer.toByteArray());
  12. } catch (Exception e) {
  13. lalalalala
  14. }
  15. }
  16. @Override
  17. public LocalDateTime convertToEntityAttribute(final Byte[] bytes) {
  18. // this line not executed in case of old data & works fine wit old data & gives 51 bytes
  19. log.info("Inside convertToEntityAttribute: {}", Array.getLength(bytes));
  20. if (bytes == null) {
  21. return null;
  22. }
  23. try (final ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(ArrayUtils.toPrimitive(bytes));
  24. final ObjectInputStream objectInputStream = new ObjectInputStream(byteArrayInputStream)) {
  25. final LocalDateTime dateTime = LocalDateTime.parse(objectInputStream.readObject().toString());
  26. return dateTime;
  27. } catch (Exception e) {
  28. lalalal
  29. }
  30. }

其他尝试的解决方案

第一次

如果我不添加这个JdbcType注解,它将无法启动并给出错误Failed to initialize JPA EntityManagerFactory: [PersistenceUnit: default] Unable to build Hibernate SessionFactory; nested exception is org.hibernate.MappingException: Unable to determine SQL type name for column 'last_modified' of table 'resource': class org.hibernate.type.internal.CustomMutabilityConvertedBasicTypeImpl cannot be cast to class org.hibernate.type.BasicPluralType (org.hibernate.type.internal.CustomMutabilityConvertedBasicTypeImpl and org.hibernate.type.BasicPluralType are in unnamed module of loader org.springframework.boot.loader.launch.LaunchedClassLoader @37a71e93)"

2

我也在一些地方找到了下面的答案,但它在运行时会出错。

  1. @Column(name = "last_modified", columnDefinition = "bytea")
  2. @Convert(converter = LocalDateTimeConverter.class)
  3. private LocalDateTime lastModified;
  4. // Error
  5. // org.hibernate.exception.SQLGrammarException: could not execute statement [ERROR: column \"last_modified\" is of type bytea but expression is of type smallint[]"}
  6. // Hint: You will need to rewrite or cast the expression."}

版本号

'org.springframework. Boot :spring- Boot :3.2.0' 'org.postgresql:postgresql:42.5.4' 'org.hibernate. boot:hibernate-core:6.4.0.Final' Java 17

摘要

  • 能够推送新数据并从数据库中获取新数据,但数据库中的大小真的很大
  • 由于新旧数据大小不同,导致错误. java.lang.ClassCastException: class java.time.LocalDateTime cannot be cast to class [Ljava.lang.Object; (java.time.LocalDateTime and [Ljava.lang.Object; are in module java.base of loader 'bootstrap')
a6b3iqyw

a6b3iqyw1#

发现修复。所以问题是Byte数组序列化时不转换为byte数组。这是奇怪的,不知道为什么。
首先在Entity中,与解决方案2相同。

  1. @Column(name = "last_modified", columnDefinition = "bytea")
  2. @Convert(converter = LocalDateTimeConverter.class)
  3. private LocalDateTime lastModified;

字符串
然后将转换器更改为使用byte数组而不是Byte数组。

  1. @Override
  2. public byte[] convertToDatabaseColumn(final LocalDateTime localDateTime) {
  3. log.info("Inside convertToDatabaseColumn");
  4. if (localDateTime == null) {
  5. return null;
  6. }
  7. try (final ByteArrayOutputStream buffer = new ByteArrayOutputStream();
  8. final ObjectOutputStream oos = new ObjectOutputStream(buffer)) {
  9. oos.writeObject(localDateTime);
  10. log.info("buffer size: {}", buffer.size());
  11. return buffer.toByteArray();
  12. } catch (Exception e) {
  13. lalalalala
  14. }
  15. }
  16. @Override
  17. public LocalDateTime convertToEntityAttribute(final byte[] bytes) {
  18. if (bytes == null) {
  19. return null;
  20. }
  21. try (final ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(bytes);
  22. final ObjectInputStream objectInputStream = new ObjectInputStream(byteArrayInputStream)) {
  23. final LocalDateTime dateTime = LocalDateTime.parse(objectInputStream.readObject().toString());
  24. return dateTime;
  25. } catch (Exception e) {
  26. lalalal
  27. }
  28. }

展开查看全部

相关问题