기존에는 회사에서 프로젝트를 진행할 때 mybatis를 사용했는데
신규 프로젝트는 JPA 2.0을 사용하게 되었습니다.
PK로 sequence를 사용하는 신규 테이블의 경우 요즘 대세인 테이블 전략을 사용하여 구성했습니다.
하지만, Login History는 같은 테이블을 사용하게 되면서 문제가 발생했습니다.
먼저, 기본키 자동 생성 전략에 대해 알아보면 아래와 같습니다.
- IDENTITY : 기본 키 생성을 데이터베이스에 위임하는 방법(데이터베이스에 의존적)
> 주로 MySQL, PostgresSQL, SQL Server, DB2에서 사용
- SEQUENCE : 데이터베이스 시퀀스를 사용해서 기본 키를 할당하는 방법(데이터베이스에 의존적)
> 주로 Oracle, PostgreSQL, DB2, H2에서 사용
> @SequenceGenerator를 사용하여 시퀀스 생성기를 등록하고,
실제 데이터베이스에 생성될 시퀀스 이름을 지정해 줘야 함
- TABLE : 키 생성 테이블을 사용하는 방법
> 키 생성 전용 테이블을 하나 만들고, 여기에 이름과 값으로 사용할 컬럼을 만드는 방법
> 테이블을 사용하므로 데이터베이스에 상관없이 모든 데이터베이스에 적용 가능
- AUTO : 데이터베이스에 의존하지 않고 기본키를 할당하는 방법
> 데이터베이스에 따라서 IDENTITY, SEQUENCE, TABLE 전략 중 하나를 자동으로 선택해 주는 방법.
> 예를 들어, Oracle일 경우 SEQUENCE를 자동으로 선택해서 사용.
따라서 데이터베이스를 변경해도 코드를 수정할 필요가 없음
[제약사항 및 문제점]
- 이전에는 각 DB에서 제공하는 auto_increment를 사용. 따라서 테이블 전략 적용 불가
- AUTO 전략을 사용했는데 IDENTITY 전략을 사용해야 하는 상황에서 TABLE 전략을 사용하는 상황 발생
에러가 발생한 코드는 아래와 같습니다.
@Setter
@Getter
@Entity
public class LoginHistory {
@Id
@GeneratedValue(strategy = GenerationType.AUTO)
private Long seqlog;
}
위 코드를 실행했을 때 에러 메시지는 아래와 같습니다.
Caused by: com.microsoft.sqlserver.jdbc.SQLServerException: 개체 이름 'hibernate_sequence'이(가) 잘못되었습니다.
이 오류가 발생하게 된 이유는 아래와 같습니다.
Spring boot의 변경사항
Spring Boot 2.0에서 spring.jpa.hibernate.use-new-id-generator-mappings의 default 값은 false입니다.
Spring Boot 1.5에서 spring.jpa.hibernate.use-new-id-generator-mappings의 default 값은 true입니다.
hibernate.id.new_generator_mappings = false일 경우
이 경우 사용하는 DB(Dialect)에 의해 Generator가 결정됩니다.
Dialect에서 supportsIdentityColumns()가 true 인 경우 IdentityGenerator를 사용하게 됩니다.
false 인 경우 SequenceStyleGenerator를 사용하게 됩니다.
hibernate.id.new_generator_mappings = true일 경우
이 경우 Sequence 기능을 지원하는 경우 SequenceGenerator를 사용하고,
Sequence 기능을 사용하지 않는 경우 TableGenerator를 사용합니다.
Hibernate의 변경사항
Hibernate 5부터 MSSQL에서의 GenerationType.AUTO는 IDENTITY가 아닌 TABLE을 기본 시퀀스 전략으로 사용합니다.
따라서 해결 방법은 application.properties에 아래 내용을 추가합니다.
spring.jpa.hibernate.use-new-id-generator-mappings=false
수정한 후 해당 코드를 실행하니 아래와 같은 Log가 나오며 성공했습니다.
20200720 14:19:50.028 [http-nio-8089-exec-1] DEBUG o.h.i.IdentifierGeneratorHelper - Natively generated identity: 175402
20200720 14:19:50.028 [http-nio-8089-exec-1] INFO j.resultsettable -
|---------------|
|generated_keys |
|---------------|
|175402 |
|---------------|
[참고사이트]
https://vladmihalcea.com/why-should-not-use-the-auto-jpa-generationtype-with-mysql-and-hibernate/
'Database ( DB ) > Database' 카테고리의 다른 글
HikariCP Dead lock (0) | 2020.10.18 |
---|---|
HikariCP 소개 (0) | 2020.10.17 |
SQL Server JDBC Driver(MSSQL)에서의 NVARCHAR, VARCHAR (0) | 2020.10.10 |
MSSQL 오류: 'xxx'은(는) 열 'xxx'에 종속되어 있습니다. (0) | 2020.09.24 |
[mssql ] DB 백업 후 다른 DB로 복원 (0) | 2020.04.03 |