SpringBoot + JPA问题汇总
实体类有继承父类,但父类没有单独标明注解
异常表现
1 Caused by: org.hibernate.AnnotationException: No identifier specified for entity: com.xxx.ProjectDTO
解决方式
- 可以看到ProjectDTO有继承一个BaseDTO,那么在父类中肯定存在某些字段需要与数据库表字段对应
- 因此类父需要使用
@MappedSuperclass
标注为映射的父类,即可解决上述问题@Entity @Table(name = "al_project") public class ProjectDTO extends BaseDTO { ... } @MappedSuperclass public class BaseDTO { ... }
自定义接口实现了JpaRepository,但没有单独标明注解
异常表现
Caused by: org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'baseJpaRepository': Invocation of init method failed; nested exception is java.lang.IllegalArgumentException: Not a managed type: class java.lang.Object ... Caused by: java.lang.IllegalArgumentException: Not a managed type: class java.lang.Object
解决方式
- 上述报错中提到BaseJpaRepository创建失败,其实是因为该接口继承JpaRepository,继承CrudRepository同理
- 导致SpringBoot把该类认为是Jpa的某一个存储库,所以需要添加
@NoRepositoryBean
告知SpringBoot该类不是一个存储库
@NoRepositoryBean public interface BaseJpaRepository<T, ID> extends JpaRepository<T, ID> { ... }
实体类创建后没有单独标明注解@Entity
异常表现
- 需要注意的是这个错误可能不完全以下内容
*************************** APPLICATION FAILED TO START *************************** ... The injection point has the following annotations: - @org.springframework.beans.factory.annotation.Autowired(required=true) ...
解决方式
- 检查所有的实体类,并添加全部上
@Entity
注解 - 需要注意的是以下实体类ConfigDTO继承了BaseDTO,而BaseDTO作为父类不应该使用
@Entity
注解 - 而使用应该问题上述提到中的
@MappedSuperclass
标明的英文映射父类
@Entity @Table(name = "al_config") public class ConfigDTO extends BaseDTO { ... }
实体类中的某个字段是一个对象,却错误的使用了@Column注解
异常表现
Caused by: org.hibernate.MappingException: Could not determine type for: com.asing1elife.teamnote.dto.ProjectGroupDTO, at table: al_project, for columns: [org.hibernate.mapping.Column(project_group)]
解决方式
- 以下类中ProjectGroupDTO应该是一个对象,所以根据应该具体映射情况决定的英文使用
@ManyToOne
还是@OneToMay
,或其他对象映射规则
@Entity @Table(name = "al_project") public class ProjectDTO extends BaseDTO { @Column private ProjectGroupDTO projectGroup; }
实体类ID使用了错误的生成规则
异常表现
java.sql.SQLSyntaxErrorException: Table 'asl_station.hibernate_sequence' doesn't exist
public class BaseDTO { @Id @GeneratedValue private Long id = 0L; }
解决方式
- 上述类中在ID上通过
@GeneratedValue
直接指定生成规则,就会抛出上述异常 - 因为默认的生成规则是
GeneratedType.AUTO
- 而我们在创建表时,通常使用的ID自增规则都是
auto_increment
,这对应的应该是GenerationType.IDENTITY
- 所以应该使用如下方式指定ID的自增规则
public class BaseDTO { @Id @GeneratedValue(strategy = GenerationType.IDENTITY) private Long id = 0L; }
修改JPA的字段命名规则为驼峰式
异常表现
- 在实体类和数据库中对应的字段都是createTime,但是JPA在连接对应表时却抛出如下错误
- 因为JPA默认的字段映射规则是下划线风格
Caused by: java.sql.SQLSyntaxErrorException: Unknown column 'create_time' in 'field list'
@Column private Date createTime = DateUtil.getSysDate();
解决方式
- 在配置文件中指定JPA的字段映射规则为驼峰式即可
spring:
jpa:
hibernate:
naming:
physical-strategy: org.hibernate.boot.model.naming.PhysicalNamingStrategyStandardImpl
获取单个实体类JSON转换异常
异常表现
- Hibernate的获取单个实体类数据后,为会每个实体类添加一个
hibernateLazyInitializer
属性,改属性在进行JSON转换时抛出异常
com.fasterxml.jackson.databind.exc.InvalidDefinitionException: No serializer found for class org.hibernate.proxy.pojo.bytebuddy.ByteBuddyInterceptor and no properties discovered to create BeanSerializer (to avoid exception, disable SerializationFeature.FAIL_ON_EMPTY_BEANS) (through reference chain: com.asing1elife.teamnote.core.bean.ResponseData["data"]->com.asing1elife.teamnote.dto.OrganizationDTO$HibernateProxy$f3Tr4vBI["hibernateLazyInitializer"])
解决方式一
- 在实体类顶部添加
@JsonIgnoreProperties("hibernateLazyInitializer")
,确保数据转换时会过滤掉休眠生成的属性@JsonIgnoreProperties("hibernateLazyInitializer") public class BaseDTO { ... }
解决方式二
- 上述根据报错
disable SerializationFeature.FAIL_ON_EMPTY_BEANS
可以尝试将JSON转换时的以下属性设置为false
spring: jackson: serialization: fail-on-empty-beans: false
前端数据传递到后端过程中,反序列失败
异常表现
- 杰克逊反序列化时需要无参构造函数,因为数据对应的实体类某个对象属性没有无参构造函数,就会抛出以下异常
(although at least one Creator exists): cannot deserialize from Object value (no delegate- or property-based Creator)
解决方式
- 检查实体类自身以及所有对象属性,为每个对象添加对应的无参构造函数即可解决
- 自参考博客园