이 튜토리얼은 웹 기반의 어플리케이션 용 Apache Tomcat 서블릿 컨테이너에 대한 Hibernate 3.0 셋업을 설명한다(우리는 버전 4.1을 사용했다. 5.0에 대한 차이점들은 적을 것이다). Hibernate는 모든 주요 J2EE 어플리케이션 서버들의 관리되는 환경에서 잘 동작하고, 또는 심지어 스탠드얼론 어플리케이션들에서도 잘 동작한다. 이 튜토리얼에 사용되는 데이터베이스 시스템은 PostgreSQL 7.4이고, 다른 데이터베이스에 대한 지원은 단지 Hibernate SQL dialect 구성과 커넥션 프로퍼티들을 변경시키는 것에만 관계된다.
먼저 우리는 모든 필수적인 라이브러리들을 Tomcat 설치 장소에 복사해야 한다. 우리는 이 튜토리얼을 위해 별도의 웹 컨텍스트(webapps/quickstart)를 사용하며, 따라서 우리는 전역 라이브러리 검색 경로(TOMCAT/common/lib)와 webapps/quickstart/WEB-INF/lib(JAR 파일들의 경우)와 webapps/quickstart/WEB-INF/classes 내 에 있는 컨텍스트 레벨에서 클래스로더 양자를 고려해야 한다. 우리는 두 개의 클래스로더 레벨들을 전역 classpath와 컨텍스트 classpath로 언급한다.
이제 라이브러리들을 두 개의 classpath들에 복사하라:
- 데이터베이스 용 JDBC 드라이버를 전역 classpath로 복사하라. 이것은 Tomcat에 번들로구성된 DBCP 커넥션 풀 소프트웨어에 필요하다. Hibernate는 데이터베이스 상에서 SQL을 실행시키는데 JDBC 커넥션들을 사용하므로, 당신은 풀링된 JDBC 커넥션들을 제공해야 하거나, 직접 지원되는 풀들(C3P0, Proxool) 중 하나를 사용하기 위해 Hibernate를 구성해야 한다. 이 튜토리얼을 위해, (PostgreSQL 7.4와 JDK 1.4용) pg74jdbc3.jar 라이브러리를 전역 classloaders 경로로 복사하라. 만일 당신이 다른 데이터베이스를 사용하고자 원할 경우, 간단하게 그것의 적절한 JDBC 드라이버를 복사하라.
- 그 밖의 어떤 것을 Tomcat 내의 전역 클래스로더 경로에 복사하지 말라. 또는 당신은 Log4j, commons-logging 그리고 다른 것들을 포함하는 여러 가지 도구들에 관련된 문제점들을 얻게 될 것이다. 각각의 웹 어플리케이션에 대해 컨텍스트 classpath를 사용하라. 즉 라이브러리들을 WEB-INF/lib에 복사하고, 당신 자신의 클래스들과 구성 파일들/프로퍼티 파일들을 WEB-INF/classes에 복사하라. 두 디렉토리들 양자는 디폴트로 컨텍스트 classpath 내에 있다.
- Hibernate는 JAR 라이브러리로서 패키지화 되어 있다. hibernate3.jar 파일은 어플리케이션의 다른 클래스들과 함께 컨텍스트 classpath 속에 복사되어야 한다. Hibernate는 실행 시에 어떤 제 3의 라이브러리들을 필요로하고, 이것들은 lib/ 디렉토리 내의 Hibernate 배포본에 번들화되어 있다; 표 1.1, “ Hibernate 3rd party libraries ”를 보라. 필요한 제3의 라이브러리들을 컨텍스트 classpath로 복사하라.
표 1.1. Hibernate 3rd party 라이브러리
라이브러리 |
설명 |
antlr (필수) |
Hibernate는 질의 파서들을 산출하는데 ANTLR을 사용하고, 이 라이브러리는 또한 실행 시에 필요하다. |
dom4j (필수) |
Hibernate는 XML 구성과 XML 매핑 메타데이터 파일들을 파싱하는데 dom4j를 사용한다. . |
CGLIB, asm (필수) |
Hibernate는 (Java reflection과 결합하여) 런타임 시에 클래스들을 고양시키는데 코드 생성 라이브러리를 사용한다. |
Commons Collections, Commons Logging (필수) |
Hibernate는 Apache Jakarta Commons 프로젝트로부터 다양한 유틸리티 라이브러리들을 사용한다. |
EHCache (필수) |
Hibernate는 second-level 캐시를 위한 다양한 캐시 프로바이더들을 사용할 수 있다. 만일 구성에서 변하지 않을 경우 EHCache가 디폴트 캐시 프로바이더이다. |
Log4j (옵션) |
Hibernate는 기본 로깅 메커니즘으로서 Log4j를 사용할 수 있는, Commons Logging API를 사용한다. 만일 Log4j 라이브러리가 컨텍스트 라이브러리 디렉토리 속에서 이용가능하다면, Commons Logging은 Log4j와 컨텍스트 classpath 내에 있는 log4j.properties 구성을 사용할 것이다. Log4j에 대한 예제 properties 파일은 Hibernate 배포본에 번들화 되어 있다. 따라서 당신이 이면에서 무엇이 진행되는 지을 보고자 원할 경우에 log4j.jar와 (src/에 있는) 구성 파일을 당신의 컨텍스트 classpath 속으로 복사하라. |
필수 여부? |
Hibernate 배포본 내에 있는 lib/README.txt 파일을 살펴보라. 이것은 Hibernate에 배포된 제 3의 라이브러리들의 최신 목록이다. 당신은 그곳에 열거된 모든 필수 라이브러리들과 옵션 라이브러리들을 찾게 될 것이다(여기서 "buildtime required"는 당신의 어플리케이션이 아니라 Hibernate에 대한 의미임을 노트하라). |
우리는 이제 Tomcat과 Hibernate 양자에서 데이터베이스 풀링과 공유를 설정한다. 이것은 Tomcat이 (그것의 빌드되어 있는 DBCP 풀링 특징을 사용하여) 풀링된 JDBC 커넥션들을 제공할 것이고, Hibernate가 JNDI를 통해 이들 커넥션들을 요청한다는 것을 의미한다. 달리 당신은 Hibernate로 하여금 커넥션 풀을 관리하도록 할 수 있다. Tomcat은 그것의 커넥션 풀을 JNDI에 바인드 시킨다; 우리는 리소스 선언을 Tomcat 메인 구성 파일 TOMCAT/conf/server.xml에 추가한다:
<Context path="/quickstart" docBase="quickstart">
<Resource name="jdbc/quickstart" scope="Shareable" type="javax.sql.DataSource"/>
<ResourceParams name="jdbc/quickstart">
<parameter>
<name>factory</name>
<value>org.apache.commons.dbcp.BasicDataSourceFactory</value>
</parameter>
<!-- DBCP database connection settings -->
<parameter>
<name>url</name>
<value>jdbc:postgresql://localhost/quickstart</value>
</parameter>
<parameter>
<name>driverClassName</name><value>org.postgresql.Driver</value>
</parameter>
<parameter>
<name>username</name>
<value>quickstart</value>
</parameter>
<parameter>
<name>password</name>
<value>secret</value>
</parameter>
<!-- DBCP connection pooling options -->
<parameter>
<name>maxWait</name>
<value>3000</value>
</parameter>
<parameter>
<name>maxIdle</name>
<value>100</value>
</parameter>
<parameter>
<name>maxActive</name>
<value>10</value>
</parameter>
</ResourceParams>
</Context>
우리가 이 예제에서 구성하는 컨텍스트는 quickstart로 명명되고, 그것의 베이스는 TOMCAT/webapp/quickstart 디렉토리이다. 임의의 서블릿들에 접근하기 위해, (물론 당신의 web.xml 속에 매핑된 서블릿의 이름을 추가하여) 당신의 브라우저에서 http://localhost:8080/quickstart 경로를 호출하라. 당신은 또한 계속 진행하고 이제 공백의 process() 메소드를 가진 간단한 서블릿을 생성시킬 수 있다.
Tomcat은 이제 java:comp/env/jdbc/quickstart로 JNDI을 통해 커넥션들을 제공한다. 만일 당신이 실행 중인 커넥션 풀을 얻는 것에 문제가 있다면 Tomcat 문서를 참조하라. 당신이 JDBC 드라이버 예외상황 메시지를 얻을 경우, 먼저 Hibernate 없이 JDBC 커넥션 풀을 셋업하라. Tomcat & JDBC 튜토리얼들은 그 웹 서이트에서 이용 가능하다.
당신의 다음 단계는 Hibernate를 구성하는 것이다. Hibernate는 그것이 JDBC 커넥션들을 얻는 방법을 알고 있어야 한다. 우리는 Hibernate의 XML 기반 구성을 사용한다. properties 파일을 사용하는 다른 접근법은 거의 동일하지만 XML 구문이 허용하는 몇몇 특징들을 누락하고 있다. XML 구성 파일은 hibernate.cfg.xml로서 컨텍스트 classpath (WEB-INF/classes) 내에 위치해 있다:
<?xml version='1.0' encoding='utf-8'?>
<!DOCTYPE hibernate-configuration PUBLIC
"-//Hibernate/Hibernate Configuration DTD//EN"
"http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">
<hibernate-configuration>
<session-factory>
<property name="connection.datasource">java:comp/env/jdbc/quickstart</property>
<property name="show_sql">false</property>
<property name="dialect">org.hibernate.dialect.PostgreSQLDialect</property>
<!-- Mapping files -->
<mapping resource="Cat.hbm.xml"/>
</session-factory>
</hibernate-configuration>
우리는 SQL 명령들에 대한 로깅을 사용하지 않고 Hibernate에게 사용되는 데이터베이스 SQL direct가 무엇인지 그리고 (Tomcat 바인드된 풀의 JNDI 주소를 선언하여) JDBC 커넥션들을 얻는 곳을 알려준다. dialect는 필수적인 설정이고, 데이터베이스들은 SQL "표준"에 대한 그것들의 해석을 달리한다. Hibernate는 차이점들을 처리하고 모든 주요 상용 데이터베이스들 및 오픈 소스 데이터베이스들 용도의 direct들을 번들로 포함하고 있다.
SessionFactory 는 단일 데이터저장소에 관한 개념이고, 여러 데이터베이스들은 여러 개의 XML 구성 파일들을 생성시키고 당신의 어플리케이션 속에서 여러 개의 Configuration 및 SessionFactory 객체들을 생성시켜서 사용될 수 있다.
hibernate.cfg.xml의 마지막 요소는 영속 클래스 Cat에 대한 Hibernate XML 매핑 파일의 이름으로써 Cat.hbm.xml을 선언한다. 이 파일은 데이터베이스 테이블(또는 테이블들)로 POJO 클래스 Cat 을 매핑시키는 메타데이터를 포함한다. 우리는 곧 그 파일로 되돌아 갈 것이다. 먼저 POJO 클래스를 작성하고 그런 다음 그것을 위한 매핑 메타데이터를 선언하자.
Hibernate는 영속 클래스들에 대한 Plain Old Java Objects (POJOs, 종종 Plain Ordinary Java Objects로 명명된다) 프로그래밍 모형으로 가장 잘 동작한다. POJO는 공용으로 가시적인 인터페이스로부터 내부적인 표상을 은폐시켜, getter와 setter 메소드들을 통해 접근가능한 클래스들의 프로퍼티들을 가진 자바빈과 꽤 유사하다(필요하다면 Hibernate는 또한 필드들에 직접 접근할 수 있다):
package org.hibernate.examples.quickstart;
public class Cat {
private String id;
private String name;
private char sex;
private float weight;
public Cat() {
}
public String getId() {
return id;
}
private void setId(String id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public char getSex() {
return sex;
}
public void setSex(char sex) {
this.sex = sex;
}
public float getWeight() {
return weight;
}
public void setWeight(float weight) {
this.weight = weight;
}
}
Hibernate는 그것의 프로퍼티 타입들의 사용에 제약되지 않고, 자바 콜렉션 프레임웍에서의 클래스들을 포함하여, 모든 자바 JDK 타입들과 (String, char와 Date 같은) 원시타입들이 매핑될 수 있다. 당신은 그것을 값들로서, 값들을 가진 콜렉션들로서, 또는 다른 엔티티들에 대한 연관들로서 매핑시킬 수 있다. id 는 그 클래스의 데이터베이스 식별자(프라이머리 키)를 표현하는 특별한 프로퍼티이고, 그것은 Cat과 같은 엔티티들에 대해 매우 권장된다. Hibernate는 내부적으로만 식별자들을 사용할 수 있지만, 우리는 우리의 어플리케이션 아키텍처에서 어떤 유연성을 상실하게 될 것이다.
특정 인터페이스는 영속 클래스들에 대해 구현되지 말아야 하거나 특정 루트 영속 클래스로부터 서브 클래스로 만들지 말아야 한다. Hibernate는 또한 바이트 코드 처리와 같은, 어떤 빌드 시 처리를 필요로 하지 않고, 그것은 오직 자바 reflection과 (CGLIB를 통한) 런타임 클래스 고양에만 의존한다. 따라서 Hibernate에 대한 POJO 클래스의 어떤 의존성 없이도, 우리는 그것을 데이터베이스 테이블로 매핑할 수 있다.
Cat.hbm.xml 매핑파일은 객체/관계형 매핑에 필요한 메타데이터를 포함한다. 메타데이터는 영속 클래스들의 선언과 데이터베이스 테이블들에 대한 (컬럼들과 다른 엔티티들에 대한 foreign 키 관계들에 대한) 프로퍼티들의 매핑을 포함한다.
<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mapping>
<class name="org.hibernate.examples.quickstart.Cat" table="CAT">
<!-- A 32 hex character is our surrogate key. It's automatically
generated by Hibernate with the UUID pattern. -->
<id name="id" type="string" unsaved-value="null" >
<column name="CAT_ID" sql-type="char(32)" not-null="true"/>
<generator class="uuid.hex"/>
</id>
<!-- A cat has to have a name, but it shouldn' be too long. -->
<property name="name">
<column name="NAME" length="16" not-null="true"/>
</property>
<property name="sex"/>
<property name="weight"/>
</class>
</hibernate-mapping>
모든 영속 클래스는 식별자 속성을 가져야 한다(실제로 value-타입의 클래스들에 의존하지 않는, 엔티티들을 표현하는 유일한 클래스들은 엔티티을 가진 컴포넌트들로 매핑된다). 이 프로퍼티는 영속 객체들을 구별짓는데 사용된다: 만일 catA.getId().equals(catB.getId())이 true일 경우, 두 개의 cat들은 같고, 이 개념은 database identity로 명명된다. Hibernate는 (데이터베이스 시퀀스, hi/lo 식별자 테이블들, 그리고 어플리케이션 할당 식별자들에 대한 native 생성기들을 포함하는) 다른 시나리오들에 대해 여러 가지 식별자 생성기들을 번들로 갖고 있다. 우리는 UUID 생성기(데이터이스에 의해 생성된 정수 대용 키들이 선호될 것이므로, 테스트용으로만 권장됨)를 사용하고 또한 Hibernate 생성된 식별자 값을 위한 CAT 테이블의 CAT_ID 컬럼을 (테이블의 프라이머리 키로서) 지정한다.
Cat 의 모든 다른 프로퍼티들은 동일한 테이블로 매핑된다. name 프로퍼티의 경우에, 우리는 그것을 명시적인 데이터베이스 컬럼 선언으로 매핑시켰다. 데이터베이스 스키마가 Hibernate의 SchemaExport 도구에 의해 매핑 선언으로부터 (SQL DDL 문장들로) 자동적으로 생성될 때 이것이 특별히 유용하다. 모든 다른 프로퍼티들은 Hibernate의 디폴트 설정들을 사용하여 매핑되고, 디폴트 설정들은 당신이 가장 많은 시간을 필요로 하는 것이다. 데이터베이스 내의 테이블 CAT 은 다음과 같다:
Column | Type | Modifiers
--------+-----------------------+-----------
cat_id | character(32) | not null
name | character varying(16) | not null
sex | character(1) |
weight | real |
Indexes: cat_pkey primary key btree (cat_id)
당신은 이제 수작업으로 당신의 데이터베이스 내에 이 테이블을 생성시킬 것이고, 만일 당신이 hbm2ddl 도구로 이 단계를 자동화 시키고자 원할 경우 20 장, 도구셋 안내를 읽어라. 이 도구는 테이블 정의, 맞춤형 컬럼 타입 컨스트레인트들, 유일 컨스트레인트들과 인덱스들을 포함하는, 전체 SQL DDL을 생성시킬 수 있다.
우리는 이제 Hibernate의 Session을 시작할 준비가 되어 있다. 그것은 persistence manager(영속 관리자)이고, 우리는 데이터베이스로 Cat를 저장하고 데이터베이스로부터 Cat를 검색하는데 그것을 사용한다. 그러나 먼저 우리는 SessionFactory로부터 Session (Hibernate의 작업 단위)를 얻어야 한다:
SessionFactory sessionFactory =
new Configuration().configure().buildSessionFactory();
configure() 호출은 hibernate.cfg.xml 구성 파일을 로드시키고 Configuration 인스턴스를 초기화 시킨다. 당신이 SessionFactory(불변적임)를 빌드하기 이전에 당신은 Configuration에 접근함으로써 다른 프로퍼티들을 설정할 수 있다(그리고 심지어 매핑 메타데이터를 변경시킬 수 있다). 우리는 어디서 SessionFactory를 생성시키고 우리의 어플리케이션 속에서 어떻게 그것에 접근할 수 있나?
SessionFactory 는 대개 오직 한번만, 예를 들어 대개 load-on-startup 서블릿으로 시작 시에 빌드된다. 이것은 또한 당신이 당신의 서블릿들 내에 있는 인스턴스 변수 속에 그것을 유지하지 않을 것이지만 어떤 다른 위치에 유지시킬 것임을 의미한다. 더구나 우리는 어떤 종류의 Singleton을 필요로 하며, 따라서 우리는 어플리케이션 코드로 쉽게 SessionFactory 에 액세스 할 수 있다. 다음에 보여진 접근법은 두 문제 모두를 해결한다: 시작 구성과 SessionFactory에 대한 쉬운 접근.
우리는 HibernateUtil helper 클래스를 구현한다:
import org.hibernate.*;
import org.hibernate.cfg.*;
public class HibernateUtil {
private static Log log = LogFactory.getLog(HibernateUtil.class);
private static final SessionFactory sessionFactory;
static {
try {
// Create the SessionFactory
sessionFactory = new Configuration().configure().buildSessionFactory();
} catch (Throwable ex) {
// Make sure you log the exception, as it might be swallowed
log.error("Initial SessionFactory creation failed.", ex);
throw new ExceptionInInitializerError(ex);
}
}
public static final ThreadLocal session = new ThreadLocal();
public static Session currentSession() {
Session s = (Session) session.get();
// Open a new Session, if this Thread has none yet
if (s == null) {
s = sessionFactory.openSession();
session.set(s);
}
return s;
}
public static void closeSession() {
Session s = (Session) session.get();
if (s != null)
s.close();
session.set(null);
}
}
이 클래스는 static 초기자를 가진 SessionFactory를 처리할 뿐만 아니라 또한 현재의 쓰레드를 위한 Session 을 보관하는 ThreadLocal변수를 갖는다. 이 helper를 사용하려고 시도하기 전에 thread-local 변수에 대한 자바 개념을 이해해야 한다. 보다 복잡하고 강력한 HibernateUtil 클래스는 http://caveatemptor.hibernate.org/의 CaveatEmptor에서 찾을 수 있다.
SessionFactory는 threadsafe이고, 많은 쓰레드들이 동시에 그것에 접근할 수 있고 Session들을 요청할 수 있다. 하나의 Session은 데이터베이스에 대해 한 개의 단위 작업을 나타내는 non-threadsafe 객체이다. Session들은 SessionFactory로부터 열려지고 모든 작업이 완료될 때 닫혀진다. 당신의 서블릿의 process() 메소드 내에 있는 예제는 다음과 같을 수 있다(예외상황 처리 없이):
Session session = HibernateUtil.currentSession();
Transaction tx = session.beginTransaction();
Cat princess = new Cat();
princess.setName("Princess");
princess.setSex('F');
princess.setWeight(7.4f);
session.save(princess);
tx.commit();
HibernateUtil.closeSession();
하나의 Session 내에서 모든 데이터베이스 오퍼레이션은 데이터베이스 오퍼레이션들(심지어 읽기 전용 오퍼레이션들 조차도)을 격리시키는 하나의 트랜잭션 내부에서 발생한다. 우리는 기본 트랜잭션 방도(우리의 경우, JDBC 틀내잭션들)로부터 추상화시키는데 Hibernates Transaction API 를 사용한다. 이것은 우리의 코드가 임의의 변경들 없이도 (JTA를 사용하는) 컨테이너-관리되는 트랜잭션들에 배치되는 것을 허용해준다.
당신이 원하는 만큼 당신이 HibernateUtil.currentSession();을 호출할 수 있고, 당신은 이 쓰레드의 현재 Session을 항상 얻을 것임을 노트하라. 당신은 서블릿 코드 내에서든 또는 서블릿 필터 내에서든 HTTP response가 전송되기 전에, 당신의 단위 작업이 완료된 후에 Session을 확실히 닫아야 한다. 두 번째 옵션의 좋은 측면은 쉬운 lazy 초기화이다: 뷰가 렌더링 될 때 Session이 여전히 열려져 있어서, Hibernate는 당신이 현재 객체 그래프를 네비게이트 하는 동안 초기화 되지 않은 객체들을 로드시킬 수 있다.
Hibernate는 데이터베이스로부터 객체들을 검색하는데 사용될 수 있는 다양한 메소드들을 갖고 있다. 가장 유연한 방법은 Hibernate Query Language (HQL)을 사용하는 것이다. Hibernate Query Language (HQL)은 배우기가 쉽고 SQL에 대한 강력한 객체 지향 확장이다:
Transaction tx = session.beginTransaction();
Query query = session.createQuery("select c from Cat as c where c.sex = :sex");
query.setCharacter("sex", 'F');
for (Iterator it = query.iterate(); it.hasNext();) {
Cat cat = (Cat) it.next();
out.println("Female Cat: " + cat.getName() );
}
tx.commit();
Hibernate는 또한 type-safe 질의들을 공식화 시키는데 사용될 수 있는 객체 지향 query by criteria API을 제공한다. 물론 Hibernate는 데이터베이스와의 모든 SQL 통신을 위해 PreparedStatement들과 파라미터 바인딩을 사용한다. 당신은 또한 Hibernate 직접적인 SQL 질의 특징을 사용할 수도 있거나 드문 경우에 Session 로부터 plain JDBC 커넥션을 얻을 수도 있다.
우리는 이 작은 튜토리얼 내에서 단지 Hibernate의 표면을 훑기만 했다. 우리는 우리의 예제들 속에 어떤 서블릿 지정적 코드를 포함하지 않음을 노트하라. 당신이 적합한지를 알려고 할 때 당신은 당신 자신의 서블릿을 생성시켜야 하고 Hibernate 코드를 삽입해야 한다.
데이터 접근 계층으로서 Hibernate는 당신의 어플리케이션에 강하게 통합됨을 염두에 두라. 대개 모든 다른 레이어들은 영속 메커니즘에 의존했다. 당신은 이 설계의 함축을 확실히 이해하도록 하라.
보다 복잡한 어플리케이션 예제는 http://caveatemptor.hibernate.org/ 를 보고 http://caveatemptor.hibernate.org/에 있는 다른 튜토리얼들을 살펴보라.
Hibernate가 많은 다른 환경들에서 동작하도록 설계되어 있으므로, 많은 개수의 구성 파라미터들이 존재한다. 다행히 대부분은 유의미한 디폴트 값들이고 Hibernate는 다양한 옵션들을 보여주는 etc/ 내 의 예제 파일 hibernate.properties로 배포된다. 당신은 대개 당신의 classpath 경로 속에 그 파일을 집어넣고 그것을 커스트마이징하기만 해야 한다.
org.hibernate.cfg.Configuration 의 인스턴스는 어플리케이션의 Java 타입들을 SQL 데이터베이스 타입으로의 전체 매핑 집합을 표현한다. Configuration 은 (불변의) SessionFactory를 빌드하는데 사용된다. 매핑들은 여러 가지 XML 매핑 파일들로부터 컴파일 된다.
당신은 Configuration 인스턴스를 초기화 함으로써 Configuration 인스턴스를 얻을 수 있다. 다음은 (classpath 내에 있는) 두 개의 XML 매핑 파일 속에 정의된 매핑들로부터 데이터저장소를 설정하는 예제이다:
Configuration cfg = new Configuration()
.addFile("Item.hbm.xml")
.addFile("Bid.hbm.xml");
(때때로 더 나은) 다른 방법은 getResourceAsStream()을 사용하여 Hibernate로 하여금 매핑 파일을 로드시키도록 하는 것이다:
Configuration cfg = new Configuration()
.addClass(org.hibernate.auction.Item.class)
.addClass(org.hibernate.auction.Bid.class);
그런 다음 Hibernate는 classpath 내에 /org/hibernate/auction/Item.hbm.xml 과 /org/hibernate/auction/Bid.hbm.xml로 명명된 매핑 파일들을 룩업할 것이다. 이 접근법은 임의의 하드코딩된 파일 이름들을 제거한다.
Configuration 은 또한 다양한 옵션 프로퍼티들을 지정한다:
Configuration cfg = new Configuration()
.addClass(org.hibernate.auction.Item.class)
.addClass(org.hibernate.auction.Bid.class)
.setProperty("hibernate.dialect", "org.hibernate.dialect.MySQLInnoDBDialect")
.setProperty("hibernate.connection.datasource", "java:comp/env/jdbc/test")
.setProperty("hibernate.order_updates", "true");
이것은 컨피그레이션 프로퍼티들을 Hibernate에 전달하는 유일한 방법이 아니다. 여러 가지 옵션들은 다음을 포함한다:
- java.util.Properties 의 인스턴스를 Configuration.setProperties()에 전달한다 .
- classpath의 루트 디렉토리에 hibernate.properties 를 위치지운다.
- java -Dproperty=value를 사용하여 System properties를 설정한다.
- hibernate.cfg.xml에 <property> 요소들을 포함한다(나중에 논의됨).
hibernate.properties 는 당신이 빠르게 시작하고 원할 경우 가장 쉬운 접근법이다.
Configuration 은 시작 시(startup-time) 객체로서 일단 SessionFactory가 생성되면 폐기되게끔 예정되어 있다.
모든 매핑들이 Configuration에 의해 파싱되었을 때, 어플리케이션은 Session 인스턴스들에 대한 팩토리를 얻어야 한다. 이 팩토리는 모든 어플리케이션 쓰레드들에 의해 공유되도록 고안되었다:
SessionFactory sessions = cfg.buildSessionFactory();
하지만 Hibernate는 당신의 어플리케이션이 하나 이상의 SessionFactory를 초기화 시키는 것을 허용한다. 이것은 당신이 하나 이상의 데이터베이스를 사용하는 경우에 유용하다.
SessionFactory 는 사용자 제공 JDBC 커넥션 상에서 Session 을 열 수 있다. 이 설계 선택은 그것이 만족될 때마다 어플리케이션이 JDBC 커넥션들을 얻는 것을 자유롭게 해준다:
java.sql.Connection conn = datasource.getConnection();
Session session = sessions.openSession(conn);
// 어떤 데이터 액세스 작업을 행한다
어플리케이션은 동일한 JDBC 커넥션 상에서 두 개의 동시적인 Session들 을 열지 않도록 주의해야 한다!
우리는 사용자 제공 JDBC 커넥션들을 권장하지 않는다. 왜냐하면 Hibernate가 캐싱을 불가능하게 할 것이기 때문이고(그것은 당신이 주어진 커넥션 상에서 실행할 수 있다는 사실을 알지 못한다) 대개 다음 옵션들 중 하나가 더 적절하기 때문이다.
대개 당신은 SessionFactory 를 생성시키고 당신을 위한 JDBC 커넥션들을 풀링시키고자 원한다. 만일 당신이 이 접근법을 취하라 경우, Session을 여는 것은 다음과 같이 간단하다:
Session session = sessions.openSession(); // 새로운 Session을 연다
데이터베이스에 대한 업근을 필요로 하는 어떤 것을 당신이 행하자 마자, JDBC 커넥션이 그 풀로부터 획득될 것이다.
이것이 동작하도록 하기 위해, 우리는몇몇 JDBC 커넥션 프로퍼티들을 Hibernate에 전달할 필요가 있다. 모든 Hibernate 프로퍼티 이름들과 의미들은 org.hibernate.cfg.Environment 클래스 상에 정의되어 있다.우리는 이제 JDBC 커넥션 구성을 위한 가장 중요한 설정들을 설명할 것이다.
당신이 다음 프로퍼티들을 설정할 경우, Hibernate는 java.sql.DriverManager를 사용하여 커넥션들을 획득(그리고 풀링)할 것이다.:
표 3.1. Hibernate JDBC 프로퍼티들
Property name |
Purpose |
hibernate.connection.driver_class |
jdbc 드라이버 클래스 |
hibernate.connection.url |
jdbc URL |
hibernate.connection.username |
데이터베이스 사용자 |
hibernate.connection.password |
데이터베이스 사용자 패스워드 |
hibernate.connection.pool_size |
풀링된 커넥션들의 최대 개수 |
하지만 Hibernate 자신의 커넥션 풀링 알고리즘은 아주 기본적이다. 그것은 당신이 시작하는 것을 도와주려고 의도되었고 제품 시스템 용도 또는 퍼포먼스 테스트용으로 고안되지 않았다. 최상의 퍼포먼스와 안정성을 위해서는 제 3의 풀을 사용하라. 즉 hibernate.connection.pool_size 프로퍼티를 커넥션 풀 지정 설정들로 대체하라. 이것은 Hibernate의 내부 pool을 오프시킬 것이다. 예를 들어 당신은 C3P0를 사용할 수도 있다.
C3P0는 lib 디펙토리 속에 Hibernate에 배포된 오픈 소스 JDBC 커넥션 풀이다. 당신이 hibernate.c3p0.* 프로퍼티들을 설정할 경우 Hibernate는 커넥션 풀링을 위해 그것의 C3P0ConnectionProvider 를 사용할 것이다. 만일 당신이 Proxool 을 사용하고자 원할 경우 패키지화 된 hibernate.properties 를 참조하고 추가 정보는 Hibernate 웹 사이트를 참조하라.
다음은 C3P0에 대한 사용하는 예제 hibernate.properties 파일이다:
hibernate.connection.driver_class = org.postgresql.Driver
hibernate.connection.url = jdbc:postgresql://localhost/mydatabase
hibernate.connection.username = myuser
hibernate.connection.password = secret
hibernate.c3p0.min_size=5
hibernate.c3p0.max_size=20
hibernate.c3p0.timeout=1800
hibernate.c3p0.max_statements=50
hibernate.dialect = org.hibernate.dialect.PostgreSQLDialect
어플리케이션 서버 내부의 용도로, 당신은 JNDI로 등록된 어플리케이션 서버 Datasource로부터 커넥션을 얻기 위해 항상 Hibernate를 구성해야 한다. 당신은 적어도 다음 프로퍼티들 중 하나를 설정할 필요가 잇을 것이다.
표 3.2. Hibernate Datasource 프로퍼티들
프로퍼티 이름 |
용도 |
hibernate.connection.datasource |
datasource JNDI 이름 |
hibernate.jndi.url |
NDI 프로바이더의 URL(옵션) |
hibernate.jndi.class |
JNDI InitialContextFactory의 클래스 (옵션) |
hibernate.connection.username |
데이터베이스 사용자(옵션) |
hibernate.connection.password |
데이터베이스 사용자 패스워드(옵션) |
다음은 어플리케이션 서버 제공 JNDI 데이터소스용 예제 hibernate.properties 파일이다:
hibernate.connection.datasource = java:/comp/env/jdbc/test
hibernate.transaction.factory_class = \
org.hibernate.transaction.JTATransactionFactory
hibernate.transaction.manager_lookup_class = \
org.hibernate.transaction.JBossTransactionManagerLookup
hibernate.dialect = org.hibernate.dialect.PostgreSQLDialect
JNDI datasource로부터 얻어진 JDBC 커넥션들은 어플리케이션 서버의 컨테이너에 의해 관리되는 트랜잭션들에 자동적으로 참여할 것이다.
임의의 커넥션 프로퍼티들은 프로퍼티 이름 앞에 "hibernate.connnection" 을 첨가하여 부여될 수 있다. 예를 들어 당신은 hibernate.connection.charSet을 사용하여 charSet을 지정할 수도 있다.
당신은 org.hibernate.connection.ConnectionProvider 인터페이스를 구현함으로써 JDBC 커넥션들을 얻는 당신 자신의 플러그인 방도를 정의할수도 있다. 당신은 hibernate.connection.provider_class를 설정하여 맞춤형 구현을 선택할 수도 있다.
실행 시에 Hibernate의 행위를 제어하는 많은 다른 프로퍼티들이 존재한다. 모든 것이 옵션이지만 합당한 디폴트 값들을 갖는다.
경고: 이들 프로퍼티들 중 몇몇은 ''system-level" 전용이다. 시스템 레벨 프로퍼티들은 오직 java -Dproperty=value 또는 hibernate.properties를 통해서만 설정될 수 있다. 그것들은 위에 설명된 다른 기법들에 의해 설정될 수 없다. .
표 3.3. Hibernate 구성 프로퍼티들
프로퍼티 이름 |
용도 |
hibernate.dialect |
Hibernate Dialect - 의 클래스명은 어떤 플랫폼 의존적인 특징들을 가능하도록 해준다.
예. full.classname.of.Dialect |
hibernate.show_sql |
모든 SQL 문장들을 콘솔에 기록한다
예. true | false |
hibernate.default_schema |
생성된 SQL 내에 주어진 schema/tablespace로서 unqualified tablenames을 qualify 시킨다.
예. SCHEMA_NAME |
hibernate.default_catalog |
주어진 SQL 내에 주어진 카타록으로서 unqualified tablenames을 qualify 시킨다.
eg. CATALOG_NAME |
hibernate.session_factory_name |
SessionFactory는 그것이 생성된 후에 JNDI 내에 있는 이 이름에 자동적으로 바인드 될 것이다.
예. jndi/composite/name |
hibernate.max_fetch_depth |
single-ended 연관관계들(one-to-one, many-to-one)의 경우에 outer join fetch 트리의 최대 "깊이"를 설정한다. 0 은 디폴트 outer join fetching을 사용불가능하게 만든다.
예. 권장 0과 3 사이의 값들이권장된다 |
hibernate.default_batch_fetch_size |
연관들의 Hibernate 배치 페칭에 대한 디폴트 크기를 설정한다.
예. 권장되는 값들은 4, 8, 16 |
hibernate.default_entity_mode |
이 SessionFactory로부터 열려진 모든 세션들에 대해 엔티티 표현을 default 모드로 설정한다
dynamic-map, dom4j, pojo |
hibernate.order_updates |
업데이트 주인 항목들의 프라이머리 키 값에 의해 SQL 업데이트들이 순서(ordering) 지워지도록 Hibernate에게 강제시킨다. 이것은 고도의 동시성 시스템들에서 더 적은 트랜잭션 데드락들로 귀결될 것이다.
예. true | false |
hibernate.generate_statistics |
가능할 경우, Hibernate는 퍼포먼스 튜닝에 유용한 통계들을 수집할 것이다.
예. true | false |
hibernate.use_identifer_rollback |
가능할 경우, 생성된 식별자 프로퍼티들은 객체가 삭제될 때 디폴트 값들로 재설정될 것이다.
예. true | false |
hibernate.use_sql_comments |
사용가능이 되면, Hibernate는 보다 쉬운 디버깅을 위해 SQL 내에 주석들을 생성시킬 것이다. 디폴트는 false.
예. true | false |
표 3.4. Hibernate JDBC 및 커넥션 프로퍼티들
프로퍼티 이름 |
용도 |
hibernate.jdbc.fetch_size |
0 아닌 값은 JDBC fetch 사이즈를 결정한다(Statement.setFetchSize()을 호출한다 ). |
hibernate.jdbc.batch_size |
0 아닌 값은 Hibernate에 의한 JDBC2 배치 업데이트의 사용을 이용 가능하게 한다.
예. 권장되는 값들은 5 와 30 사이의 값 |
hibernate.jdbc.batch_versioned_data |
당신의 JDBC 드라이버가 executeBatch()로부터 정확한 행 카운트들을 반환할 경우에 이 프로퍼티를 true 로 설정하라(대개 이 옵션을 사용 가능하게 하는 것이 안전하다). 그러면 Hibernate는 자동적으로 버전화 된 데이터에 대해 배치화된(batched) DML을 사용할 것이다. 디폴트는 false.
예. true | false |
hibernate.jdbc.factory_class |
맞춤형 Batcher를 선택한다. 대부분의 어플리케이션들은 이 구성 프로퍼티를 필요로 하지 않을 것이다.
예. classname.of.Batcher |
hibernate.jdbc.use_scrollable_resultset |
Hibernate에 의한 JDBC2 스크롤 가능한 결과셋들의 사용을 가능하게 해준다. 이 프로퍼티는 사용자가 제공한 JDBC커넥션들을 사용할 때에만 필수적이고, 그 밖의 경우 Hibernate는 커넥션 메타데이터를 사용한다.
예. true | false |
hibernate.jdbc.use_streams_for_binary |
JDBC로 binary 또는 serializable 타입들을 기록하고/JDBC로부터 binary 또는 serializable타입 들을 읽어들일 때 스트림들을 사용한다(시스템 레벨 프로퍼티).
예. true | false |
hibernate.jdbc.use_get_generated_keys |
insert 후에 고유하게 생성된 키들을 검색하는데 JDBC3 PreparedStatement.getGeneratedKeys()의 사용을 이용 가능하도록 만든다. JDBC3+ 드라이버와 JRE1.4+를 필요로 하고, 당신의 드라이버가 Hibernate 식별자 생성자들에 문제가 있을 경우에 false로 설정하라. 디폴트로 커넥션 메타 데이터를 사용하여 드라이버 가용성들을 결정하도록 시도하라.
예. true|false |
hibernate.connection.provider_class |
Hibernate에 JDBC 커넥션들을 제공하는 맞춤형 ConnectionProvider의 클래스명.
예. classname.of.ConnectionProvider |
hibernate.connection.isolation |
JDBC transaction isolation 레벨을 설정한다. 의미있는 값들로 java.sql.Connection을 체크하지만 대부분의 데이터베이스들이 모든 isolate 레벨들을 지원하지 않음을 유의하라.
예. 1, 2, 4, 8 |
hibernate.connection.autocommit |
JDBC 풀링된 커넥션들에 대해 자동커밋을 이용 가능하도록 한다(권장되지 않음).
예. true | false |
hibernate.connection.aggressive_release |
Hibernate로 하여금 만일 지원될 경우 ConnectionProvider에 의해 지원되는 JDBC Connection들을 적극적으로 해제하는 것을 가능하도록 한다. 디폴트로 (연결 해제된 상태를 제외하면) 한 개의 JDBC 커넥션은 Session의 수명 동안 보관된다; 이것은 커넥션 "containment(봉쇄)" 체크들을 구현하는 관리 환경들에서는 잘 동작하지 않는다. 관리 환경들에서 권장됨.
eg. true | false |
hibernate.connection.<propertyName> |
JDBC 프로퍼티 propertyName 를 DriverManager.getConnection()에 전달한다. |
hibernate.jndi.<propertyName> |
propertyName 프로퍼티를 JNDI InitialContextFactory에 전달한다 |
표 3.5. Hibernate Cache 프로퍼티들
프로퍼티 이름 |
용도 |
hibernate.cache.provider_class |
맞춤형 CacheProvider의 클래스명.
예. classname.of.CacheProvider |
hibernate.cache.use_minimal_puts |
읽기가 매우 빈번한 경우에, 쓰기를 최소화 시키기 위해 second-level 캐시 연산을 최적화 시킨다. 이 설정은 클러스터된 캐시들에 가장 유용하고, Hibernate3에서는 클러스터링된 캐시 구현들에 대해 디폴트로 이용 가능하다.
예. true|false |
hibernate.cache.use_query_cache |
질의 캐시를 가능하게 만든다. 개별적인 질의들은 여전히 캐시 가능으로 설정되어야 한다.
예. true|false |
hibernate.cache.use_second_level_cache |
second-level 캐시를 완전히 사용 불가능하게 하는데 사용될 수 있고, 그것은 <cache> 매핑을 지정하는 클래스들에 대해 디폴트로 이용 가능이다.
예. true|false |
hibernate.cache.query_cache_factory |
맞춤형 QueryCache 인터페이스의 클래스명. 디폴트는 미리 빌드된 StandardQueryCache.
예. classname.of.QueryCache |
hibernate.cache.region_prefix |
second-level 캐시 영역 이름들에 사용할 접두어.
예. prefix |
hibernate.cache.use_structured_entries |
인간에게 보다 더 친숙한 형식으로 second-level 캐시 속에 데이터를 저장하도록 Hibernate에게 강제시킨다.
예. true|false |
표 3.6. Hibernate Transaction 프로퍼티들
프로퍼티 이름 |
용도 |
hibernate.transaction.factory_class |
Hibernate Transaction API 에 사용할 TransactionFactory의 클래스 이름.(디폴트는 JDBCTransactionFactory).
예. classname.of.TransactionFactory |
jta.UserTransaction |
어플리케이션 서버로부터 JTA UserTransaction을 얻기 위해 JTATransactionFactory에 의해 사용되는 JNDI 이름.
예. jndi/composite/name |
hibernate.transaction.manager_lookup_class |
TransactionManagerLookup 의 클래스명- JVM 레벨의 캐싱이 이용 가능할 때 또는 JTA 환경에서 hilo generator를 사용할 때 필요하다.
예. classname.of.TransactionManagerLookup |
hibernate.transaction.flush_before_completion |
만일 사용가능토록 하면, 트랜잭션의 before completion 단계 동안에 세션이 자동적으로 flush 될 것이다.(CMT에 대해 Hibernate를 사용할 때 매우 유용하다.)
예. true | false |
hibernate.transaction.auto_close_session |
만일 사용가능토록 하면, before completion 단계 동안에 세션이 자동적으로 닫혀질 것이다. (CMT에 대해 Hibernate를 사용할 때 매우 유용하다.)
예. true | false |
표 3.7. 여러가지 프로퍼티들
프로퍼티 이름 |
용도 |
hibernate.query.factory_class |
Chooses the HQL 파서 구현을 선택한다.
예. org.hibernate.hql.ast.ASTQueryTranslatorFactory 또는 org.hibernate.hql.classic.ClassicQueryTranslatorFactory |
hibernate.query.substitutions |
Hibernate 질의들 내의 토큰들로부터 SQL 토큰들로의 매핑(예를 들어 토큰들은 함수 이름 또는 리터럴 이름일 수 있다).
예. hqlLiteral=SQL_LITERAL, hqlFunction=SQLFUNC |
hibernate.hbm2ddl.auto |
SessionFactory 가 생성될 때 스키마 DDL을 데이터베이스로 자동적으로 내보낸다. create-drop의 경우, SessionFactory 가 명시적으로 닫혀질 때,, 데이터베이스 스키마가 드롭될 것이다.
예. update | create | create-drop |
hibernate.cglib.use_reflec tion_optimizer |
런타임 reflection 대신에 CGLIB의 사용을 가능하도록 만든다(시스템 레벨 프로퍼티). Reflection은 문제가 발생할 시에 때때로 유용할 수 있고, 당신이 optimizer를 사용하지 않을 경우조차도 Hibernate는 항상 필요로 함을 유의하라. 당신은 ibernate.cfg.xml 속에 이 프로퍼티를 설정할수 없다.
예. true | false |
3.4.1. SQL Dialects
당신은 항상 당신의 데이터베이스를 위해 hibernate.dialect 프로퍼티를 정확한 org.hibernate.dialect.Dialect 서브클래스로 설정해야 한다. 만일 당신이 dialect를 지정할 경우, 당신이 프로퍼티들을 수작업으로 지정하는 노력을 절약하도록 Hibernate는 위에 열거된 다른 프로퍼티들 중 몇몇에 대해 의미있는 디폴트들을 사용할 것이다.
표 3.8. Hibernate SQL Dialects (hibernate.dialect)
RDBMS |
Dialect |
DB2 |
org.hibernate.dialect.DB2Dialect |
DB2 AS/400 |
org.hibernate.dialect.DB2400Dialect |
DB2 OS390 |
org.hibernate.dialect.DB2390Dialect |
PostgreSQL |
org.hibernate.dialect.PostgreSQLDialect |
MySQL |
org.hibernate.dialect.MySQLDialect |
Oracle (any version) |
org.hibernate.dialect.OracleDialect |
Oracle 9/10g |
org.hibernate.dialect.Oracle9Dialect |
Sybase |
org.hibernate.dialect.SybaseDialect |
Sybase Anywhere |
org.hibernate.dialect.SybaseAnywhereDialect |
Microsoft SQL Server |
org.hibernate.dialect.SQLServerDialect |
SAP DB |
org.hibernate.dialect.SAPDBDialect |
Informix |
org.hibernate.dialect.InformixDialect |
HypersonicSQL |
org.hibernate.dialect.HSQLDialect |
Ingres |
org.hibernate.dialect.IngresDialect |
Progress |
org.hibernate.dialect.ProgressDialect |
Mckoi SQL |
org.hibernate.dialect.MckoiDialect |
Interbase |
org.hibernate.dialect.InterbaseDialect |
Pointbase |
org.hibernate.dialect.PointbaseDialect |
FrontBase |
org.hibernate.dialect.FrontbaseDialect |
Firebird |
org.hibernate.dialect.FirebirdDialect |
3.4.2. Outer Join Fetching
만일 당신의 데이터베이스가 ANSI, Oracle, 또는 Sybase 스타일의 outer join들을 지원할 경우, outer join fetching은 (데이터베이스 그 자체에 의해 보다 더 많은 작업이 수행되는 비용으로) 데이터베이스로의 그리고 데이터베이스로부터의 라운드 트립들의 개수를 제한함으로써 종종 퍼포먼스를 증가시킬 것이다. Outer join fetching은 many-to-one, one-to-many, many-to-many,one-to-one 연관관계들이 에 의해 연결된 객체들의 전체 그래프가 하나의 SQL SELECT 속에서 검색되게끔 허용해준다.
Outer join fetching은 hibernate.max_fetch_depth 프로퍼티를 0으로 설정함으로써 전역적으로 사용 불가능하게 할 수 있다. 1 이상의 값을 설정하는 것은 fetch="join"으로 매핑되었던 모든 one-to-one 및 many-to-one 연관관계들에 대해 outer join fetching을 사용 가능하도록 만든다.
추가 정보는 19.1 절, “페칭 방도들”를 보라
Oracle은 JDBC 드라이버 로/부터 전달되는 byte 배열들의 크기를 제한시킨다. 만일 당신이 binary 또는 serializable 타입의 대형 인스턴스를 사용하고자 원할 경우에, 당신은 hibernate.jdbc.use_streams_for_binary를 사용 가능하게 해야 할 것이다. 이것은 오직 시스템 레벨 설정이다.
3.4.4. 두 번째 레벨 캐시 및 query 캐시
hibernate.cache 접두어가 붙은 프로퍼티들은 Hibernate에 대해 프로세스 또는 클러스터 범위의 두 번째 레벨 캐시 시스템을 사용하는 것을 허용해준다. 상세한 것은 19.2 절, “Second Level 캐시”를 보라.
당신은 hibernate.query.substitutions을 사용하여 새로운 Hibernate 질의 토큰들을 정의할 수 있다. 예를 들어:
hibernate.query.substitutions true=1, false=0
은 true과 false 토큰들로 하여금 생성된 SQL 내에서 정수 리터럴들로 번역되도록 강제할 것이다.
hibernate.query.substitutions toLowercase=LOWER
은 SQL LOWER 함수 이름을 변경하는 것을 당신에게 허용해 줄 것이다.
만일 당신이 hibernate.generate_statistics를 사용 가능하도록 할 경우, Hibernate는 을 SessionFactory.getStatistics() 통해 시스템 실행을 튜닝할 때 유용한 많은 측정들을 노출시킬 것이다. Hibernate는 심지어 JMX를 통해 이들 통계들을 노출시키도록 구성될 수 있다. 추가 정보는 org.hibernate.stats에 있는 인터페이스에 관한 Javadoc를 읽어라.
Hibernate는 Apache commons-logging를 사용하여 다양한 이벤트들을 로그시킨다.
commons-logging 서비스는 (만일 당신이 classpath 내에 log4j.jar를 포함할 경우) Apache Log4j로 또는 (JDK1.4 이상의 버전에서 실행될 경우) JDK 1.4 로깅으로 직접 출력할 것이다. 당신은 http://jakarta.apache.org에서 Log4j를 다운로드 할 수 있다. Log4j를 사용하기 위해, 당신은 log4j.properties 파일을 당신의 classpath 내에 위치지울 필요가 있을 것이고, 예제 properties 파일은 Hibernate의 src/ 디렉토리 내에 배포되어 있다.
우리는 당신이 Hibernate의 로그 메시지들에 익숙해지는 것을 강력하게 권장한다. 많은 작업은 Hibernate 로그를 읽기 불가능하게 만들지 않게끔 가능한 한 상세하게 Hibernate 로그를 집어 넣었다. 그것은 본질적인 문제던지기 장치이다. 가장 흥미로운 로그 카테고리들이 다음에 있다:
표 3.9. Hibernate Log 카테고리들
카테고리 |
기능 |
org.hibernate.SQL |
SQL DML 문장들이 실행될 때 그것들 모두를 로그 시킨다 |
org.hibernate.type |
모든 JDBC 파라미터들을 로그시킨다 |
org.hibernate.tool.hbm2ddl |
SQL DDL 문장들이 실행될 때 그것들 모두를 로그 시킨다 |
org.hibernate.pretty |
flush 시점에서 세션과 연관된 모든 엔티티들(최대 20개의 엔티티들)의 상태를 로그 시킨다 |
org.hibernate.cache |
모든 second-level 캐시 액티비티를 로그시킨다 |
org.hibernate.transaction |
트랜잭션 관련 액티비티를 로그 시킨다 |
org.hibernate.jdbc |
모든 JDBC 리소스 취득을 로그 시킨다 |
org.hibernate.hql.ast |
HQL AST와 SQL AST 그리고 질의 파싱에 관한 다른 정보를 로그시킨다 |
org.hibernate.secure |
모든 JAAS 허가 요청들을 로그시킨다 |
org.hibernate |
모든 것을 로그시킨다(많은 정보이지만, 문제해결에 매우 유용하다) |
Hibernate로 어플리케이션들을 개발할 때, 당신은 거의 항상 org.hibernate.SQL 카테고리에 대해 debug 모드로 설정하거나 hibernate.show_sql 프로퍼티를 이용가능하게 하여 작업해야 할 것이다.
org.hibernate.cfg.NamingStrategy 인터페이스는 데이터베이스 객체들과 스키마 요소들에 대한 "네이밍 표준"을 지정하는 것을 당신에게 허용해준다.
당신은 Java 식별자들로부터 데이터베이스 식별자들을 자동적으로 생성시키거나 매핑 파일에 주어진 "논리적" 컬럼과 테이블 이름들을 "물리적" 테이블과 컬럼 이름들로 자동적으로 처리하는 규칙들을 제공할 수 있다. 이 특징은 반복되는 노이즈(예를 들어 TBL_접두어들)를 제거함으로써, 매핑 문서의 말많은 장황함을 감소시키도록 도와준다. Hibernate에 의해 사용되는 디폴트 방도는 아주 작은 작품이다.
당신은 매핑들을 추가하기 이전에 Configuration.setNamingStrategy()를 호출함으로써 다른 방도를 지정할 수 있다:
SessionFactory sf = new Configuration()
.setNamingStrategy(ImprovedNamingStrategy.INSTANCE)
.addFile("Item.hbm.xml")
.addFile("Bid.hbm.xml")
.buildSessionFactory();
org.hibernate.cfg.ImprovedNamingStrategy 는 어떤 어플리케이션들에 대한 유용한 시작점일 수 있는 미리 빌드된 방도이다.
구성에 대한 다른 접근법은 hibernate.cfg.xml로 명명된 파일 속에 전체 구성을 지정하는 것이다. 이 파일은 hibernate.properties 파일에 대한 대용물로서 사용될 수 있거나, 만일 둘 다 존재할 경우에 프로퍼티들을 중복정의하는데 사용될 수 있다.
XML 구성 파일은 디폴트로 당신의 CLASSPATH의 루트에 존재하는 것이 기대된다. 다음은 예제이다:
<?xml version='1.0' encoding='utf-8'?>
<!DOCTYPE hibernate-configuration PUBLIC
"-//Hibernate/Hibernate Configuration DTD//EN"
"http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">
<hibernate-configuration>
<!-- a SessionFactory instance listed as /jndi/name -->
<session-factory
name="java:hibernate/SessionFactory">
<!-- properties -->
<property name="connection.datasource">java:/comp/env/jdbc/MyDB</property>
<property name="dialect">org.hibernate.dialect.MySQLDialect</property>
<property name="show_sql">false</property>
<property name="transaction.factory_class">
org.hibernate.transaction.JTATransactionFactory
</property>
<property name="jta.UserTransaction">java:comp/UserTransaction</property>
<!-- mapping files -->
<mapping resource="org/hibernate/auction/Item.hbm.xml"/>
<mapping resource="org/hibernate/auction/Bid.hbm.xml"/>
<!-- cache settings -->
<class-cache class="org.hibernate.auction.Item" usage="read-write"/>
<class-cache class="org.hibernate.auction.Bid" usage="read-only"/>
<collection-cache class="org.hibernate.auction.Item.bids" usage="read-write"/>
</session-factory>
</hibernate-configuration>
당신이 볼 수 있듯이, 이 접근법의 장점은 구성에 대한 매핑 파일 이름들을 구체화 시키는 것이다. hibernate.cfg.xml은 또한 당신이 Hibernate 캐시를 튜닝해야할 때 보다 편리하다. hibernate.properties 또는 hibernate.cfg.xml 어느 것을 사용하는가는 당신의 선택이다. XML 구문을 사용하는 위에 언급된 이점들을 제외하면 둘다 같은 것임을 노트하라.
Hibernate 구성을 사용할 경우, Hibernate 사용은 다음과 같이 간단하다
SessionFactory sf = new Configuration().configure().buildSessionFactory();
당신은 다음을 사용하여 다른 XML 구성 파일을 찾아낼 수 있다
SessionFactory sf = new Configuration()
.configure("catdb.cfg.xml")
.buildSessionFactory();
예를 들어 Hibernate는 영속 콜렉션-값을 가진 필드들이 인터페이스 타입으로서 선언되는 것을 필요로 한다:
public class Product {
private String serialNumber;
private Set parts = new HashSet();
public Set getParts() { return parts; }
void setParts(Set parts) { this.parts = parts; }
public String getSerialNumber() { return serialNumber; }
void setSerialNumber(String sn) { serialNumber = sn; }
}
실제 인터페이스는 java.util.Set, java.util.Collection, java.util.List, java.util.Map, java.util.SortedSet, java.util.SortedMap 또는 당신이 좋아하는 어떤 것일 수 있다!(여기서 "당신이 좋아하는 어떤 것"은 당신이 org.hibernate.usertype.UserCollectionType에 대한 구현을 작성해야 함을 의미한다.)
우리가 HashSet의 인스턴스를 가진 인스턴스 변수를 초기화 시켰던 방법을 주목하라. 이것은 새로이 초기화 된(비-영속) 인스턴스들을 가진 콜렉션 값 프로퍼티들을 초기화 시키는 최선의 방법이다. 당신이 -예를 들어 persist()를 호출하여- 인스턴스를 영속화 시킬 때 Hibernate는 실제로 HashSet을 Hibernate 자신의 Set 구현의 인스턴스로 대체시킬 것이다. 다음과 같은 오류들을 관찰하라:
Cat cat = new DomesticCat();
Cat kitten = new DomesticCat();
....
Set kittens = new HashSet();
kittens.add(kitten);
cat.setKittens(kittens);
session.persist(cat);
kittens = cat.getKittens(); // 좋다, kittens 콜렉션은 Set이다
(HashSet) cat.getKittens(); //오류!
Hibernate에 의해 도입된 영속 콜렉션들은 인터페이스 타입에 따라 HashMap, HashSet, TreeMap, TreeSet 또는 ArrayList과 같이 행위한다.
콜렉션 인스턴스들은 value 타입들을 가진 통상의 특징을 갖는다. 그것들은 영속 객체에 의해 참조될 때 자동적으로 영속화 되고 참조 해제될 때 자동적으로 삭제된다. 만일 하나의 콜렉션이 하나의 영속 객체로부터 또 다른 영속 객체로 전달될 때, 그것의 요소들은 하나의 테이블로부터 다른 테이블로 이동될 수 있다. 두 개의 엔티티들은 동일한 콜렉션 인스턴스에 대한 참조를 공유하지 않는다. 기본 관계형 모형 때문에 콜렉션 값 프로퍼티들은 null 값 의미들을 지원하지 않는다; Hibernate는 null 콜렉션 참조와 공백의 콜렉션 사이를 구별 짓지 않는다.
당신은 이것의 어떤 것에 대해 너무 많이 걱정하지 않아도 될 것이다. 당신이 통상의 자바 콜렉션들을 사용하는 것과 동일한 방법으로 영속 콜렉션들을 사용하라. 단지 당신이 양방향 연관관계들에 대한 의미를 확실히 이해하도록 하라(나중에 논의됨).
콜렉션을 매핑하는데 사용되는 Hiberante 매핑 요소는 인터페이스의 타입에 의존한다. 예를 들어 <set> 요소는 Set 타입의 매핑 프로퍼티들에 사용된다.
<class name="Product">
<id name="serialNumber" column="productSerialNumber"/>
<set name="parts">
<key column="productSerialNumber" not-null="true"/>
<one-to-many class="Part"/>
</set>
</class>
<set>과는 별도로, 또한 <list>, <map>, <bag>, <array>, <primitive-array> 매핑 요소들이 존재한다. <map> 요소가 대표적이다:
<map
name="propertyName" (1)
table="table_name" (2)
schema="schema_name" (3)
lazy="true|false" (4)
inverse="true|false" (5)
cascade="all|none|save-update|delete|all-delete-orphan" (6)
sort="unsorted|natural|comparatorClass" (7)
order-by="column_name asc|desc" (8)
where="arbitrary sql where condition" (9)
fetch="join|select|subselect" (10)
batch-size="N" (11)
access="field|property|ClassName" (12)
optimistic-lock="true|false" (13)
>
<key .... />
<map-key .... />
<element .... />
</map>
(1) |
name 콜렉션 프로퍼티 이름 |
(2) |
table (옵션-디폴트는 프로퍼티 이름) 콜렉션 테이블의 이름(one-to-many 연관관계들에 대해서는 사용되지 않음) |
(3) |
schema (옵션) 루트 요소 상에 선언된 스키마를 오버라이드 시키는 테이블 스키마의 이름 |
(4) |
lazy (옵션-디폴트는 true) lazy 초기화를 가능하게 한다 (배열들에 대해서는 이용 불가능함) |
(5) |
inverse (옵션-디폴트는 false) 이 콜렉션을 양방향 연관관계의 "inverse" 끝(end)으로 표시한다 |
(6) |
cascade (옵션-디폴트는 none) 오퍼레이션들이 자식 엔티티들에 대해 케스케이드하는 것을 이용 가능하게 한다 |
(7) |
sort (옵션) natural 정렬 순서로 정렬된 콜렉션 또는 주어진 comparator 클래스를 지정한다 |
(8) |
order-by (옵션, JDK1.4 에서만) asc 또는 desc 옵션과 함께 Map, Set 또는 bag의 반복 순서를 정의하는 테이블 컬럼(또는 컬럼들)을 지정한다 |
(9) |
where (옵션) 콜렉션을 검색하거나 제거할 때 사용될 임의적인 SQL WHERE 조건을 지정한다(콜렉션이 오직 이용 가능한 데이터의 부분집합 만을 포함할 경우에 유용하다) specify an arbitrary SQL WHERE condition to be used when retrieving or removing the collection (useful if the collection should contain only a subset of the available data) |
(10) |
fetch (옵션,디폴트는 select) outer-join 페칭과 순차적인 select에 의한 페칭, 그리고 순차적인 subselect 페칭에 의한 페칭 사이에서 선택하라. 오직 하나의 콜렉션 만이 SQL SELECT에 대한 outer join에 의해 페치될 수 있다. |
(11) |
batch-size (옵션,디폴트는 1) 이 콜렉션의 lazily fetching 인스턴스에 대해 "배치 사이즈"를 지정하라. |
(12) |
access (옵션-디폴트는 property): Hibernate가 프로퍼티 값에 접근하는데 사용할 방도. |
(13) |
optimistic-lock (옵션-디폴트는 true): 콜렉션의 상태에 대한 변경들이 소유하는 엔티티의 버전의 증가로 귀결될 것인지를 지정한다. (one to many 연관들에 대해, 이 설정을 사용 불가능하게 하는 것이 종종 합당하다.) |
콜렉션 인스턴스들은 그 콜렉션을 소유하는 엔티티의 foreign 키에 의해 데이터베이스 내에서 구별지워진다. 이 foreign 키는 그 콜렉션 테이블의 콜렉션 키 컬럼 (또는 컬럼들)로서 참조된다. 그 콜렉션 키 키러럼은 <key> 요소에 의해 매핑된다.
foreign 키 컬럼에 대한 null 허용 가능 컨스트레인트가 존재할 수 있다. 대부분의 콜렉션들에 대해, 이것이 당연히 수반된다. 단방향 one to many 연관들의 경우, foreign 키는 디폴트로 null 허용 가능하여서, 당신은 not-null="true"를 지정할 필요가 있을 수 있다.
<key column="productSerialNumber" not-null="true"/>
foreign 키 컨스트레인트는 ON DELETE CASCADE를 사용할 수도 있다.
<key column="productSerialNumber" on-delete="cascade"/>
<key> 요소에 대한 전체 정의는 앞 장을 보라.
콜렉션들은 모든 기본 타입들, 컴포넌트들, 그리고 물론 다른 엔티티들에 대한 참조들을 포함하여 거의 대부분의 어떤 다른 Hibernate 타입을 포함할 수도 있다. 이것은 중요한 구분이다: 콜렉션 내에 있는 객체는 "값(value)" 의미로 처리될 수도 있거나(그것의 생명주기는 콜렉션 소유자에 의존한다) 그것은 그것 자신의 생명주기를 가진 또 다른 엔티티에 대한 참조일 수 있다. 후자의 경우, 두 개의 객체들 사이의 "링크" 만이 그 콜렉션에 의해 소유된 상태로 간주된다.
포함된 타입은 콜렉션 요소 타입으로서 불려진다. 콜렉션 요소들은 <element> 또는 <composite-element>에 의해 매핑되거나, 엔티티 참조들의 경우에 <one-to-many> 또는 <many-to-many>로서 매핑된다. 처음 두 개는 요소들을 value 의미로 매핑시키고 다음 두개는 엔티티 연관들을 매핑하는데 사용된다.
set 과 bag 의미들을 가진 것들을 제외하면 모든 콜렉션 매핑들은 콜렉션 테이블 내에 인덱스 컬럼-배열 인덱스, 또는 List 인덱스 또는 Map 키로 매핑시키는 컬럼-을 필요로 한다. Map의 인덱스는 <map-key>로 매핑된 어떤 기본 타입일 수 있고, 그것은 <map-key-many-to-many>로 매핑된 엔티티 참조일 수 있거나, 그것은 <composite-map-key>로 매핑된 composite 타입일 수 있다. 배열 또는 리스트의 인덱스는 항상 integer 타입이고 <list-index> 요소를 사용하여 매핑된다. 매핑된 컬럼은 순차적인 정수들을 포함한다(디폴트로 0에서 시작하는 번호가 붙여짐).
<list-index
column="column_name" (1)
base="0|1|..."/>
(1) |
column_name (필수): 콜렉션 인덱스 값들을 보관하는 컬럼의 이름. |
(1) |
base (옵션 디폴트는 0): 리스트 또는 배열의 첫 번째 요소에 대응하는 인덱스 컬럼의 값. |
<map-key
column="column_name" (1)
formula="any SQL expression" (2)
type="type_name" (3)
length="N"/>
(1) |
column (옵션): 콜렉션 인덱스 값들을 보관하는 컬럼의 이름. |
(2) |
formula (옵션): map의 키를 평가하는데 사용되는 SQL formula. |
(3) |
type (옵션, 디폴트는 integer): 콜렉션 인덱스의 타입. |
<map-key-many-to-many
column="column_name" (1)
formula="any SQL expression" (2)
class="ClassName" (3)
/>
(1) |
column (옵션): 콜렉션 인덱스 값들에 대한 foreign 키 컬럼의 이름. |
(2) |
formula (옵션): map의 foreign 키를 평가하는데 사용되는 SQL formula. |
(3) |
class (필수): 콜렉션 인덱스로서 사용되는 엔티티 클래스. |
만일 당신의 테이블이 인덱스 컬럼을 가지 않고, 당신이 여전히 프로퍼티 타입으로서 List 를 사용하고자 원할 경우, 당신은 그 프로퍼티를 Hibernate <bag>으로서 매핑해야 한다. bag이 데이터베이스로부터 검색될 때 그것은 그것의 순서를 보유하지 않지만, 그것은 선택적으로 정렬(sorting)되거나 ordering될 수도 있다.
많은 공통된 관계형 모형들을 다루는, 콜렉션들에 대해 생성될 수 있는 매핑들의 영역이 꽤 존재한다. 여러가지 매핑 선언들이 데이터베이스 테이블들로 변환되는 방법을 당신이 느끼려면 스키마 생성으로 실험하기를 우리는 제안한다.
6.2.4. value들을 가진 콜렉션들과 many-to-many 연관들
어떤 값들을 가진 콜렉션과 many-to-many 연관은 foreign 키 컬럼이나 컬럼들, 콜렉션 요소 컬럼이나 컬럼들 그리고 가능하면 인덱스 컬럼들이나 컬럼들을 가진 전용 콜렉션 테이블을 필요로 한다.
값들을 가진 콜렉션의 경우, 우리는 <element> 태그를 사용한다.
<element
column="column_name" (1)
formula="any SQL expression" (2)
type="typename" (3)
length="N"
precision="N"
scale="N"
not-null="true|false"
unique="true|false"
/>
(1) |
column (옵션): 콜렉션 요소 값들을 보관하는 컬럼의 이름. |
(2) |
formula (옵션): 요소를 평가하는데 사용되는 SQL formula. |
(3) |
type (필수): 콜렉션 요소의 타입. |
many-to-many 연관은 <many-to-many> 요소를 사용하여 지정된다.
<many-to-many
column="column_name" (1)
formula="any SQL expression" (2)
class="ClassName" (3)
fetch="select|join" (4)
unique="true|false" (5)
not-found="ignore|exception" (6)
entity-name="EntityName"
/>
(1) |
column (옵션): 요소 foreign 키 컬럼의 이름. |
(2) |
formula (옵션): 요소 foreign 키 값을 평가하는데 사용되는 SQL formula. |
(3) |
class (필수): 연관된 클래스의 이름. |
(4) |
fetch (옵션- 디폴트는 join): 이 연관에 대해 outer-join 페칭 또는 sequential select 페칭을 이용 가능하게 만든다. 이것은 특별한 경우이다; 엔티티 그리고 다른 엔티티들에 대한 그것의 many-to-many 관계들에 대한 (하나의 SELECT 내에서) 전체 eager 페칭의 경우, 당신은 콜렉션 그 자체에 대해서 뿐만 아니라 내포된 요소 <many-to-many> 상의 이 속성에 대해 join 페칭을 이용 가능하게 할 것이다. |
(5) |
unique (옵션): foreign-key 컬럼에 대한 유일 컨스트레인트의 DDL 생성을 가능하도록 한다. 이것은 연관 다중성(association multiplicity)을 효과적으로 one to many로 만든다. |
(6) |
not-found (옵션- 디폴트는 exception): 누락된 행들을 참조하는 foreign key들이 어떻게 처리될 것인지를 지정한다: ignore는 한 개의 누락된 행을 하나의 null 연관으로 처리할 것이다. |
몇몇 예제들, 먼저 문자열들을 가진 set:
<set name="names" table="person_names">
<key column="person_id"/>
<element column="person_name" type="string"/>
</set>
(order-by 속성에 의해 결정되는 반복 순서를 가진) 정수들을 포함하는 bag :
<bag name="sizes"
table="item_sizes"
order-by="size asc">
<key column="item_id"/>
<element column="size" type="integer"/>
</bag>
엔티티들을 가진 배열 - 이 경우에, many to many 연관 :
<array name="addresses"
table="PersonAddress"
cascade="create">
<key column="personId"/>
<list-index column="sortOrder"/>
<many-to-many column="addressId" class="Address"/>
</array>
date들에 대한 문자열 인덱스들을 가진 map :
<map name="holidays"
table="holidays"
schema="dbo"
order-by="hol_name asc">
<key column="id"/>
<map-key column="hol_name" type="string"/>
<element column="hol_date" type="date"/>
</map>
컴포넌트들의 리스트(다음 장에서 논의됨):
<list name="carComponents"
table="CarComponents">
<key column="carId"/>
<list-index column="sortOrder"/>
<composite-element class="CarComponent">
<property name="price"/>
<property name="type"/>
<property name="serialNumber" column="serialNum"/>
</composite-element>
</list>
one to many 연관은 중재하는 콜렉션 테이블 없이 foreign 키를 통해 두 개의 클래스들의 테이블들을 연결시킨다. 이 매핑은 통상의 자바 콜렉션들의 어떤 의미를 상실한다:
- 포함된 엔티티 클래스의 인스턴스는 그 콜렉션의 하나 이상의 인스턴스에 속하지 않는다
- 포함된 엔티티 클래스의 인스턴스는 콜렉션 인덱스의 하나 이상의 값에서 나타나지 않는다
Product로부터 Part로의 연관은 foreign 키 컬럼과 Part 테이블에 대한 인덱스 컬럼의 존재를 필요로 한다. <one-to-many> 태그는 이것이 one to many 연관임을 나타낸다.
<one-to-many
class="ClassName" (1)
not-found="ignore|exception" (2)
entity-name="EntityName"/>
(1) |
class (필수): 연관된 클래스의 이름. |
(2) |
not-found (옵션- 디폴트는 exception): 누락된 행들을 참조하는 foreign key들이 어떻게 처리될 것인지를 지정한다: ignore는 한 개의 누락된 행을 하나의 null 연관으로 처리할 것이다. |
<one-to-many> 요소는 어떤 컬럼들을 선언하는데 필요하지 않음을 주목하라. 어딘가에 table 이름을 지정하는 것도 필수적이지 않다.
매우 중요한 노트: 만일 <one-to-many> 연관의 foreign 키 컬럼이 NOT NULL로 선언될 경우, 당신은 <key> 매핑 not-null="true"를 선언해야 하거나 inverse="true"로 마크된 콜렉션 매핑을 가진 양방향 연관을 사용해야 한다. 양방향 연관들에 대한 논의는 이 장의 뒷부분을 보라.
이 예제는 name으로 Part 엔티티들을 가진 map을 보여준다(여기서 partName은 Part의 영속 프로퍼티이다). formula-기반 index의 사용을 주목하라.
<map name="parts"
cascade="all">
<key column="productId" not-null="true"/>
<map-key formula="partName"/>
<one-to-many class="Part"/>
</map>
Hibernate는 java.util.SortedMap과 java.util.SortedSet를 구현하는 콜렉션들을 지원한다. 당시은 이 매핑 파일 속에 하나의 comparator를 지정해야 한다:
<set name="aliases"
table="person_aliases"
sort="natural">
<key column="person"/>
<element column="name" type="string"/>
</set>
<map name="holidays" sort="my.custom.HolidayComparator">
<key column="year_id"/>
<map-key column="hol_name" type="string"/>
<element column="hol_date" type="date"/>
</map>
sort 속성에 허용되는 값들은 unsorted, natural 그리고 java.util.Comparator를 구현하는 클래스의 이름이다.
Sorted 콜렉션들은 java.util.TreeSet 또는 java.util.TreeMap처럼 행동한다.
만일 당신이 데이터베이스 그 자체가 콜렉션 요소들을 순서지움(ordering)하기 원할 경우 set, bag, map 매핑들의 속성을 사용하라. 이 해결책은 JDK 1.4 이상의 버전에서만 이용 가능하다(그것은 LinkedHashSet 또는 LinkedHashMap을 사용하여 구현된다). 이것은 메모리 내에사가 아닌, SQL 질의 내에서 순서지움(ordering)을 수행한다.
<set name="aliases" table="person_aliases" order-by="lower(name) asc">
<key column="person"/>
<element column="name" type="string"/>
</set>
<map name="holidays" order-by="hol_date, hol_name">
<key column="year_id"/>
<map-key column="hol_name" type="string"/>
<element column="hol_date type="date"/>
</map>
order-by 속성의 값은 HQL 순서지움(ordering)이 아니라 SQL 순서지움(ordering)임을 노트하라!
연관들은 콜렉션 filter()를 사용하여 실행 시에 어떤 임의의 criteria(기준)을 사용하여 정렬(sort)될 수도 있다.
sortedUsers = s.createFilter( group.getUsers(), "order by this.name" ).list();
양방향 연관은 연관의 양 "끝(end)들"로부터 네비게이션을 허용한다. 두 가지 종류의 양방향 연관들이 지원된다:
- one-to-many
- 한쪽 끝에 set 또는 bag 값을 갖고, 다른 쪽 긑에 단일 값을 가진 연관
- many-to-many
- 양 끝에서 set 또는 bag 값을 가진 연관
당신은 동일한 두 개의 many-to-many 연관들을 데이터베이스 테이블로 간단하게 매핑함으로써 그리고 한 쪽 끝을 inverse(당신의 선택은 하나이지만, 그것은 인덱싱된 콜렉션일 수 없다)로 선언함으로써 하나의 many-to-many 양방향 연관을 지정할 수도 있다.
다음은 양방향 many-to-many 연관에 관한 예제이다; 각각의 카테고리는 많은 아이템들을 가질 수 있고 각각의 아이템은 많은 카테고리들 속에 있을 수 있다:
<class name="Category">
<id name="id" column="CATEGORY_ID"/>
...
<bag name="items" table="CATEGORY_ITEM">
<key column="CATEGORY_ID"/>
<many-to-many class="Item" column="ITEM_ID"/>
</bag>
</class>
<class name="Item">
<id name="id" column="CATEGORY_ID"/>
...
<!-- inverse end -->
<bag name="categories" table="CATEGORY_ITEM" inverse="true">
<key column="ITEM_ID"/>
<many-to-many class="Category" column="CATEGORY_ID"/>
</bag>
</class>
연관의 inverse end에 대해서만 행해진 변경들은 영속화 되지 않는다. 이것은 Hibernate가 모든 양방향 연관에 대해 메모리 내에 두 개의 표상들을 갖는다는 점을 의미한다: A로부터 B로의 하나의 링크와 B로부터 A로의 하나의 링크. 만일 당신이 자바 객체 모형에 대해 그리고 자바에서 many-to-many 관계를 생성시키는 방법에 대해 생각하면 이것은 이해하기가 더 쉽다:
category.getItems().add(item); // category는 이제 관계에 대해 "안다"
item.getCategories().add(category); // item은 이제 관계에 대한 "안다"
session.persist(item); // 관계는 저장되지 않을 것이다!
session.persist(category); // 관계가 저장될 것이다.
non-inverse 측은 메모리 내 표상을 데이터베이스로 저장하는데 사용된다.
당신은 동일한 테이블 컬럼(들)에 대한 one-to-many 연관을 many-to-one 연관으로서 매핑하고 many-값을 가진 끝(end)을 inverse="true"로 선언함으로써 양방향 연관을 정의할 수도 있다.
<class name="Parent">
<id name="id" column="parent_id"/>
....
<set name="children" inverse="true">
<key column="parent_id"/>
<one-to-many class="Child"/>
</set>
</class>
<class name="eg.Child">
<id name="id" column="id"/>
....
<many-to-one name="parent"
class="Parent"
column="parent_id"
not-null="true"/>
</class>
연관의 한쪽 끝을 inverse="true"로 매핑하는 것은 cascade들을 가진 오퍼레이션에 영향을 주지 않고, orthogonal 개념들이다!
6.3.3. Ternary associations(세겹 연관들)
세 겹의 연관을 매핑하는 세 가지 가능한 접근법들이 존재한다. 하나의 접근법은 그것의 인덱스로서 연관관계를 가진 Map을 사용하는 것이다:
<map name="contracts">
<key column="employer_id" not-null="true"/>
<map-key-many-to-many column="employee_id" class="Employee"/>
<one-to-many class="Contract"/>
</map>
<map name="connections">
<key column="incoming_node_id"/>
<map-key-many-to-many column="outgoing_node_id" class="Node"/>
<many-to-many column="connection_id" class="Connection"/>
</map>
두 번째 접근법은 그 연관을 엔티티 클래스로서 단순하게 리모델링 하는 것이다. 이것은 우리가 가장 공통적으로 사용하는 접근법이다.
마지막 대안은 우리가 나중에 논의하게 될 composite 요소들을 사용하는 것이다.
만일 당신이 composite 키가 나쁜 것이고 그 엔티티들이 합성 식별자들(대용 키들, surrogate keys)을 가져야 한다는 우리의 견해를 전적으로 수용할 경우, 당신은 우리가 지금까지 보았던 값들을 가진 콜렉션들과 many to many 연관들이 composite 키들을 가진 테이블들로 모두 매핑된다는 약간 이상한 점을 발견할 수도 있다! 이제 이 점은 꽤 논의의 여지가 있다; 순수한 연관 테이블은 (비록 composite 값들을 가진 콜렉션일 수 있을지라도) 대용 키로부터 많은 이점을 취하지 않는 것처럼 보인다. 그럼에도 불구하고 Hibernate는 당신이 many to many 연관들과 값들을 가진 콜렉션들을 대용 키를 가진 테이블로 매핑시키는 것을 당신에게 허용해주는 특징을 제공한다.
<idbag> 요소는 bag 의미를 가진 List (또는 Collection)을 매핑하도록 당신에게 허용해준다.
<idbag name="lovers" table="LOVERS">
<collection-id column="ID" type="long">
<generator class="sequence"/>
</collection-id>
<key column="PERSON1"/>
<many-to-many column="PERSON2" class="eg.Person" outer-join="true"/>
</idbag>
당신이 볼 수 있듯이, <idbag>은 마치 엔티티 클래스인양 synthetic id generator(합성 id 생성자)를 갖는다! 다른 대용 키는 각각의 콜렉션 행에 할당된다. 하지만 Hibernate는 특정 행의 대용 키 값을 발견하는 메커니즘을 제공하지 않는다.
<idbag>의 업데이트 퍼포먼스는 정규 <bag> 보다 훨씬 좋음을 노트하라! Hibernate는 마치 list, map, 또는 set인양, 개별 행들을 효율적으로 위치지울 수 있고 그것들을 개별적으로 업데이트 하거나 삭제시킬 수 있다.
현재 구현에서, native identifier 생성 방도는 <idbag> 콜렉션 식별자들에 대해 지원되지 않는다.
앞의 절들은 꽤 혼동스럽다. 따라서 예제를 살펴보자. 다음 클래스:
package eg;
import java.util.Set;
public class Parent {
private long id;
private Set children;
public long getId() { return id; }
private void setId(long id) { this.id=id; }
private Set getChildren() { return children; }
private void setChildren(Set children) { this.children=children; }
....
....
}
는 Child 인스턴스들을 가진 하나의 콜렉션을 갖고 있다. 만일 각각의 자식이 최소한 한 개의 부모를 가질 경우, 대부분의 고유한 매핑은 one-to-many 연관이다:
<hibernate-mapping>
<class name="Parent">
<id name="id">
<generator class="sequence"/>
</id>
<set name="children">
<key column="parent_id"/>
<one-to-many class="Child"/>
</set>
</class>
<class name="Child">
<id name="id">
<generator class="sequence"/>
</id>
<property name="name"/>
</class>
</hibernate-mapping>
이것은 다음 테이블 정의들로 매핑된다: :
create table parent ( id bigint not null primary key )
create table child ( id bigint not null primary key, name varchar(255), parent_id bigint )
alter table child add constraint childfk0 (parent_id) references parent
만일 부모가 필수적이라면, 양방향 one-to-many 연관관계를 사용하라:
<hibernate-mapping>
<class name="Parent">
<id name="id">
<generator class="sequence"/>
</id>
<set name="children" inverse="true">
<key column="parent_id"/>
<one-to-many class="Child"/>
</set>
</class>
<class name="Child">
<id name="id">
<generator class="sequence"/>
</id>
<property name="name"/>
<many-to-one name="parent" class="Parent" column="parent_id" not-null="true"/>
</class>
</hibernate-mapping>
NOT NULL 컨스트레인트를 주목하라:
create table parent ( id bigint not null primary key )
create table child ( id bigint not null
primary key,
name varchar(255),
parent_id bigint not null )
alter table child add constraint childfk0 (parent_id) references parent
대안적으로, 만일 당신이 이 연관관계가 단방향이어야 함을 절대적으로 역설할 경우, 당신은 <key> 매핑 상에 NOT NULL 컨스트레인트를 선언할 수 있다:
<hibernate-mapping>
<class name="Parent">
<id name="id">
<generator class="sequence"/>
</id>
<set name="children">
<key column="parent_id" not-null="true"/>
<one-to-many class="Child"/>
</set>
</class>
<class name="Child">
<id name="id">
<generator class="sequence"/>
</id>
<property name="name"/>
</class>
</hibernate-mapping>
반면에, 만일 자식이 여러 부모들을 가질 수 있을 경우, many-to-many 연관이 적절하다:
<hibernate-mapping>
<class name="Parent">
<id name="id">
<generator class="sequence"/>
</id>
<set name="children" table="childset">
<key column="parent_id"/>
<many-to-many class="Child" column="child_id"/>
</set>
</class>
<class name="Child">
<id name="id">
<generator class="sequence"/>
</id>
<property name="name"/>
</class>
</hibernate-mapping>
테이블 정의들:
create table parent ( id bigint not null primary key )
create table child ( id bigint not null primary key, name varchar(255) )
create table childset ( parent_id bigint not null,
child_id bigint not null,
primary key ( parent_id, child_id ) )
alter table childset add constraint childsetfk0 (parent_id) references parent
alter table childset add constraint childsetfk1 (child_id) references child
부모/자식 관계 매핑을 연습하는 더 많은 예제들과 전체 리허설은 21 장, 예제: Parent/Child를 보라.
비록 더 많은 신종 연관 매핑들이 가능할지라도, 우리는 다음 장에서 모든 가능성들을 분류할 것이다.
연관 매핑들은 올바른 것을 얻기가 흔히 가장 어려운 것이다. 이 절에서 우리는 단방향 매핑들에서 시작하고 양방향 매핑들을 검토함으로써 하나씩 표준적인 경우들을 상세히 논의할 것이다. 우리는 모든 예제들에서 Person 과 Address를 사용할 것이다.
우리는 그것들을 중재하는 join 테이블로 매핑할 것인지 여부, 그리고 multiplicity(다중성)에 따라 연관들을 분류할 것이다.
null 허용 가능한 foreign 키들은 전통적인 데이터 모델링에서 좋은 실례로 간주되지 않아서, 모든 우리의 예제들은 not null foreign 키들을 사용한다. 이것은 Hibernate에서 필수가 아니고, 매핑들은 당신이 null 허용 가능 컨스트레인트들을 드롭시킬 경우에도 모두 동작할 것이다.
단방향 many-to-one 연관은 가장 공통적인 종류의 단방향 연관이다.
<class name="Person">
<id name="id" column="personId">
<generator class="native"/>
</id>
<many-to-one name="address"
column="addressId"
not-null="true"/>
</class>
<class name="Address">
<id name="id" column="addressId">
<generator class="native"/>
</id>
</class>
create table Person ( personId bigint not null primary key, addressId bigint not null )
create table Address ( addressId bigint not null primary key )
foreign 키에 대한 단방향 one-to-one 연관은 대개 아주 동일하이다. 유일한 차이점은 컬럼이 유일(unique) 컨스트레인트라는 점이다.
<class name="Person">
<id name="id" column="personId">
<generator class="native"/>
</id>
<many-to-one name="address"
column="addressId"
unique="true"
not-null="true"/>
</class>
<class name="Address">
<id name="id" column="addressId">
<generator class="native"/>
</id>
</class>
create table Person ( personId bigint not null primary key, addressId bigint not null unique )
create table Address ( addressId bigint not null primary key )
프라이머리 키에 대한 단방향 one-to-one 연관은 대개 특별한 id generator를 사용한다. (이 예제에서 연관의 방향이 역전되었음을 주목하라.)
<class name="Person">
<id name="id" column="personId">
<generator class="native"/>
</id>
</class>
<class name="Address">
<id name="id" column="personId">
<generator class="foreign">
<param name="property">person</param>
</generator>
</id>
<one-to-one name="person" constrained="true"/>
</class>
create table Person ( personId bigint not null primary key )
create table Address ( personId bigint not null primary key )
foreign 에 대한 단방향 one-to-many 연관은 매우 통상적인 경우이고, 실제로 권장되지 않는다.
<class name="Person">
<id name="id" column="personId">
<generator class="native"/>
</id>
<set name="addresses">
<key column="personId"
not-null="true"/>
<one-to-many class="Address"/>
</set>
</class>
<class name="Address">
<id name="id" column="addressId">
<generator class="native"/>
</id>
</class>
create table Person ( personId bigint not null primary key )
create table Address ( addressId bigint not null primary key, personId bigint not null )
우리는 이런 종류의 연관에 대해 join 테이블을 사용하는 것이 더 좋다고 생각한다.
7.3. join 테이블들에 대한 단방향 연관들
join 테이블에 대한 단방향 one-to-many 연관이 훨씬 더 선호된다. unique="true"를 지정함으로써 우리는 many-to-many에서 one-to-many로 multiplicity를 변경시켰음을 주목하라.
<class name="Person">
<id name="id" column="personId">
<generator class="native"/>
</id>
<set name="addresses" table="PersonAddress">
<key column="personId"/>
<many-to-many column="addressId"
unique="true"
class="Address"/>
</set>
</class>
<class name="Address">
<id name="id" column="addressId">
<generator class="native"/>
</id>
</class>
create table Person ( personId bigint not null primary key )
create table PersonAddress ( personId not null, addressId bigint not null primary key )
create table Address ( addressId bigint not null primary key )
join 테이블에 대한 단방향 many-to-one 연관은 그 연관이 선택적일 때 매우 공통적이다.
<class name="Person">
<id name="id" column="personId">
<generator class="native"/>
</id>
<join table="PersonAddress"
optional="true">
<key column="personId" unique="true"/>
<many-to-one name="address"
column="addressId"
not-null="true"/>
</join>
</class>
<class name="Address">
<id name="id" column="addressId">
<generator class="native"/>
</id>
</class>
create table Person ( personId bigint not null primary key )
create table PersonAddress ( personId bigint not null primary key, addressId bigint not null )
create table Address ( addressId bigint not null primary key )
join 테이블에 대한 단방향 one-to-one 연관은 극히 통상적이지 않지만 가능하다.
<class name="Person">
<id name="id" column="personId">
<generator class="native"/>
</id>
<join table="PersonAddress"
optional="true">
<key column="personId"
unique="true"/>
<many-to-one name="address"
column="addressId"
not-null="true"
unique="true"/>
</join>
</class>
<class name="Address">
<id name="id" column="addressId">
<generator class="native"/>
</id>
</class>
create table Person ( personId bigint not null primary key )
create table PersonAddress ( personId bigint not null primary key, addressId bigint not null unique )
create table Address ( addressId bigint not null primary key )
마지막으로, 우리는 단방향 many-to-many 연관을 갖는다.
<class name="Person">
<id name="id" column="personId">
<generator class="native"/>
</id>
<set name="addresses" table="PersonAddress">
<key column="personId"/>
<many-to-many column="addressId"
class="Address"/>
</set>
</class>
<class name="Address">
<id name="id" column="addressId">
<generator class="native"/>
</id>
</class>
create table Person ( personId bigint not null primary key )
create table PersonAddress ( personId bigint not null, addressId bigint not null,
primary key (personId, addressId) )
create table Address ( addressId bigint not null primary key )
7.4.1. one to many / many to one
양방향 many-to-one 연관은 가장 공통된 종류의 연관이다.(이것은 표준 부모/자식 관계이다. )
<class name="Person">
<id name="id" column="personId">
<generator class="native"/>
</id>
<many-to-one name="address"
column="addressId"
not-null="true"/>
</class>
<class name="Address">
<id name="id" column="addressId">
<generator class="native"/>
</id>
<set name="people" inverse="true">
<key column="addressId"/>
<one-to-many class="Person"/>
</set>
</class>
create table Person ( personId bigint not null primary key, addressId bigint not null )
create table Address ( addressId bigint not null primary key )
foreign에 대한 양방향 one-to-one 연관은 꽤 공통적이다.
<class name="Person">
<id name="id" column="personId">
<generator class="native"/>
</id>
<many-to-one name="address"
column="addressId"
unique="true"
not-null="true"/>
</class>
<class name="Address">
<id name="id" column="addressId">
<generator class="native"/>
</id>
<one-to-one name="person"
property-ref="address"/>
</class>
create table Person ( personId bigint not null primary key, addressId bigint not null unique )
create table Address ( addressId bigint not null primary key )
프라이머리 키에 대한 양방향 one-to-one 연관은 특별한 id generator를 사용한다.
<class name="Person">
<id name="id" column="personId">
<generator class="native"/>
</id>
<one-to-one name="address"/>
</class>
<class name="Address">
<id name="id" column="personId">
<generator class="foreign">
<param name="property">person</param>
</generator>
</id>
<one-to-one name="person"
constrained="true"/>
</class>
create table Person ( personId bigint not null primary key )
create table Address ( personId bigint not null primary key )
7.5. join 테이블들에 대한 양방향 연관들
7.5.1. one to many / many to one
join 테이블에 대한 양방향 one-to-many 연관. Note that the inverse="true"는 연관의 어느 쪽 끝이든 콜렉션 측으로 또는 join 측으로 갈 수 있다.
<class name="Person">
<id name="id" column="personId">
<generator class="native"/>
</id>
<set name="addresses"
table="PersonAddress">
<key column="personId"/>
<many-to-many column="addressId"
unique="true"
class="Address"/>
</set>
</class>
<class name="Address">
<id name="id" column="addressId">
<generator class="native"/>
</id>
<join table="PersonAddress"
inverse="true"
optional="true">
<key column="addressId"/>
<many-to-one name="person"
column="personId"
not-null="true"/>
</join>
</class>
create table Person ( personId bigint not null primary key )
create table PersonAddress ( personId bigint not null, addressId bigint not null primary key )
create table Address ( addressId bigint not null primary key )
join 테이블에 대한 양방향 one-to-one 연관은 극히 통상적이지 않지만, 가능하다.
<class name="Person">
<id name="id" column="personId">
<generator class="native"/>
</id>
<join table="PersonAddress"
optional="true">
<key column="personId"
unique="true"/>
<many-to-one name="address"
column="addressId"
not-null="true"
unique="true"/>
</join>
</class>
<class name="Address">
<id name="id" column="addressId">
<generator class="native"/>
</id>
<join table="PersonAddress"
optional="true"
inverse="true">
<key column="addressId"
unique="true"/>
<many-to-one name="address"
column="personId"
not-null="true"
unique="true"/>
</join>
</class>
create table Person ( personId bigint not null primary key )
create table PersonAddress ( personId bigint not null primary key,
addressId bigint not null unique )
create table Address ( addressId bigint not null primary key )
마지막으로, 우리는 양방향 many-to-many 연관을 갖는다.
<class name="Person">
<id name="id" column="personId">
<generator class="native"/>
</id>
<set name="addresses">
<key column="personId"/>
<many-to-many column="addressId"
class="Address"/>
</set>
</class>
<class name="Address">
<id name="id" column="addressId">
<generator class="native"/>
</id>
<set name="people" inverse="true">
<key column="addressId"/>
<many-to-many column="personId"
class="Person"/>
</set>
</class>
create table Person ( personId bigint not null primary key )
create table PersonAddress ( personId bigint not null,
addressId bigint not null,
primary key (personId, addressId) )
create table Address ( addressId bigint not null primary key )
Hibernate에서 component 의 개념은 다른 용도로 몇몇 다른 컨텍스트들 내에서 재사용된다.
컴포넌트는 엔티티 참조가 아닌, value 타입으로서 영속화 되는 포함된 객체이다. "컴포넌트" 용어는 (아키텍처 수준의 컴포넌트들이 아닌) composition(구성, 조립,합성)에 대한 객체-지향적인 개념을 언급한다. 예를 들어 당신은 다음과 같이 개인을 모형화 시킬 수도 있다:
public class Person {
private java.util.Date birthday;
private Name name;
private String key;
public String getKey() {
return key;
}
private void setKey(String key) {
this.key=key;
}
public java.util.Date getBirthday() {
return birthday;
}
public void setBirthday(java.util.Date birthday) {
this.birthday = birthday;
}
public Name getName() {
return name;
}
public void setName(Name name) {
this.name = name;
}
......
......
}
public class Name {
char initial;
String first;
String last;
public String getFirst() {
return first;
}
void setFirst(String first) {
this.first = first;
}
public String getLast() {
return last;
}
void setLast(String last) {
this.last = last;
}
public char getInitial() {
return initial;
}
void setInitial(char initial) {
this.initial = initial;
}
}
이제 Name은 Person의 컴포넌트로서 영속화 될 수도 있다. Name 이 그것의 영속 프로퍼티들에 대한 getter 메소드와 setter 메소드를 정의하지만, 어떤 인터페이스들이나 identifier 프로퍼티들을 선언하는 것을 필요로 하지 않음을 주목하라.
우리의 Hibernate 매핑은 다음과 같을 것이다:
<class name="eg.Person" table="person">
<id name="Key" column="pid" type="string">
<generator class="uuid.hex"/>
</id>
<property name="birthday" type="date"/>
<component name="Name" class="eg.Name"> <!-- class attribute optional -->
<property name="initial"/>
<property name="first"/>
<property name="last"/>
</component>
</class>
person 테이블은 pid, birthday, initial, first, last 컬럼들을 가질 것이다.
모든 value 타입들처럼, 컴포넌트들은 공유된 참조들을 지원하지 않는다. 달리 말해, 두 명의 개인들은 동일한 이름을 가질 수 있지만, 두 개의 person 객체들은 오직 값 만이 "동일한", 두 개의 독립적인 name 객체들을 포함할 것이다. 컴포넌트의 null 값 의미는 특별한 용도를 위한 것이다. 포함된 객체를 다시 로드시킬 때, Hibernate는 모든 컴포넌트 컬럼들이 null일 경우에 전체 컴포넌트가 null이라고 가정할 것이다. 이것은 대부분의 용도로는 맞을 것이다.
컴포넌트의 프로퍼티들은 임의의 Hibernate 타입일 수 있다(콜렉션들, many-to-one 연관들, 다른 컴포넌트들, 기타). 내포된 컴포넌트들은 신종의 사용례로 간주되지 않을 것이다. Hibernate는 매우 잘 정제된 객체 모형을 지원하도록 고안되어있다.
<component> 요소는 컴포넌트 클래스의 프로퍼티는 포함되는 엔티티에 대한 참조로서 매핑시키는 <parent> 서브요소를 허용한다.
<class name="eg.Person" table="person">
<id name="Key" column="pid" type="string">
<generator class="uuid.hex"/>
</id>
<property name="birthday" type="date"/>
<component name="Name" class="eg.Name" unique="true">
<parent name="namedPerson"/> <!-- reference back to the Person -->
<property name="initial"/>
<property name="first"/>
<property name="last"/>
</component>
</class>
컴포넌트들을 가진 콜렉션들이 지원된다(예를 들면 Name 타입을 가진 배열). <element> 태그를 <composite-element> 태그로 바꾸어서 당신의 컴포넌트 콜렉션을 선언하라.
<set name="someNames" table="some_names" lazy="true">
<key column="id"/>
<composite-element class="eg.Name"> <!-- class attribute required -->
<property name="initial"/>
<property name="first"/>
<property name="last"/>
</composite-element>
</set>
노트: 만일 당신이 composite 요소를 가진 Set를 정의할 경우, equals()와 hashCode()를 정확하게 구현하는 것이 매우 중요하다.
Composite 요소들은 컴포넌트들을 포함하지만 콜렉션들을 포함하지 않는다. 만일 당신의 composite 요소 자체가 컴포넌트들을 포함할 경우, <nested-composite-element> 태그를 사용하라. 이것은 꽤 신종의 경우-자체로 컴포넌트들을 갖고 있는 컴포넌트들을 가진 콜렉션-이다. 이 방도로 당신은 one-to-many 연관이 더 적절한지를 당신 스스로에게 질문하게 될 것이다. composite 요소를 다시 모델링하려고 시도하라 - 그러나 자바 모형들이 동일할지라도, 관계형 모형과 영속화 의미들은 여전히 약간 다르다.
당신이 <set>를 사용할 경우, composite 요소 매핑은 null 가능한 프로퍼티들을 지원하지 않음을 노트하길 바란다. Hibernate는 객체들을 삭제할 때 레코드를 식별하기 위해 각각의 컬럼들 값을 사용해야 하며, (composite 요소 테이블 내에 별도의 프라이머리 키 컬럼이 존재하지 않는다), 그것은 null 값들을 갖는 것이 불가능하다. 당신은 composite-요소 내에 오직 not-null 프로퍼티들을 해야 하거나 <list>, <map>, <bag> 또는 <idbag>을 선택해야 한다.
composite 요소의 특별하나 경우는 내포된 <many-to-one> 요소를 가진 composite 요소이다. 이같은 매핑은 many-to-many 연관 테이블의 특별한 컬럼들을 composite 요소 클래스로 매핑시키는 것을 당신에게 허용해준다. 다음은 Order로부터 Item으로의 many-to-many 연관이다. 여기서 purchaseDate, price, quantity 는 연관의 프로퍼티들이다:
<class name="eg.Order" .... >
....
<set name="purchasedItems" table="purchase_items" lazy="true">
<key column="order_id">
<composite-element class="eg.Purchase">
<property name="purchaseDate"/>
<property name="price"/>
<property name="quantity"/>
<many-to-one name="item" class="eg.Item"/> <!-- class 속성은 옵션이다 -->
</composite-element>
</set>
</class>
물론, 양방향 연관 네비게이션의 경우, 다른 측 상에 있는 purchase에 대한 참조가 참조가 존재할 수 없다. 컴포넌트들이 value 타입들이고 공유된 참조들을 허용하지 않음을 기억하라. 하나의 Purchase는 Order를 가진 set 내에 있을 수 있지만, 그것은 동시에 Item에 의해 참조될 수 없다.
심지어 세겹의(또는 네 겹의, 기타) 연관들이 가능하다:
<class name="eg.Order" .... >
....
<set name="purchasedItems" table="purchase_items" lazy="true">
<key column="order_id">
<composite-element class="eg.OrderLine">
<many-to-one name="purchaseDetails class="eg.Purchase"/>
<many-to-one name="item" class="eg.Item"/>
</composite-element>
</set>
</class>
composite 요소들은 다른 엔티티들에 대한 연관들과 동일한 구문을 사용하여 질의들 내에 나타날 수도 있다.
<composite-map-key> 요소는 당신이 컴포넌트 클래스를 Map의 키로서 매핑시키도록 한다. 당신은 컴포넌트 클래스 상에서 hashCode()와 equals()를 정확하게 오버라이드 시키도록 하라.
8.4. composite 식별자들로서 컴포넌트들
당신은 컴포넌트를 엔티티 클래스의 식별자로서 사용할 수도 있다. 당신의 컴포넌트 클래스는 어떤 사양들을 충족해야 한다:
- 그것은 java.io.Serializable을 구현해야 한다.
- 그것은 composite 키 등가(equality)에 대한 데이터베이스 개념과 일치되게 equals()와 hashCode()를 다시 구현해야 한다.
노트: Hibernate3에서, 두 번째 사양은 Hibernate의 절대적으로 엄격한 사양이 아니다. 그러나 아무튼 그것을 행하라.
당신은 compsite 키들을 생성시키는데 IdentifierGenerator를 사용할 수 없다. 대신에 어플리케이션은 그것 자신의 식별자들을 할당해야 한다.
통상의 <id> 선언 위치에 (내포된 <key-property> 요소들을 가진) <composite-id> 태그를 사용하라. 예를 들어, OrderLine 클래스는 Order의 (composite) 프라이머리 키에 의존하는 프라이머리 키를 갖는다.
<class name="OrderLine">
<composite-id name="id" class="OrderLineId">
<key-property name="lineId"/>
<key-property name="orderId"/>
<key-property name="customerId"/>
</composite-id>
<property name="name"/>
<many-to-one name="order" class="Order"
insert="false" update="false">
<column name="orderId"/>
<column name="customerId"/>
</many-to-one>
....
</class>
이제 OrderLine 테이블을 참조하는 임의의 foreign 키들이 또한 compsite이다. 당신은 다른 클래스들에 대한 당신의 매핑 속에 이것을 선언해야 한다. OrderLine 에 대한 연관은 다음과 같이 매핑될 것이다:
<many-to-one name="orderLine" class="OrderLine">
<!-- the "class" attribute is optional, as usual -->
<column name="lineId"/>
<column name="orderId"/>
<column name="customerId"/>
</many-to-one>
(<column> 태그가 모든 곳에서 column 속성에 대한 대안임을 노트하라.)
OrderLine에 대한 many-to-many 연관은 또한 composite foreign 키를 사용한다:
<set name="undeliveredOrderLines">
<key column name="warehouseId"/>
<many-to-many class="OrderLine">
<column name="lineId"/>
<column name="orderId"/>
<column name="customerId"/>
</many-to-many>
</set>
Order 에서 OrderLine들의 콜렉션이 사용될 것이다:
<set name="orderLines" inverse="true">
<key>
<column name="orderId"/>
<column name="customerId"/>
</key>
<one-to-many class="OrderLine"/>
</set>
(<one-to-many> 요소는 통상적으로 컬럼들을 선언하지 않는다.)
만일 OrderLine 자체가 하나의 콜렉션을 소유할 경우, 그것은 또한 하나의 composite foreign 키를 갖는다.
<class name="OrderLine">
....
....
<list name="deliveryAttempts">
<key> <!-- 콜렉션은 composite 키 타입을 상속받는다 -->
<column name="lineId"/>
<column name="orderId"/>
<column name="customerId"/>
</key>
<list-index column="attemptId" base="1"/>
<composite-element class="DeliveryAttempt">
...
</composite-element>
</set>
</class>
당신은 Map 타입의 프로퍼티를 매핑시킬 수도 있다:
<dynamic-component name="userAttributes">
<property name="foo" column="FOO"/>
<property name="bar" column="BAR"/>
<many-to-one name="baz" class="Baz" column="BAZ_ID"/>
</dynamic-component>
<dynamic-component> 매핑의 의미는 <component>와 동일하다. 이런 종류의 매핑의 장점은 배치 시에 매핑 문서를 편집함으로써 그 bean의 실제 프로퍼티들을 결정하는 가용성이다. 매핑 문서에 대한 런타임 처리는 또한 DOM 파서를 사용하여 가능하다. 더 좋게는 당신이 Configuration 객체를 통해 Hibernate의 구성-시 메타모형에 액세스할 수 있다(그리고 그것을 변경시킬 수 있다)
Hibernate는 세 개의 기본적인 상속 매핑 방도들을 지원한다:
-
- table per class hierarchy
- table per subclass
- table per concrete class
게다가 Hibernate는 네 번째의 약간 다른 종류의 다형성을 지원한다:
implicit polymorphism(함축적인 다형성)
동일한 상속 계층구조의 다른 가지들에 대해 다른 매핑 방도들을 사용하는 것이 가능하고, 전체 계층 구조를 가로질러 다형성을 성취하는데 함축적인 다형성을 사용하라. 하지만 Hibernate는 동일한 루트 <class> 요소 하에서 <subclass>, 그리고 <joined-subclass> 그리고 <union-subclass> 매핑들을 혼합하는 것을 지원하지 않는다. 동일한 <class> 요소 하에서 <subclass> 요소와 <join> 요소를 결합시킴으로써 table per hierarchy 방도와 table per subclass 방도를 함께 혼합시키는 것이 가능하다(아래를 보라).
9.1.1. Table per class hierarchy
우리가 CreditCardPayment, CashPayment, ChequePayment 구현자들을 가진 인터페이스 Payment를 갖고 있다고 가정하자. table per hierarchy 매핑은 다음과 같을 것이다:
<class name="Payment" table="PAYMENT">
<id name="id" type="long" column="PAYMENT_ID">
<generator class="native"/>
</id>
<discriminator column="PAYMENT_TYPE" type="string"/>
<property name="amount" column="AMOUNT"/>
...
<subclass name="CreditCardPayment" discriminator-value="CREDIT">
<property name="creditCardType" column="CCTYPE"/>
...
</subclass>
<subclass name="CashPayment" discriminator-value="CASH">
...
</subclass>
<subclass name="ChequePayment" discriminator-value="CHEQUE">
...
</subclass>
</class>
정확히 하나의 테이블이 필요하다. 이 매핑 방도에는 다음의 하나의 큰 제약이 존재한다: CCTYPE과 같이 서브 클래스들에 의해 선언된 컬럼들은 NOT NULL 컨스트레인트들을 가질 수 없다.
9.1.2. Table per class subclass
table per subclass 매핑은 다음과 같을 것이다:
<class name="Payment" table="PAYMENT">
<id name="id" type="long" column="PAYMENT_ID">
<generator class="native"/>
</id>
<property name="amount" column="AMOUNT"/>
...
<joined-subclass name="CreditCardPayment" table="CREDIT_PAYMENT">
<key column="PAYMENT_ID"/>
<property name="creditCardType" column="CCTYPE"/>
...
</joined-subclass>
<joined-subclass name="CashPayment" table="CASH_PAYMENT">
<key column="PAYMENT_ID"/>
...
</joined-subclass>
<joined-subclass name="ChequePayment" table="CHEQUE_PAYMENT">
<key column="PAYMENT_ID"/>
...
</joined-subclass>
</class>
네 개의 테이블들이 필요하다. 세 개의 서브클래스 테이블들은 슈퍼클래스 테이블에 대한 프라이머리 키 연관들을 갖는다(따라서 그 관계형 모형은 실제로 one-to-one 연관이다).
9.1.3. discriminator를 사용하는 table per subclass
table-per-subclass에 대한 Hibernate의 구현은 discriminator(판별자) 컬럼을 필요로 하지 않음을 노트하라. 다른 객체/관계형 매핑기들은 슈퍼클래스 테이블 속에 타입 discriminator 컬럼을 필요로 하는 table-per-subclass에 대한 다른 구현을 사용한다. Hibernate에 채택된 접근법은 구현하기가 훨씬 더 어렵지만 관계형 관점에서는 아마 틀림없이 보다 더 정확하다. 만일 당신이 table per subclass 방도를 대해 discriminator 컬럼을 사용하고 싶다면, 당신은 다음과 같이 <subclass>와 <join>의 사용을 결합시킬 수도 있다:
<class name="Payment" table="PAYMENT">
<id name="id" type="long" column="PAYMENT_ID">
<generator class="native"/>
</id>
<discriminator column="PAYMENT_TYPE" type="string"/>
<property name="amount" column="AMOUNT"/>
...
<subclass name="CreditCardPayment" discriminator-value="CREDIT">
<join table="CREDIT_PAYMENT">
<property name="creditCardType" column="CCTYPE"/>
...
</join>
</subclass>
<subclass name="CashPayment" discriminator-value="CASH">
<join table="CASH_PAYMENT">
...
</join>
</subclass>
<subclass name="ChequePayment" discriminator-value="CHEQUE">
<join table="CHEQUE_PAYMENT" fetch="select">
...
</join>
</subclass>
</class>
선택적인 fetch="select" 선언은 슈퍼클래스를 질의할 때 outer join을 사용하여 ChequePayment 서브클래스 데이터를 페치시키지 않도록 Hibernate에게 알려준다.
9.1.4. table per class hierarchy와 table per subclass를 혼합하기
당신은 다음 접근법을 사용하여 table per hierarchy 방도와 table per subclass 방도를 혼합시킬 수 있다:
<class name="Payment" table="PAYMENT">
<id name="id" type="long" column="PAYMENT_ID">
<generator class="native"/>
</id>
<discriminator column="PAYMENT_TYPE" type="string"/>
<property name="amount" column="AMOUNT"/>
...
<subclass name="CreditCardPayment" discriminator-value="CREDIT">
<join table="CREDIT_PAYMENT">
<property name="creditCardType" column="CCTYPE"/>
...
</join>
</subclass>
<subclass name="CashPayment" discriminator-value="CASH">
...
</subclass>
<subclass name="ChequePayment" discriminator-value="CHEQUE">
...
</subclass>
</class>
이들 매핑 방도들 중 어떤 것에 대해, 루트 Payment 클래스에 대한 다형성 연관은 <many-to-one>을 사용하여 매핑된다.
<many-to-one name="payment" column="PAYMENT_ID" class="Payment"/>
9.1.5. Table per concrete class
우리가 table per concrete class 방도 매핑에 대해 취할 수 있는 두 가지 방법들이 존재한다. 첫 번째는 <union-subclass>를 사용하는 것이다.
<class name="Payment">
<id name="id" type="long" column="PAYMENT_ID">
<generator class="native"/>
</id>
<property name="amount" column="AMOUNT"/>
...
<union-subclass name="CreditCardPayment" table="CREDIT_PAYMENT">
<property name="creditCardType" column="CCTYPE"/>
...
</union-subclass>
<union-subclass name="CashPayment" table="CASH_PAYMENT">
...
</union-subclass>
<union-subclass name="ChequePayment" table="CHEQUE_PAYMENT">
...
</union-subclass>
</class>
세 개의 테이블들이 수반된다. 각각의 테이블은 상속된 프로퍼티들을 포함하여 그 클래스의 모든 프로퍼티들에 대한 컬럼들을 정의한다.
이 접근법의 제약은 만일 하나의 프로퍼티가 슈퍼클래스 상으로 매핑될 경우, 그 컬럼 이름은 모든 서브클래스 테이블들 상에서 같아야 한다는 점이다.(우리는 장래의 Hibernate 배포본에서 이 제약을 풀 수도 있다. )
9.1.6. Table per concrete class, using implicit polymorphism
대체적인 접근법은 함축적인 다형성을 사용하는 것이다:
<class name="CreditCardPayment" table="CREDIT_PAYMENT">
<id name="id" type="long" column="CREDIT_PAYMENT_ID">
<generator class="native"/>
</id>
<property name="amount" column="CREDIT_AMOUNT"/>
...
</class>
<class name="CashPayment" table="CASH_PAYMENT">
<id name="id" type="long" column="CASH_PAYMENT_ID">
<generator class="native"/>
</id>
<property name="amount" column="CASH_AMOUNT"/>
...
</class>
<class name="ChequePayment" table="CHEQUE_PAYMENT">
<id name="id" type="long" column="CHEQUE_PAYMENT_ID">
<generator class="native"/>
</id>
<property name="amount" column="CHEQUE_AMOUNT"/>
...
</class>
아무데서도 우리가 명시적으로 Payment 인터페이스를 언급하지 않음을 주목하라. 또한 Payment의 프로퍼티들이 서브클래스들 각각에서 매핑된다는 점을 주목하라. 만일 당신이 중복을 피하고자 원한다면, XML 엔티티들을 사용하는 것을 고려하라(예를 들어 매핑에서 DOCTYPE 선언과 &allproperties; 에서 [ <!ENTITY allproperties SYSTEM "allproperties.xml"> ]).
이 접근법의 단점은 다형성 질의들을 수행할 때 Hibernate가 생성된 SQl UNION들을 생성시키는 않는다는 점이다.
이 매핑 방도의 경우, Payment에 대한 다형성 연관은 대개 <any>를 사용하여 매핑된다.
<any name="payment" meta-type="string" id-type="long">
<meta-value value="CREDIT" class="CreditCardPayment"/>
<meta-value value="CASH" class="CashPayment"/>
<meta-value value="CHEQUE" class="ChequePayment"/>
<column name="PAYMENT_CLASS"/>
<column name="PAYMENT_ID"/>
</any>
9.1.7. 함축적인 다형성을 다른 상속 매핑들과 혼합하기
이 매핑에 대해 주목하는 하나 이상의 것이 존재한다. 서브클래스들이 그것들 자신의 <class> 요소 내에 각각 매핑되므로(그리고 Payment 가 단지 인터페이스이므로), 서브클래스들 각각은 쉽게 또 다른 상속 계층구조의 부분일 수 있다! (그리고 당신은 Payment 인터페이스에 대해 여전히 다형성 질의들을 사용할 수 있다.)
<class name="CreditCardPayment" table="CREDIT_PAYMENT">
<id name="id" type="long" column="CREDIT_PAYMENT_ID">
<generator class="native"/>
</id>
<discriminator column="CREDIT_CARD" type="string"/>
<property name="amount" column="CREDIT_AMOUNT"/>
...
<subclass name="MasterCardPayment" discriminator-value="MDC"/>
<subclass name="VisaPayment" discriminator-value="VISA"/>
</class>
<class name="NonelectronicTransaction" table="NONELECTRONIC_TXN">
<id name="id" type="long" column="TXN_ID">
<generator class="native"/>
</id>
...
<joined-subclass name="CashPayment" table="CASH_PAYMENT">
<key column="PAYMENT_ID"/>
<property name="amount" column="CASH_AMOUNT"/>
...
</joined-subclass>
<joined-subclass name="ChequePayment" table="CHEQUE_PAYMENT">
<key column="PAYMENT_ID"/>
<property name="amount" column="CHEQUE_AMOUNT"/>
...
</joined-subclass>
</class>
다시 한번, 우리는 Payment를 명시적으로 언급하지 않는다. 만일 우리가 Payment 인터페이스에 대해 질의를 행할 경우-예를 들면. from Payment- , Hibernate는 CreditCardPayment (와 그것의 서브클래스들, 왜냐하면 그것들이 또한 Payment를 구현하므로), CashPayment 그리고 ChequePayment 인스턴스들을 자동적으로 반환할 것이지만 NonelectronicTransaction의 인스턴스들을 반환하지 않는다.
table per concrete-class 매핑 방도에 대한 "함축적인 다형성" 접근법에는 어떤 제약들이 존재한다. <union-subclass> 매핑들에 대해서는 다소 덜 제약적인 제약들이 존재한다:
다음 표는 Hibernate에서 table per concrete-class 매핑들에 대한 제약들, 그리고 함축적인 다형성에 대한 제약들을 보여준다.
상속 방도 |
다형성 many-to-one |
다형성 one-to-one |
다형성 one-to-many |
다형성 many-to-many |
다형성 load()/get() |
다형성 queries |
다형성 joins |
Outer join 페칭 |
table per class-hierarchy |
<many-to-one> |
<one-to-one> |
<one-to-many> |
<many-to-many> |
s.get(Payment.class, id) |
from Payment p |
from Order o join o.payment p |
지원됨 |
table per subclass |
<many-to-one> |
<one-to-one> |
<one-to-many> |
<many-to-many> |
s.get(Payment.class, id) |
from Payment p |
from Order o join o.payment p |
지원됨 |
table per concrete-class (union-subclass) |
<many-to-one> |
<one-to-one> |
<one-to-many> (inverse="true" 에 대해서만) |
<many-to-many> |
s.get(Payment.class, id) |
from Payment p |
from Order o join o.payment p |
지원됨 |
table per concrete class (implicit polymorphism) |
<any> |
지원되지 않음 |
지원되지 않음 |
<many-to-any> |
s.createCriteria(Payment.class).add( Restrictions.idEq(id) ).uniqueResult() |
from Payment p |
지원되지 않음 |
지원되지 않음 |
제 10. 객체들로 작업하기
Hibernate는 기본 데이터베이스 관리 시스템의 상세로부터 개발자들을 은폐시켜줄 뿐만 아니라, 또한 객체들의 상태 관리를 제공하는 전체 객체/관계형 매핑 솔루션이다. 이것은 공통적인 JDBC/SQL 영속 계층들 내에서의 SQL statements에 대한 관리와는 반대로, 자바 어플리케이션들에서 영속에 관한 매우 고유한 객체-지향적 전망이다.
달리 말해, Hibernate 어플리케이션 개발자들은 항상 그들의 객체들의 상태에 대해 생각할 것이고, SQL 문장들의 실행에 대한 생각은 필수적이지 않다. 이 부분은 Hibernate에 의해 처리되고 어플리케이션 개발자가 시스템의 퍼포먼스를 튜닝할 때에만 관련된다.
Hibernate는 다음 객체 상태들을 정의하고 지원한다:
- Transient - 만일 객체가 오퍼레이터를 사용하여 방금 초기화 되었다면 객체는 transient이고, 그것은 Hibernate Session과 연관되어 있지 않다. 그것은 데이터베이스 내에서 영속 표상을 갖지 않고 식별자 값이 할당되지 않았다. 만일 어플리케이션이 더이상 참조를 갖지 않을 경우 transient 인스턴스들이 쓰레기 수집기에 의해 파괴돌 것이다. 객체를 영속화 시키는데(그리고 Hibernate로 하여금 이 전이(transition)에 대해 실행될 필요가 있는 SQL 문장들을 처리하도록 하는데) Hibernate Session을 사용하라.
- Persistent - 영속 인스턴스는 데이터베이스 내에 표상을 갖고 식별자 값을 갖는다. 그것은 방금 저장되었지만, 정의상 그것은 Session의 범위 내에 있다. Hibernate는 영속 상태에서 객체에 대해 행해진 임의의 변경들을 검출해낼 것이고 단위 작업이 완료될 때 데이터베이스와 그 상태를 동기화 시킬 것이다. 개발자들은 객체가 transient가 되어야 할 때 UPDATE 문장들이나 DELETE 문장들을 수작업으로 실행하지 않는다.
- Detached - detached 인스턴스는 영속화 되었지만, 그것의 Session이 닫혀진 객체이다. 물론 그 객체에 대한 참조는 여전히 유효하고 detached 인스턴스는 이 상태에서도 변경될 수도 있다. detached 인스턴스는 나중에 그것(과 모두 변경들)을 다시 영속화 시켜서새로운 Session에 다시첨부될 수 있다. 이 특징은 사용자가 생각할 시간을 필요로 하는 장 기간 실행되는 작업 단위에 대한 프로그래밍 모형을 가능하게 해준다. 우리는 그것들을 어플리케이션 트랜잭션들, 사용자의 관점의 작업 단위로 부른다.
이제 우리는 상태들과 상태 전이(transition)들(그리고 전이를 트리거 시키는 Hibernate 메소드들)을 상세하게 논의할 것이다.
영속 클래스의 새로이 생성된 인스턴스들은 Hibernate에 의해 transient 로 간주된다. 우리는 그것을 세션과 연관지어서 transient 인스턴스를 영속화 시킬 수 있다:
DomesticCat fritz = new DomesticCat();
fritz.setColor(Color.GINGER);
fritz.setSex('M');
fritz.setName("Fritz");
Long generatedId = (Long) sess.save(fritz);
Cat 가 생성된 식별자를 가질 경우, save()가 호출될 때 그 식별자가 생성되고 cat에 할당된다. 만일 Cat이 assigned 식별자나 composite key를 가질 경우, save()를 호출하기 전에 그 식별자는 cat 인스턴스에 할당될 것이다. 당신은 또한 EJB3 초기 드래프트에서 정의된 의미로 save() 대신 persist()를 사용할 수도 있다.
다른 방법으로, 당신은 save()의 오버로드된 버전을 사용하여 식별자를 할당할 수 있다.
DomesticCat pk = new DomesticCat();
pk.setColor(Color.TABBY);
pk.setSex('F');
pk.setName("PK");
pk.setKittens( new HashSet() );
pk.addKitten(fritz);
sess.save( pk, new Long(1234) );
만일 당신이 영속화 시키는 객체가 객체들(예를 들면. 앞의 예제에 있는 kittens 콜렉션)을 연관시켰다면, 이들 객체들은 당신이 foreign 키 컬럼에 대해 NOT NULL 컨스트레인트를 갖지 않는 한, 당신이 좋아하는 임의의 순서로 영속화 되었을 수도 있다. foreign 키 컨스트레인트들을 위배하는 위험성이 결코 존재하지 않는다. 하지만 당신이 잘못된 순서로 객체들을 save()할 경우 당신은 NOT NULL 컨스트레인트를 위배할 수도 있다.
당신이 연관된 객체들을 자동적으로 저장시키는데 Hibernate의 transitive persistence(과도적인 영속) 특징을 사용하는 것을 매우 좋아할 것이므로 대개 당신은 이 상세를 내버려둔다. 그때 NOT NULL 컨스트레인트 위배들이 발생되지 않을지라도 - Hibernate는 모든 것을 처리할 것이다. Transitive persistence(과도적인 영속)은 이 장에서 나중에 논의된다.
Session의 load() 메소드들은 만일 당신이 그것(영속 인스턴스)의 식별자들을 이미 알고 있을 경우에 당신에게 영속 인스턴스를 검색하는 방법을 제공한다. load()는 클래스 객체를 취하고 그 클래스의 새로이 초기화 된 인스턴스 속으로 그 상태를 영속 상태로 로드시킬 것이다.
Cat fritz = (Cat) sess.load(Cat.class, generatedId);
// 당신은 primitive 식별자들을 포장할 필요가 있다
long pkId = 1234;
DomesticCat pk = (DomesticCat) sess.load( Cat.class, new Long(pkId) );
다른 방법으로 당신은 주어진 인스턴스 속으로 상태를 로드시킬 수 있다:
Cat cat = new DomesticCat();
// pk의 상태를 cat 속으로 로드시킨다
sess.load( cat, new Long(pkId) );
Set kittens = cat.getKittens();
만일 일치하는 데이터베이스 행이 존재하지 않을 경우에 load()가 unrecoverable 예외상황을 던질 것임을 노트하라. 만일 클래스가 프락시를 갖도록 매핑된 경우, load()는 초기화 되지 않은 프락시를 반환하고 당신이 그 프락시의 메소드를 호출하기 전까지는 실제로 데이터베이스에 접속하지 않는다. 당신이 데이터베이스로부터 객체에 대한 연관을 실제로 로드시키지 않고서 객체에 대한 연관을 생성시키고자 원할 경우에 이 특징이 매우 유용하다. 만일 batch-size 가 클래스 매핑에 정의되는 경우 그것은 또한 다중 인스턴스들이 배치로서 로드되는 것을 허용해준다.
만일 당신이 일치하는 행이 존재하는지를 확신할 수 없을 경우, 당신은 get() 메소드를 사용해야 한다. 그것(get() 메소드)은 데이터베이스에 즉시 접속하고 만일 일치하는 행이 없을 경우 null을 반환한다.
Cat cat = (Cat) sess.get(Cat.class, id);
if (cat==null) {
cat = new Cat();
sess.save(cat, id);
}
return cat;
당신은 LockMode를 사용하는 SQL SELECT ... FOR UPDATE를 사용하여 객체를 로드시킬 수도 있다. 추가 정보는 API 문서를 보라.
Cat cat = (Cat) sess.get(Cat.class, id, LockMode.UPGRADE);
어떤 연관된 인스턴스들이나 포함된 콜렉션들은 당신이 그 연관에 대한 케스케이드 스타일로서 lock 또는 all을 지정하도록 결정하지 않는 한 FOR UPDATE를 select 하지 않음을 노트하라.
refresh() 메소드를 사용하여 아무때나 객체와 모든 그것의 콜렉션들을 다시 로드시키는 것이 가능하다. 데이터베이스 트리거들이 그 객체의 프로퍼티들 중 어떤 것을 초기화 시키는데 사용될 때 이것이 유용하다.
sess.save(cat);
sess.flush(); // SQL INSERT를 강제시킨다
sess.refresh(cat); // (트리거가 실행된 후에) 상태를 다시 읽어들인다
중요한 질문이 대개 이 지점에서 나타난다: Hibernate는 데이터베이로부터 얼마나 많이 로드시키고 그것은 얼마나 많은 SQL SELECT들을 사용할 것인가? 이것은 페칭 방도에 의존하고 19.1 절, “페칭 방도들”에 설명되어 있다.
만일 당신이 당신이 찾고 있는 객체들의 식별자들을 모를 경우, 당신은 질의를 필요로 한다. Hibernate는 사용이 쉽지만 강력한 객체 지향 질의 언어(HQL)를 지원한다. 프로그램 상의 질의 생성을 위해, Hibernate는 정교한 Criteria 및 Example 질의 특징(QBC와 QBE)를 지원한다. 당신은 또한 Hibernate로부터 객체들 속으로 결과 셋 변환을 위한 선택적인 지원으로, 당신의 데이터베이스의 native SQL 속에 당신의 질의를 표현할 수도 있다.
HQL 질의와 native SQL 질의는 org.hibernate.Query의 인스턴스로 표현된다. 이 인터페이스는 파라미터 바인딩, 결과셋 핸들링, 그리고 실제 질의의 실행을 위한 메소드들을 제공한다. 당신은 현재의 Session을 사용하여 항상 Query를 얻는다:
List cats = session.createQuery(
"from Cat as cat where cat.birthdate < ?")
.setDate(0, date)
.list();
List mothers = session.createQuery(
"select mother from Cat as cat join cat.mother as mother where cat.name = ?")
.setString(0, name)
.list();
List kittens = session.createQuery(
"from Cat as cat where cat.mother = ?")
.setEntity(0, pk)
.list();
Cat mother = (Cat) session.createQuery(
"select cat.mother from Cat as cat where cat = ?")
.setEntity(0, izi)
.uniqueResult();
질의는 대개 list()를 호출하여 실행되고, 질의의 결과는 메모리 내에서 콜렉션 속으로 전체적으로 로드될 것이다. 질의에 의해 검색된 엔티티 인스턴스들은 영속 상태에 있다. 당신의 질의가 하나의 객체를 오직 반환할 것임을 당신이 알고 있을 경우에 uniqueResult() 메소드는 단축을 제공한다.
종종, 당신은 iterate() 메소드를 사용하여 질의를 실행하여 더 나은 퍼포먼스를 성취하는 것이 가능할 수 있다. 이것은 오직 대개 질의에 의해 반환되는 실제 엔티티 인스턴스들이 이미 세션 내에 있거나 second-level 캐시 내에 있을 것임을 당신이 예상하는 경우일 것이다. 만일 그것들이 이미 캐시되지 않았다면, iterate()는 list() 보다 더 느릴 것이고 간단한 질의에 대해 많은 데이터베이스 접속들을, 대개 오직 식별자들을 반환하는 초기 select에 대해 1번의 접속과 실제 인스턴스들을 초기화 시키는 추가적인 select들에 대해 n 번의 접속을 필요로 할 수 있다.
// id들을 페치시킨다
Iterator iter = sess.createQuery("from eg.Qux q order by q.likeliness").iterate();
while ( iter.hasNext() ) {
Qux qux = (Qux) iter.next(); // 객체들을 페치시킨다
// 질의 속에 우리가 표현할 수 없는 어떤 것
if ( qux.calculateComplicatedAlgorithm() ) {
// 현재 인스턴스를 삭제시킨다
iter.remove();
// 나머지를 처리할 필요가 없다
break;
}
}
Hibernate 질의들은 때때로 객체들의 튜플들을 반환하고, 그 경우에 각각의 튜플은 배열로서 반환된다:
Iterator kittensAndMothers = sess.createQuery(
"select kitten, mother from Cat kitten join kitten.mother mother")
.list()
.iterator();
while ( kittensAndMothers.hasNext() ) {
Object[] tuple = (Object[]) kittensAndMothers.next();
Cat kitten = tuple[0];
Cat mother = tuple[1];
....
}
질의들은 select 절 내에 클래스의 프로퍼티를 지정할 수 있다. 그것들은 심지어 SQL 집계 함수들을 호출할 수도 있다. 프로퍼티들이나 aggregate들은 (영속 상태에 있는 엔티티들이 아닌) "스칼라" 결과들로 간주된다.
Iterator results = sess.createQuery(
"select cat.color, min(cat.birthdate), count(cat) from Cat cat " +
"group by cat.color")
.list()
.iterator();
while ( results.hasNext() ) {
Object[] row = results.next();
Color type = (Color) row[0];
Date oldest = (Date) row[1];
Integer count = (Integer) row[2];
.....
}
Query 상의 메소드들은 명명된 파라미터들 또는 JDBC-스타일의 ? 파라미터들에 바인딩 값들을 제공한다. JDBC와는 대조적으로, Hibernate 숫자 파라미터들은 0에서 시작된다. 명명된 파라미터들은 질의 문자열 속에서 :name 형식의 식별자들이다. 명명된 파라미터들의 장점들은 다음과 같다:
- 명명된 파라미터들은 그것들이 질의 문자열 내에 발생하는 순서에 관계없다
- 그것들은 동일한 질의 내에서 여러 번 발생할 수 있다
- 그것은 자기-설명적이다
//명명된 파라미터 (선호됨)
Query q = sess.createQuery("from DomesticCat cat where cat.name = :name");
q.setString("name", "Fritz");
Iterator cats = q.iterate();
//위치 파라미터
Query q = sess.createQuery("from DomesticCat cat where cat.name = ?");
q.setString(0, "Izi");
Iterator cats = q.iterate();
//명명된 파라미터 리스트
List names = new ArrayList();
names.add("Izi");
names.add("Fritz");
Query q = sess.createQuery("from DomesticCat cat where cat.name in (:namesList)");
q.setParameterList("namesList", names);
List cats = q.list();
만일 당신이 당신의 결과 셋에 경계(당신이 검색하고자 원하는 최대 개수 와/또는 당신이 검색하고자 원하는 첫 번째 행)를 지정할 필요가 있다면 당신은 Query 인터페이스의 메소드들을 사용해야 한다:
Query q = sess.createQuery("from DomesticCat cat");
q.setFirstResult(20);
q.setMaxResults(10);
List cats = q.list();
Hibernate는 이 limit 질의를 당신의 DBMS의 native SQL로 번역하는 방법을 알고 있다.
10.4.1.6. 스크롤 가능한 iteration
당신의 JDBC 드라이버가 스크롤 가능한 ResultSet들을 지원할 경우, Query 인터페이스는 ScrollableResults 객체를 얻는데 사용될 수 있고, 그것은 질의 결과들에 대한 유연한 네비게이션을 허용해준다.
Query q = sess.createQuery("select cat.name, cat from DomesticCat cat " +
"order by cat.name");
ScrollableResults cats = q.scroll();
if ( cats.first() ) {
// find the first name on each page of an alphabetical list of cats by name
firstNamesOfPages = new ArrayList();
do {
String name = cats.getString(0);
firstNamesOfPages.add(name);
}
while ( cats.scroll(PAGE_SIZE) );
// Now get the first page of cats
pageOfCats = new ArrayList();
cats.beforeFirst();
int i=0;
while( ( PAGE_SIZE > i++ ) && cats.next() ) pageOfCats.add( cats.get(1) );
}
cats.close()
열려진 데이터베이스 커넥션(과 커서)가 이 기능에 필요함을 노트하고, 만일 당신이 쪽매김 기능을 작동시킬 필요가 있다면 setMaxResult()/setFirstResult()를 사용하라.
10.4.1.7. 명명된 질의들을 구체화 시키기
당신은 또한 매핑 문서 속에 명명된 질의들을 정의할 수 있다.(만일 당신의 질의가 마크업으로서 해석될 수 있는 문자들을 포함할 경우에 CDATA 섹션을 사용하는 것을 기억하라)
<query name="eg.DomesticCat.by.name.and.minimum.weight"><![CDATA[
from eg.DomesticCat as cat
where cat.name = ?
and cat.weight > ?
] ]></query>
파라미터 바인딩과 실행은 프로그램 상으로 행해진다:
Query q = sess.getNamedQuery("eg.DomesticCat.by.name.and.minimum.weight");
q.setString(0, name);
q.setInt(1, minWeight);
List cats = q.list();
실제 프로그램 코드는 사용되는 질의 언어에 독리비적이고, 당신은 또한 메타데이터로 native SQL 질의들을 정의할 수도 있거나 그것들을 매핑 파일들 속에 위치지움으로써 기존 질의들을 Hibernate로 이전시킬 수도 있음을 노트하라.
콜렉션 필터는 영속 콜렉션 또는 배열에 적용될 수 있는 특별한 질의 타입이다. 질의 문자열은 현재의 콜렉션 요소를 의미하는 this를 참조할 수 있다.
Collection blackKittens = session.createFilter(
pk.getKittens(),
"where this.color = ?")
.setParameter( Color.BLACK, Hibernate.custom(ColorUserType.class) )
.list()
);
반환되는 콜렉션은 하나의 bag으로 간주되고, 그것은 주어진 콜렉션에 대한 사본이다. 원래의 콜렉션은 변경되지 않는다(이것은 이름 "filter"의 의미와는 정반대이지만, 예상되는 행위와 일치된다).
필터들은 from 절을 필요로 하지 않음을 관찰하라(비록 그것들이 필요할 경우에 한개의 from 절을 가질 수 있을지라도). 필터들은 콜렉션 요소들 자체들을 반환하는 것으로 제약되지 않는다.
Collection blackKittenMates = session.createFilter(
pk.getKittens(),
"select this.mate where this.color = eg.Color.BLACK.intValue")
.list();
심지어 공백의 필터 질의도 예를 들어 거대한 콜렉션 내에 잇는 요소들의 부분집합들을 로드시키는데 유용하다:
Collection tenKittens = session.createFilter(
mother.getKittens(), "")
.setFirstResult(0).setMaxResults(10)
.list();
HQL은 극히 강력하지만 몇몇 개발자들은 질의 문자열들을 빌드하는 것 보다, 객체 지향 API를 사용하여 동적으로 질의들을 빌드시키는 것을 선호한다. Hibernate는 이들 경우들에 대해 직관적인 Criteria query API를 제공한다:
Criteria crit = session.createCriteria(Cat.class);
crit.add( Expression.eq( "color", eg.Color.BLACK ) );
crit.setMaxResults(10);
List cats = crit.list();
Criteria 와 연관된 Example API 는 15장, Criteria 질의들에서 상세하게 논의된다.
당신은 createSQLQuery() 를 사용하여 SQL로 질의를 표현할 수 있고, Hibernate로 하여금 객체들에 대한 결과 셋들로부터 매핑을 처리하도록 한다. 당신은 아무때나 session.connection()을 호출할 수 있고 직접 JDBC Connection을 사용할 수 있음을 노트하라. 만일 당신이 Hibernate API를 사용하고자 선택한 경우에 , 당신은 SQL alias들을 중괄호들 속에 포함시켜야 한다:
List cats = session.createSQLQuery(
"SELECT {cat.*} FROM CAT {cat} WHERE ROWNUM<10",
"cat",
Cat.class
).list();
List cats = session.createSQLQuery(
"SELECT {cat}.ID AS {cat.id}, {cat}.SEX AS {cat.sex}, " +
"{cat}.MATE AS {cat.mate}, {cat}.SUBCLASS AS {cat.class}, ... " +
"FROM CAT {cat} WHERE ROWNUM<10",
"cat",
Cat.class
).list()
SQL 질의들은 Hibernate 질의들처럼 명명된 파라미터들과 위치 파라미터들을 포함할 수도 있다. SQL 질의들에 대한 추가 정보는 16 장, Native SQL에서 찾을 수 있다.
트랜잭션 상의 영속 인스턴스들 (예를들면. Session에 의해 로드되고, 저장되고, 생성되거나 질의된 객체들)은 어플리케이션에 의해 처리될 수 있고 영속 상태에 대한 임의의 변경들은 Session이 flush될 때 영속화 될 것이다. (이 장의 뒷 부분에서 논의됨). 당신의 변경들을 영속화 시키기 위해 (다른 용도를 가진 update()와 같은) 특별한 메소드를 호출할 필요가 없다. 따라서 객체의 상태를 업데이트 시키는 대부분의 간단한 방법은 Session이 열려 있는 동안, 그것을 load()시키고 나서, 그것을 직접 처리하는 것이다:
DomesticCat cat = (DomesticCat) sess.load( Cat.class, new Long(69) );
cat.setName("PK");
sess.flush(); // changes to cat are automatically detected and persisted
때때로 프로그래밍 모형은 불충분하다. 왜냐하면 그것은 동일한 세션 내에서 (객체를 로드시키는) SQL SELECT와 (그것의 업데이트된 사아태를 영속화 시키는) SQL UPDATE 양자를 필요로 할 것이기 때문이다. 그러므로 Hibernate는 detached 인스턴스들을 사용하는 대안적인 접근법을 제공한다.
Hibernate는 UPDATE 문장 또는 DELETE 문장의 직접적인 실행을 위한 그것 자신의 API를 제공하지 않음을 노트하라. Hibernate는 상태 관리 서비스이고, 당신은 그것을 사용할 문장들을 생각하지 말아야 한다. JDBC는 SQL 문장들을 실행시키는 완전한 API이고, 당신은 session.connection()을 호출함으로써 아무때나 JDBC Connection을 얻을 수 있다. 게다가 대규모 오퍼레이션의 개념은 온라인 트랜잭션 처리-지향적인 어플리케이션들을 위한 객체/관계형 매핑과 충돌한다. 하지만 Hibernate의 장래 버전들은 특별한 대규모의 오퍼레이션 함수들을을 제공할 수도 있다. 몇몇 가능한 배치 오퍼레이션 트릭들에 대해서는 13장, 배치 프로세싱을 보라.
10.6. detached 객체들을 변경시키기
많은 어플리케이션들은 하나의 트랜잭션 내에서 객체를 검색하고, 처리를 위한 UI 계층으로 그것을 전송하고, 새로운 트랜잭션 내에서 변경들을 저장할 필요가 있다. 높은-동시성 환경에서 이런 종류의 접근법을 사용하는 어플리케이션들은 대개 작업의 "긴" 단위를 확실히 격리시키기 위해 버전화 된 데이터를 사용한다.
Hibernate는 Session.update() 메소드 또는 Session.merge() 메소드를 사용하여 detached 객체들의 재첨부를 제공함으로써 이 모형을 지원한다:
// 첫 번째 세션에서
Cat cat = (Cat) firstSession.load(Cat.class, catId);
Cat potentialMate = new Cat();
firstSession.save(potentialMate);
// 어플리케이션의 더 높은 계층에서
cat.setMate(potentialMate);
// 나중에 새로운 세션에서
secondSession.update(cat); // cat을 업데이트 한다
secondSession.update(mate); // mate를 업데이트 한다
만일 catId 식별자를 가진 Cat이 secondSession에 의해 이미 로드되었을 경우에 어플리케이션이 그것을 다시 재첨부하려고 시도할 때, 예외상황이 던져졌을 것이다.
만일 그 세션이 동일한 식별자를 가진 영속 인스턴스를 이미 포함하지 않음을 당신이 확신하고 있다면 update()를 사용하고, 만일 당신이 세션의 상태를 고려하지 않은채로 아무때나 당신의 변경을 병합시키고자 원할 경우에 merge()를 사용하라. 달리 말해, update()는 당신의 detached 인스턴스들에 대한 재첨부가 실행되는 첫 번째 오퍼레이션임을 확실히 함으로써, 대개 갓 만들어진 세션에서 당신이 호출하게 될 첫 번째 메소드이다.
어플리케이션은 그것이 detached 인스턴스들의 상태가 또한 업데이트 되는 것을 원할 경우에만 주어진 detached 인스턴스로부터 도달 가능한 detached 인스턴스들을 개별적으로 update() 시킬 것이다. 이것은 물론 transitive persistence(과도적인 영속)을 사용하여 자동화 될 수 있고, see 10.11절, “Transitive persistence”를 보라.
lock() 메소드는 또한 새로운 세션에 대해 객체를 다시 연관시키는것을 어플리케이션에게 허용해준다. 하지만 detached 인스턴스는 변경되지 않아야 한다!
//단지 다시 연관지운다:
sess.lock(fritz, LockMode.NONE);
//버전 체크를 행하고 나서 다시 연관 짓는다:
sess.lock(izi, LockMode.READ);
//SELECT ... FOR UPDATE를 사용하여 버전 체크를 행하고 나서 다시 연관 짓는다:
sess.lock(pk, LockMode.UPGRADE);
lock() 이 여러가지 LockMode들에 사용될 수 있음을 노트하고, 상세한 것은 API 문서와 트랜잭션 처리에 관한 장을 보라. 재첨부는 lock()에 대한 유일한 쓰임새는 아니다.
긴 작업 단위에 대한 다른 모형들은 11.3절, “Optimistic 동시성 제어”에서 논의된다.
Hibernate 사용자들은 새로운 식별자를 생성시킴으로써 transient 인스턴스를 저장하는 또는 그것의 현재 식별자와 연관된 detached 인스턴스들을 업데이트/재첨부 시키는 일반적인 용도의 메소드를 요청했다. saveOrUpdate() 메소드는 이 기능을 구현한다.
// 첫 번째 세션에서
Cat cat = (Cat) firstSession.load(Cat.class, catID);
// 어플리케이션의 더 높은 티어에서
Cat mate = new Cat();
cat.setMate(mate);
// 나중에 새로운 세션에서
secondSession.saveOrUpdate(cat); // 기존 상태를 업데이트 한다 (cat은 non-null id를 갖는다)
secondSession.saveOrUpdate(mate); // 새로운 인스턴스를 지정한다 (mate는 null id를 갖는다)
saveOrUpdate()의 사용 예제와 의미는 초심자들에게는 혼동스러워 보인다. 먼저, 하나의 세션으로부터의 인스턴스를 또 다른 새로운 세션 내에 사용하려고 시도하지 않는 한, 당신은 update(), saveOrUpdate(), 또는 merge()를 사용할 필요는 없을 것이다. 몇몇 전체 어플리케이션들은 이들 메소드들 중 어느 것도 결코 사용하지 않을 것이다.
대개 update() 또는 saveOrUpdate() 는 다음 시나리오에서 사용된다:
- 어플리케이션은 첫 번째 세션 내에 객체를 로드시킨다
- 그 객체는 UI 티어로 전달된다
- 몇몇 변경들이 그 객체에 행해진다
- 그 객체는 비지니스 로직 티어로 전달된다
- 어플리케이션은 두 번째 세션에서 update()를 호출함으로써 이들 변경들을 영속화 시킨다
saveOrUpdate() 는 다음을 행한다:
- 만일 객체가 이 세션 내에서 이미 영속화 되어 있을 경우, 아무것도 행하지 않는다
- 만일 그 세션과 연관된 또 다른 객체가 동일한 식별자를 가질 경우, 예외상황을 던진다
- 만일 그 객체가 식별자 프로퍼티를 갖지 않을 경우, 그것을 save() 시킨다
- 만일 객체의 식별자가 새로이 초기화 된 객체에 할당된 값을 가질 경우, 그것을 save() 시킨다
- 만일 객체가 (<version> 또는 <timestamp>에 의해 ) 버전화 되고, version 프로퍼티 값이 새로이 초기화 된 객체에 할당된 것과 동일한 값일 경우, 그것을 save() 시킨다
- 그 밖의 경우 그 객체를 update() 시킨다
그리고 merge() 는 매우 다르다:
- 만일 세션과 현재 연관된 동일한 식별자를 가진 영속 인스턴스가 존재할 경우, 주어진 객체의 상태를 영속 인스턴스 상으로 복사한다
- 만일 세션과 현재 연관된 영속 인스턴스가 존재하지 않을 경우, 데이터베이스로부터 그것을 로드시키려고 시도하거나 새로운 영속 인스턴스를 생성시키려고 시도한다
- 영속 인스턴스가 반환된다
- 주어진 인스턴스는 세션과 연관되지 않고, 그것은 detached 상태에 머무른다.
Session.delete()는 데이터베이스로부터 객체의 상태를 제거할 것이다. 물론 당신의 어플리케이션은 여전히 detached 객체에 대한 참조를 보관할 것이다. 영속 인스턴스를 transient로 만드는 것으로서 delete()를 생각하는 것이 최상이다.
sess.delete(cat);
당신은 foreign 키 컨스트레인트 위반들의 위험성 없이 당신이 좋아하는 어떤 순서로 객체들을 삭제할 수도 있다. 잘못된 순서로 객체들을 삭제함으로써 컨스트레인트를 위배할 가능성이 여전히 있다. 예를 들어, 당신이 부모를 삭제하지만 그 자식들을 삭제하는 것을 잊은 경우.
10.9. 두 개의 다른 데이터저장소들 사이에 객체들을 복제하기
영속 인스턴스들의 그래프를 취하고 식별자 값들을 다시 생성시키지 않고서 그것들을 다른 저장소 속에 영속화 시키는 것이 가능한 것은 종종 유용하다.
// 하나의 데이터베이스로부터 cat을 검색한다
Session session1 = factory1.openSession();
Transaction tx1 = session1.beginTransaction();
Cat cat = session1.get(Cat.class, catId);
tx1.commit();
session1.close();
// 두 번째 데이터베이스와 일치시킨다
Session session2 = factory2.openSession();
Transaction tx2 = session2.beginTransaction();
session2.replicate(cat, ReplicationMode.LATEST_VERSION);
tx2.commit();
session2.close();
ReplicationMode 는 replicate()가 데이터베이스 내에 있는 기존의 행들과의 충돌을 어떻게 처리할 것인지를 결정한다.
- ReplicationMode.IGNORE - 동일한 식별자를 가진 기존 데이터베이스 행이 존재할 경우에 그 객체를 무시한다
- ReplicationMode.OVERWRITE - 동일한 식별자를 가진 어떤 기존의 데이터베이스 행을 덮어 쓴다
- ReplicationMode.EXCEPTION - 만일 동일한 식별자를 가진 기존 데이터베이스 행이 존재할 경우에 예외상황을 던진다
- ReplicationMode.LATEST_VERSION - 행의 버전 번호가 객체의 버전 번호 보다 이전의 것이면 덮어쓰고, 그 밖의 경우에 그 객체를 무시한다
이 특징의 쓰임새들은 다른 데이터베이스 인스턴스들 속으로 입력된 데이터 일치시키기, 제품 업그레이드 동안에 시스템 구성 정보 업데이트 하기, non-ACID 트랜잭션들 동안에 행해진 변경들을 롤백시키기 등을 포함한다.
10.10. Session을 flush 시키기
시간에 따라 Session 은 JDBC 커넥션의 상태와 메모리 내에 보관된 객체들의 상태를 동기화 시키는데 필요한 SQL 문장들을 실행시킬 것이다. 이 프로세스 flush 는 다음 시점들에서 디폴트로 발생한다
- 몇몇 질의가 실행되기 전에
- org.hibernate.Transaction.commit() 에서
- Session.flush() 에서
SQL 문장들은 다음 순서로 실행 명령이 내려진다
- 대응하는 객체들이 Session.save()를 사용하여 저장되었던 것과 같은 순서로, 모든 엔티티가 insert 된다
- 모든 엔티티가 업데이트 된다
- 모든 콜렉션이 삭제된다
- 모든 콜렉션 요소가 삭제되고, 업데이트 되고, 그리고 insert 된다
- 모든 콜렉션이 insert 된다
- 대응하는 객체들이 Session.delete()를 사용하여 삭제되었던 것과 같은 순서로, 모든 엔티티가 삭제된다
(예외는 객체들이 저장될 때 native ID 생성을 사용하는 객체들이 insert 되는 것이다.)
당신이 명시적으로 flush() 시킬 때를 제외하면, Session 이 JDBC 호출들을 실행할 때 그것들이 실행되는 순서만을 절대적으로 보장하지는 않는다. 하지만 Hibernate는 Query.list(..)가 실효성이 없는 데이터를 결코 반환하지 않을 것이거나; 그것들이 잘못된 데이터도 반환하지 않을 것임을 보장한다.
flush가 너무 자주 발생하지 않도록 디폴트 행위를 변경하는 것이 가능하다. FlushMode 클래스는 세 개의 다른 모드들을 정의한다: 커밋 시에(그리고 Hibernate Transaction API가 사용될 때에만) 유일한 flush 모드, 설명된 루틴을 사용하는 자동적인 flush 모드, 또는 flush()가 명시적으로 호출되지 않는 한 결과 flush 시키지 않는 모드. 마지막 모드는 오래 동안 실행되는 작업 단위에 대해 유용하고, 여기서 Session은 열려진채로 유지되고 오랜 시간 동안 연결이 해제된 채로 유지된다. (11.3.2절, “긴 세션과 자동적인 버전화”를 보라).
sess = sf.openSession();
Transaction tx = sess.beginTransaction();
sess.setFlushMode(FlushMode.COMMIT); // 질의들이 실효성 없는 상태를 반환하는 것을 허용한다
Cat izi = (Cat) sess.load(Cat.class, id);
izi.setName(iznizi);
// 실효성 없는 데이터를 반환할 수도 있다
sess.find("from Cat as cat left outer join cat.kittens kitten");
// izi에 대한 변경은 flush 되지 않는다!
...
tx.commit(); // flush가 발생한다
flush 동안에, 예외상황이 발생할 수도 있다(예를 들면. 만일 DML 오퍼레이션이 컨스트레인트를 위반할 경우). 예외상황 처리는 Hibernatem의 트랜잭션 특징에 관한 어떤 이해를 수반하며, 우리는 11장, 트랜잭션들과 동시성에서 논의한다.
10.11. Transitive persistence
특히 당신이 연관된 객체들의 그래프를 다룰 경우에, 개별 객체들을 저장하고, 삭제하거나, 재첨부시키는 것이 꽤 귀찮다. 공통된 경우는 부모/자식 관계이다. 다음 예제를 검토하자:
만일 부모/자식 관계 속에 있는 자식들이 value 타입일 경우(예를 들면. 주소들 또는 문자열들을 가진 콜렉션), 그것들의 생명주기는 부모에 의존할 것이고 더 이상의 액션은 상태 변경들에 대해 편리한 "케스케이딩"에 필요하지 않을 것이다. 만일 부모가 저장될 때, value-타입의 자식 객체들도 마찬가지로 저장되고, 부모가 삭제될 때, 자식이 삭제될 것이다. 이것은 심지어 콜렉션으로부터 하나의 자식을 제거하는 것과 같은 오퍼레이션들에 대해서도 동작한다; Hibernate는 이것을 검출하고, value-타입의 객체들이 참조를 공유할 수 없으므로, 데이터베이스로부터 그 자식을 삭제시킨다.
이제 value 타입이 아닌 엔티티들인 부모와 자식 객체들을 가진 동일한 시나리오를 검토하자(예를 들면. 카테고리들과 아이템들, 또는 부모 고양이나 자식 고양이). 엔티티들은 그것들 자신의 생명주기를 갖고, 공유된 참조들을 지원하고(따라서 콜렉션으로부터 엔티티를 제거하는 것은 그것이 삭제될 수 있음을 의미하지 않는다), 그리고 디폴트로 하나의 엔티티로부터 어떤 다른 연관된 엔티티들로의 상태의 케스케이딩이 존재하지 않는다. Hibernate는 디폴트로 도달가능성에 의한 영속성(persistence by reachability)을 구현하지 않는다.
- persist(), merge(), saveOrUpdate(), delete(), lock(), refresh(), evict(), replicate()를 포함하는- Hibernate 세션에 대한 각각의 기본 오퍼레이션에 대해 대응하는 케스케이딩 스타일이 존재한다. 케스케이드 스타일들 각각은 create, merge, save-update, delete, lock, refresh, evict, replicate로 명명된다. 만일 당신이 오퍼레이션이 하나의 연관에 따라 케스케이딩되는 것을 원할 경우, 당신은 매핑 문서 내에 그것을 지시해야 한다. 예를 들면:
<one-to-one name="person" cascade="create"/>
케스케이딩 스타일들은 결합될 수도 있다:
<one-to-one name="person" cascade="create,delete,lock"/>
당신은 모든 오퍼레이션들이 그 연관에 따라 케스케이드되어야 함을 지정하는데 cascade="all"을 사용할 수도 있다. 디폴트 cascade="none" 은 오퍼레이션들이 케스케이드 되지 않을 것임을 지정한다.
특정한 케스케이드 스타일, delete-orphan은 오직 one-to-many 연관들에만 적용되고, delete() 오퍼레이션이 그 연관으로부터 제거되는 임의의 자식에게 적용되어야 함을 나타낸다.
권장사항들:
- <many-to-one> 또는 <many-to-many> 연관에 대해 케스케이드를 가능하게 하는 것은 의미가 없다. 케스케이드는 <one-to-one> 연관과 <one-to-many> 연관에 대해 자주 유용하다.
- 만일 자식 객체들의 수명이 그 부모 객체의 수명에 의해 한정될 경우, cascade="all,delete-orphan"을 지정함으로써 그것을 생명 주기 객체로 만들어라.
- 그 밖의 경우, 당신은 케스케이드를 전혀 필요로 하지 않는다. 그러나 만일 당신이 동일한 트랜잭션 내에서 부모와 자식에 대해 자주 함께 작업하게 될 것이라 생각되고, 당신 스스로 어떤 타이핑을 절약하고자 원할 경우, cascade="create,merge,save-update"를 사용하는 것을 고려하라.
cascade="all"을 가진 연관을 매핑시키는 것(단일 값 연관이든 콜렉션이든)은 부모의 저장/업데이트/삭제가 자식 또는 자식들의 저장/업데이트/삭제로 귀결되는 부모/자식 스타일의 관계로 그 연관을 마크한다.
게다가, 영속 부모로부터 자식에 대한 단순한 참조는 자식의 저장/업데이트로 귀결될 것이다. 하지만 이 메타포는 불완전하다. 그것의 부모에 의해 참조되지 않는 자식은 cascade="delete-orphan"로 매핑된 <one-to-many> 연관의 경우를 제외하면, 자동적으로 삭제되지 않는다. 부모/자식 관계에 대한 케스케이딩 오퍼레이션의 정확한 의미는 다음과 같다:
- 만일 부모가 persist()에 전달될 경우, 모든 자식들이 persist()에 전달된다.
- 만일 부모가 merge()에 전달될 경우 모든 자식들이 merge()에 전달된다
- 만일 부모가 save(), update() 또는 saveOrUpdate()에 전달될 경우 모든 자식들이 saveOrUpdate()에 전달된다
- 만일 transient 또는 detached 자식이 영속 부모에 의해 참조될 경우, 그것은 saveOrUpdate()에 전달된다
- 만일 부모가 삭제될 경우, 모든 자식들이 delete()에 전달된다
- 만일 자식이 영속 부모에 의해 참조취소 될 경우, cascade="delete-orphan"이 아닌 한, 특별한 어떤 것도 발생하지 않는다 - 필요할 경우 어플리케이션은 자식을 명시적으로 삭제해야 한다 -, "orphaned(고아)"인 경우 자식이 삭제된다.
Hibernate는 모든 엔티티와 value 타입들을 가진 매우 풍부한 메타-레벨 모형을 필요로 한다. 시간에 따라, 이 모형은 어플리케이션 그 자체에 매우 유용하다. 예를 들어, 어플리케이션은 어느 객체들이 복사되어야 하는지(예를 들면 가변적인 value 타입들) 그리고 어느 것이 복사되지 말아야 하는지(예를 들면, 불변의 value 타입들과 가능한 연관된 엔티티들)를 인지하는 "스마트" deep-copy 알고리즘을 구현하는데 Hibernate의 메타데이터를 사용할 수도 있다.
Hibernate는 ClassMetadata 인터페이스와 CollectionMetadata 인터페이스 그리고 Type 계층구조를 통해 메타데이터를 노출시킨다. 메타데이터 인터페이스의 인스턴스들은 SessionFactory로부터 얻어질 수도 있다.
Cat fritz = ......;
ClassMetadata catMeta = sessionfactory.getClassMetadata(Cat.class);
Object[] propertyValues = catMeta.getPropertyValues(fritz);
String[] propertyNames = catMeta.getPropertyNames();
Type[] propertyTypes = catMeta.getPropertyTypes();
// 콜렉션들이나 연관들이 아닌 모든 프로퍼티들을 가진 하나의 Map을 얻는다
Map namedValues = new HashMap();
for ( int i=0; i<propertyNames.length; i++ ) {
if ( !propertyTypes[i].isEntityType() && !propertyTypes[i].isCollectionType() ) {
namedValues.put( propertyNames[i], propertyValues[i] );
}
}
Hibernate와 동시성 제어에 대한 가장 중요한 점은 그것이 이해하기가 매우 쉽다는 점이다. Hibernate는 어떤 추가적인 잠금 행위 없이 JDBC 커넥션들과 JTA 리소스들을 직접 사용한다. 우리는 당신의 데이터베이스 관리 시스템의 JDBC, ANSI, 그리고 트랜잭션 격리 명세에 약간의 시간을 할애하는 것을 매우 권장한다. Hibernate는 오직 자동적인 버전화를 추가하지만 메모리 내에서 객체들을 잠그지 않거나 당신의 데이터베이스 트랜잭션의 격리 레벨을 변경시키지 않는다. 기본적으로, 당신이 당신의 데이터베이스 리소스들에 직접 JDBC(또는 JTA/CMT)를 사용하는 것처럼 Hibernate를 사용하라.
하지만, 자동적인 버전화에 덧붙여, Hibernate는 또한 SELECT FOR UPDATE 구문을 사용하는, 행들에 대한 pessimistic 잠금을 위한 (마이너) API를 제공한다. 이 API는 이 장의 뒷 부분에서 논의된다.
우리는 Configuration, SessionFactory, Session 알갱이를 가진 Hibernate에서의 동시성 제어 뿐만 아니라 데이터베이스 트랜잭션과 긴 어플리케이션 트랜잭션에 대한 논의를 시작한다.
SessionFactory 는 모든 어플리케이션 쓰레드들에 의해 공유되도록 고안된 생성에 비용이 드는, 쓰레드안전(threadsafe) 객체이다. 그것은 대개 어플리케이션 시작 시에 Configuration 인스턴스로부터 한번 생성된다.
Session 은 단일 비지니스 프로세스, 하나의 작업 단위를 위해 한번만 사용되고 나서 폐기될 비용이 들지 않는, 쓰레드 안전하지 않은 객체이다. Session은 그것이 필요하지 않으면 JDBC Connection (또는 a Datasource)를 얻지 않을 것이어서, 당신이 특별한 요청을 제공할 필요가 있을 것이라고 확신하지 않을 경우에 당신은 Session을 안전하게 열고 닫을 수 있다. (이것은 당신이 요청 인터셉션을 사용하여 다음 패턴들 중 어떤 것을 구현하자 마자 중요하게 된다.)
이 그림을 완성하기 위해 당신은 또한 데이터베이스 트랜재션들에 대해 생각해야 한다. 데이터베이스 트랜잭션은 데이터베이스에서 잠금 투쟁을 줄이기 위해 가능한 짧아야 한다. 긴 데이터베이스 트랜잭션들은 당신의 어플리케이션이 고도의 동시성 로드를 높이는 것을 방지할 것이다.
작업 단위의 영역은 무엇인가? 하나의 Hibernate Session은 몇몇 데이터베이스 트랜잭션들에 걸칠 수 있는가 또는 이것은 영역들의 one-to-one 관계인가? 당신은 언제 Session을 열고 닫는가 그리고 당신은 데이터베이스 트랜잭션 경계들을 어떻게 한정하는가?
첫번째로, session-per-operation anti-패턴을 사용하지 말라. 즉, 단일 쓰레드 내에서 모든 간단한 데이터베이스 호출에 대해 Session을 열고 닫지 말라! 물론 동일한 것은 데이터베이스 트랜잭션들에 대해서도 참이다. 어플리케이션 내의 데이터베이스 호출들은 계획된 순서를사용하여 행해지며, 그것들은 자동적인 작업 단위 속으로 그룹지워진다. (이것은 또한 모든 하나의 SQL 문장이 어플리케이션 내에서 무용지물이 된 후에 자동-커밋됨을 의미하고, 이 모드는 SQL 콘솔 작업을 돕도록 고안되었음을 노트하라. Hibernate는 즉시 자동-커밋 모드를 사용 불가능하게 하거나, 어플리케이션 서버가 그렇게 하도록 기대한다.)
다중 사용자 클라이언트/서버 어플리케이션에서 가장 공통된 패턴은 session-per-request이다. 이 모형에서, 클라이언트로부터의 요청은 (Hibernate 영속 계층이 실행되는) 서버로 전송되고, 새로운 Hibernate Session이 열려지고, 모든 데이터베이스 오퍼레이션들이 이 작업 단위 내에서 실행된다. 일단 그 작업이 완료되면(그리고 클라이언트에 대한 응답이 준비되었다면), 세션은 flush되고 닫혀진다. 당신은 또한 당신이 Session을 열고 닫을 때 그것을 시작하고 커밋시켜서 클라이언트 요청에 서비스하는데 한 개의 데이터베이스 트랜잭션을 사용하게 될 것이다. 두 가지 사이의 관계는 일대일 대응이고 이 모형은 많은 어플리케이션들에서 완전히 적합하다.
도전점은 구현에 놓여있다: Session과 트랜잭션이 정확하게 시작되고 끝나야 할 뿐만 아니라, 그것들은 또한 데이터 액세스 오퍼레이션들에 대해 액세스 가능해야 한다. 작업의 단위에 대한 경계 구분은 요청이 서버에 도착하고 응답이 전송되기 전에 실행되는 인터셉터(예를 들면 ServletFilter)를 사용하여 이상적으로 구현된다. 우리는 ThreadLocal 변수를 사용하여 Session을 요청에 서비스하는 쓰레드에 바인드 시키는 것을 권장한다. 이것은 이 쓰레드 내에서 실행되는 모든 모드 내에서 (static 변수에 접근하는 것처럼) 쉽게 접근을 허용해준다. 당신이 선택하는 데이터베이스 트랜잭션 경계 구분 메커니즘에 따라, 당신은 또한 ThreadLocal 변수 내에 트랜잭션 컨텍스트를 유지할 수도 있다. 이것을 위한 구현 패턴들은 뷰 내의 ThreadLocal Session 및 Open Session(ThreadLocal Session 및 Open Session in View)로 알려져 있다. 당신은 이것을 구현하기 위해 이 문서의 앞에서 보여졌던 HibernateUtil helper 클래스를 쉽게 확장할 수 있다. 물론 당신은 인터셉터를 구현하고 그것을 당신의 환경에 설정하는 방법을 찾아야 한다. 팁들과 예제들은 Hibernate 웹 사이트를 보라.
session-per-request 패턴은 당신이 작업 단위를 설계하는데 사용할 수 있는 유일한 유용한 개념이 아니다. 많은 비지니스 프로세스들은 사용자와 데이터베이스 액세스들을 중재하는 전체 일련의 상호작용들을 필요로 한다. 웹과 엔터프라이즈 어플리케이션에서 데이터베이스 트랜잭션이 사용자 상호작용에 걸치는 것은 수용되지 않는다. 다음 예제를 검토하자:
- 대화의 첫 번째 화면이 열리고, 사용자에 의해 전송된 데이터는 특정 Session과 데이터베이스 트랜잭션 속에 로드되었다. 사용자는 객체들을 변경시키는 것이 자유롭다.
- 사용자는 5분 후에 "저장"을 클릭하고 그의 변경들이 영속화 되기를 기대한다; 그는 또한 그가 이 정보를 편집하는 유일한 개인이고 변경 충돌이 발생하지 않기를 기대한다.
우리는 사용자의 관점에서, 이것을 작업 단위, 장기간 실행되는 어플리케이션 트랜잭션이라고 명명한다. 당신의 어플리케이션에서 당신이 이것을 어떻게 구현할 수 있는 많은 방법들이 존재한다.
첫 번째 있는 그대로의 구현은 사용자가 생각하는동안 동시성 변경을 방지하기 위해, 그리고 격리와 atomicity(원자 단위성)을 보장하기 위해 데이터베이스에 의해 억제되는 잠금으로 Session과 데이터베이스 트랜잭션을 유지할 수도 있다. 이것은 물론 anti-패턴이다. 왜냐하면 잠금 투쟁은 어플리케이션이 동시 사용자들의 수를 높이는 것을 허용하지 않을 것이기 때문이다.
명료하게, 우리는 어플리케이션 트랜잭션을 구현하는데 몇몇 데이터베이스 트랜잭션들을 사용해야 한다. 이 경우에, 비지니스 프로세스들의 격리를 유지하는 것은 어플리케이션 티어의 부분적인 책임이 된다. 단일 어플리케이션 트랜잭션은 대개 여러 개의 데이터베이스 트랜잭션들에 걸친다. 그것은 이들 데이터베이스 트랜잭션들 중 하나(마지막 트랜잭션) 만이 업데이트된 데이터를 저장하고, 모든 다른 트랜잭션들은 단순히 데이터를 읽는다(예를 들면, 몇몇 요청/응답 주기에 걸치는 마법사 스타일의 대화 상자에서). 특히 당신이 Hibernate의 특징들을 사용할 경우에 , 소리나는 것보다 이것은 구현이 더 쉽다:
- 자동적인 버전화 - Hibernate는 당신을 위해 자동적인 optimistic 동시성 제어를 행할 수 있고, 그것은 사용자가 생각하는 시간 동안 동시적인 변경이 발생했는지를 자동적으로 검출할 수 있다.
- Detached 객체들 - 만일 당신이 이미 논의된 session-per-request 패턴을 사용하고자 결정한 경우, 모든 로드된 인스턴스들은 사용자가 생각하는 시간 동안 detached 상태에 있을 것이고, 그 패턴은 session-per-request-with-detached-objects(detached-객체들을 가진 요청 당 세션)으로 명명된다. 자동적인 버전화는 동시성 변경들을 격리시키는데 사용된다.
- Long Session - Hibernate Session은 데이터베이스 트랜잭션이 커밋된 후에 기본 JDBC 커넥션이 연결 해제될 수도 있고, 새로운 클라이언트 요청이 발생할 때 다시 연결될 수 있다. 이 패턴은 session-per-application-transaction(어플리케이션 트랜잭션 당 세션)으로 알려져 있고 재첨부를 불필요하게 만든다. 자동적인 버전화는 동시성 변경들을 격리시키는데 사용된다.
session-per-request-with-detached-objects 과 session-per-application-transaction 양자는 장점들과 단점들을 갖는데, 우리는 이 장의 뒷 부분에서 optimistic 동시성 제어 단락에서 그것들을 논의한다.
어플리케이션은 두 개의 다른 Session들 내에 있는 동일한 영속 상태에 동시에 접근할 수도 있다. 하지만 영속 클래스의 인스턴스는 두 개의 Session 인스턴스들 사이에 결코 공유되지 않는다. 그러므로 identity에 대한 두 개의 다른 개념들이 존재한다:
- 데이터베이스 Identity
- foo.getId().equals( bar.getId() )
- JVM Identity
- foo==bar
그때 (예를 들어 의 영역에서) 특정 Session에 첨부된 객체들의 경우 두 개의 개념들은 동등한 것이고, 데이터베이스 identity에 대한 JVM identity는 Hibernate에 의해 보장된다. 하지만, 어플리케이션이 두 개의 다른 세션들 내에 있는 "동일한" (영속 identity) 비지니스 객체에 동시에 접근하는 동안, 두 개의 인스턴스들은 실제로 "다르다"(JVM identity). 충돌들은 flush/커밋 시에 (자동적인 버전화)를 사용하여, optimistic 접근법을 사용하여 해결된다.
이 접근법은 Hibernate와 데이터베이스가 동시성에 대해 걱장하지 않도록 해준다; 그것은 또한 최상의 scalability를 제공한다. 왜냐하면 단일 쓰레드-작업 단위 내에서 identity의 보장은 단지 비용이 드는 잠금이나 다른 동기화 수단들을 필요로 하지 않기 때문이다. 어플리케이션은 그것이 Session 당 단일 쓰레드를 강제하는 한, 어떤 비지니스 객체에 대해 결코 동기화 시킬 필요가 없다. Session 내에서 어플리케이션은 객체들을 비교하는데 ==를 안전하게 사용할 수가 있다.
하지만, Session 외부에서 ==를 사용하는 어플리케이션은 예기치 않은 결과들을 볼 수도 있다. 이것은 어떤 예기치 않은 장소들에서, 예를 들어 당신이 두 개의 detached 인스턴스들을 동일한 Set 내에 집어넣을 경우에 발생할 수도 있다. 둘 다 동일한 데이터베이스 identity를 가질 수 있지만(예를 들어 그것들은 동일한 행을 표현한다), JVM identity는 정의상 인스턴스들이 detached 상태에 있음을 보장하지 않는다. 개발자는 영속 클래스들내에 있는 equals() 메소드와 hashCode() 메소드를 오버라이드 시켜야 하고 객체 equality에 대한 그 자신의 개념을 구현해야 한다. 하나의 경고가 존재한다: equality를 구현하는데 데이터베이스 identifier를 결코 사용하지 말고, 비지니스 키, 유일한, 대개 불변의 속성들의 조합을 사용하라. 데이터베이스 identifier는 만일 transient 객체가 영속화되는 경우에 변경될 것이다. 만일 transient 인스턴스가(대개 detached 인스턴스들과 함께) Set 내에 보관되는 경우에 hashcode 변경은 Set의 계약을 파기시킨다. 비지니스 키들에 대한 속성들은 데이터베이스 프라이머리 키들 만큼 안정적이어서는 안되고, 당신은 오직 객체들이 동일한 Set 내에 있는 한에서 안정성을 보장해야만 한다. 이 쟁점에 관한 논의에 대한 더 많은 것을 Hibernate 웹 사이트를 보라. 또한 이것이 Hibernate 쟁점이 아니지만, 단지 자바 객체 identity와 equality가 구현되어야 하는 방법임을 노트하라.
안티-패턴들 session-per-user-session 또는 session-per-application 을 결코 사용하지 말라(물론 이 규칙에 대한 드문 예외상황들이 존재한다). 다음 쟁점들 중 몇몇이 또한 권장되는 패턴들로 나타남을 노트하고, 설계 결정을 내리기 전에 내포된 의미들을 확실히 이해하라:
- Session 은 쓰레드-안전하지 않다. HTTP 요청들, 세션 빈즈, 또는 Swing worker들처럼 동시에 작업하는 것으로 가정되는 것들은 Session 인스턴스가 공유될 경우에 경쟁 조건들을 발생시킬 것이다. 만일 당신이 당신의 HttpSession 내에 Hibernate Session을 유지시키는 경우(나중에 논의됨), 당신은 당신의 Http 세션에 대한 접근을 동기화 시키는 것을 고려해야 한다. 그 밖의 경우, 충분히 빠르게 reload를 클릭하는 사용자는 두 개의 동시적으로 실행되는 쓰레드들에서 동일한 Session을 사용할 수도 있다.
- Hibernate에 의해 던져진 예외상황은 당신이 당신의 데이터베이스 트랜잭션을 롤백 시키고 즉시 Session을 닫아야 함을 의미한다(나중에 상세히 논의됨). 만일 당신의 Session이 어플리케이션에 바인드 되어 있는 경우, 당신은 어플리케이션을 중지시켜야 한다. 데이터베이스 트랜잭션 롤백은 당신의 비지니스 객체들을 그것들이 트랜잭션의 시작 시에 머물렀던 상태로 집어넣지 않는다. 이것은 데이터베이스 상태와 비지니스 객체들이 동기화를 벗어남을 의미한다. 대개 이것은 문제가 아니다. 왜냐하면 예외상황들은 회복가능한 것이 아니고 당신이 어떻게든 롤백 후에 시작해야 하기 때문이다.
- Session 은 (Hibernate에 의해 dirty 상태로 관찰되었거나 체크된) 영속 상태에 있는 모든 객체를 캐시 시킨다. 이것은 당신이 오랜 시간 동안 그것을 열린 채로 유지하거나 단순히 너무 많은 데이터를 로드시킬 경우에, 당신이 OutOfMemoryException을 얻기 전까지, 그것이 끝없이 설장한다는 점을 의미한다. 이것에 대한 하나의 해결책은 Session 캐시를 관리하기 위해 clear()와 evict()를 호출하는 것이지만, 당신이 대용량 데이터 오퍼레이션들을 필요로 하는 경우에 당신은 대개 내장 프로시저를 고려해야 할 것이다. 몇몇 해결책들이 13 장, 배치 처리에 보여져 있다. 사용자 세션 동안에 Session을 열려진 채로 유지하는 것은 또한 실효성이 떨어진 데이터에 대한 높은 확률을 의미한다.
데이터베이스 (또는 시스템) 트랜잭션 경계들이 항상 필수적이다. 데이터베이스와의 통신은 데이터베이스 트랜잭션의 외부에서 발생할 수 없다(이것은 자동-커밋 모드로 사용되는 많은 개발자들에게는 혼동스러워 보인다). 항상 심지어 읽기 전용 오퍼레이션들에 대해서도 명료한 트랜잭션 경계들을 사용하라. 당신의 격리 레벨과 데이터베이스 가용성들에 따라, 이것은 필요하지 않을 수 있지만, 만일 당신이 항상 트랜잭션들을 명시적으로 경계 설정할 경우에는 하강하는 결점들이 존재하지 않는다.
Hibernate 어플리케이션은 관리되지 않는 환경(예를 들면. 스탠드얼론, 간단히 웹 어플리케이션들 또는 Swing 어플리케이션들)과 관리되는 J2EE 환경에서 실행될 수 있다. 관리되지 않는 환경에서, Hibernate는 대개 그것 자신의 데이터베이스 커넥션 풀에 대한 책임이 있다. 어플리케이션 개발자는 트랜잭션 경계들을 손수 설정해야 한다. 달리 말해, 데이터베이스 트랜잭션들 자체를 시작하고, 커미시키거나 롤백시켜야 한다. 관리되는 환경은 예를 들어 EJB 세션 빈즈의 배치 디스크립터 속에 선언적으로 정의된 트랜잭션 어셈블리를 가진, 컨테이너에 의해-관리되는 트랜잭션들을 제공한다. 그때 프로그램 상의 트랜잭션 경계 설정은 더 이상 필요하지 않다. 심지어 Session을 flush 시키는 것이 자동적으로 행해진다.
하지만, 당신의 영속 계층이 이식성을 유지하게끔 자주 희망된다. Hibernate는 당신의 배치 환경의 고유한 트랜잭션 시스템 속으로 변환되는 Transaction으로 명명되는 wrapper API 를 제공한다. 이 API는 실제로 옵션이지만 우리는 당신이 CMT session bean 속에 있지 않는 한 그것의 사용을 강력하게 권장한다.
대개 Session 종료는 네 개의 구분되는 단계들을 수반한다:
- 세션을 flush 시킨다
- 트랜잭션을 커밋 시킨다
- 세션을 닫는다
- 예외상황들을 처리한다
세션을 flush 시키는 것은 앞서 논의되었고, 우리는 이제 관리되는 환경과 관리되지 않는 환경 양자에서 트랜잭션 경계 설정과 예외상황을 더 자세히 살펴볼 것이다.
만일 Hibernate 영속 계층이 관리되지 않는(non-managed) 환경에서 실행될 경우, 데이터베이스 커넥션들은 대개 Hibernate의 풀링 메커니즘에 의해 처리된다. session/transaction 처리 관용구는 다음과 같이 보여진다:
//Non-managed environment idiom
Session sess = factory.openSession();
Transaction tx = null;
try {
tx = sess.beginTransaction();
// 어떤 작업을 행한다
...
tx.commit();
}
catch (RuntimeException e) {
if (tx != null) tx.rollback();
throw e; // 또는 오류 메시지를 디스플레이 한다
}
finally {
sess.close();
}
당신은 Session을 명시적으로 flush()시키지 말아야 한다 - commit()에 대한 호출은 동기화를 자동적으로 트리거 시킨다.
close()에 대한 호출은 세션의 종료를 마크한다. close()의 주요 함축은 JDBC 커넥션이 그 세션에 의해 포기될 것이라는 점이다.
이 자바 코드는 이식성이 있고 관리되지 않는 환경과 JTA 환경 양자에서 실행된다.
당신은 통상의 어플리케이션에서 비지니스 코드 속에 이 관용구를 결코 보게 되지 않을 것이다; 치명적인(시스템) 예외상황들은 항상 "상단"에서 잡아 내어야 한다. 달리 말해, Hibernate를 실행하는 코드가 (영속 계층에서) 호출되고 RuntimeException을 처리하는 (그리고 대개 오직 제거하고 빠져나갈 수 있는) 코드는 다른 계층들 속에 있다. 이것은 당신 자신이 설계하는 도전점일 수 있고 당신은 J2EE/EJB 컨테이너 서비스들이 이용 가능할 때마다 J2EE/EJB 컨테이너 서비스들을 사용할 것이다. 예외상황 처리는 이 장의 뒷부분에서 논의된다.
당신은 (디폴트인) org.hibernate.transaction.JDBCTransactionFactory를 선택해야 함을 노트하라.
만일 당신의 영속 계층이 어플리케이션 서버에서(예를 들어, EJB 세션 빈즈 이면에서) 실행될 경우, Hibernate에 의해 획득된 모든 데이터소스 커넥션은 자동적으로 전역 JTA 트랜잭션의 부분일 것이다. Hibernate는 이 통합을 위한 두 개의 방도들을 제공한다.
만일 당신이 bean-managed transactions (BMT)을 사용할 경우 Hibernate는 당신이 Transaction API를 사용할 경우에 BMT 트랜잭션을 시작하고 종료하도록 어플리케이션 서버에게 알려줄 것이다. 따라서 트랜잭션 관리 코드는 non-managed 환경과 동일하다.
// BMT idiom
Session sess = factory.openSession();
Transaction tx = null;
try {
tx = sess.beginTransaction();
// 어떤 작업을 행한다
...
tx.commit();
}
catch (RuntimeException e) {
if (tx != null) tx.rollback();
throw e; // 또는 오류 메시지를 디스플레이 한다
}
finally {
sess.close();
}
CMT의 경우, 트랜잭션 관할[경계 설정]은 프로그램 상이 아닌, session bean 배치 디스크립터들 속에서 행해진다. 당신이 hibernate.transaction.flush_before_completion, hibernate.transaction.auto_close_session 프로퍼티들을 true로 설정하지 않는 한, 당신은 여전히 당신 스스로 Session을 flush하고 닫아야 한다. 그런 다음 Hibernate는 당신을 위핸 Session을 자동적으로 flush 시키고 닫을 것이다. 유일하게 남은 것은 예외상황이 일어날 때 트랜잭션을 롤백 시키는 것이다. 다행히 CMT bean에서, 이것이 자동적으로 일어난다. 왜냐하면 session bean 메소드에 의해 던져진 처리되지 않은 RuntimeException이 전역 트랜잭션을 롤백시키도록 컨테이너에게 알려주기 때문이다. 이것은 당신이 CMT에서 Hibernate Transaction API를 사용할 필요가 전혀 없음을 의미한다.
당신이 BMT session bean에서 org.hibernate.transaction.JTATransactionFactory를 선택해야하고,당신이 Hibernate의 트랜잭션 팩토리를 구성할 때 CMT session bean에서 org.hibernate.transaction.CMTTransactionFactory를 선택해야 함을 노트하라. 또한 org.hibernate.transaction.manager_lookup_class를 설정하는 것을 염두에 두라.
만일 당신이 CMT 환경에서 작업하고 있고, 세션의 자동적인 flushing과 닫기를 사용할 경우, 당신은 또한 당신의 코드의 다른 부분에서 동일한 세션을 사용하고자 원할 수도 있다. 일반적으로 관리되지 않는 환경에서 당신은 세션을 소유하는데 ThreadLocal 변수를 사용할 것이지만, 한 개의 EJB 메소드는 다른 쓰레드(예를 들면 또 다른 세션 빈을 호출하는 세션 빈) 내에서 실행될 수도 있다. 만일 당신이 당신의 Session 인스턴스를 전달하는 것을 고민하고 싶지 않다면, SessionFactory는 JTA 트랜잭션 컨텍스트에 바인드되어 있는 한 개의 세션을 반환하는 getCurrentSession()메소드를 제공한다. 이것은 Hibernate를 어플리케이션 속으로 통합시키는 가장 손쉬운 방법이다! "current" 세션은 항상 (위의 프로퍼티 설정들에 관계없이) auto-flush와 auto-close를 이용 가능하게 한다. 우리의 session/transaction 관리 idiom은 다음과 같이 감소된다:
// CMT idiom
Session sess = factory.getCurrentSession();
// 어떤 작업을 행한다
...
달리 말해, 당신이 관리 환경에서 행해야 하는 모든 것은 SessionFactory.getCurrentSession()을 호출하고, 당신의 데이터 접근 작업을 행하고, 그리고 나머지는 컨테이너에게 남겨두는 것이다. 트랜잭션 경계들은 당신의 session bean의 배치 디스크립터들 속에서 선언적으로 설정된다. 그 세션의 생명주기는 Hibernate에 의해 완전하게 관리된다.
auto_close_session의 사용에는 한 가지 단서가 존재한다. JTA 명세서의 제약성으로 인해, Hibernate가 scroll() 또는 iterate()에 의해 반환된 어떤 닫혀지지 않은 ScrollableResults 또는 Iterator 인스턴스들을 자동적으로 제거하는 것이 불가능하다. 당신은 finally 블록에서 ScrollableResults.close() 또는 Hibernate.close(Iterator)를 명시적으로 호출하여 기본 데이터베이스 커서를 해제시켜야 한다. (물론 대부분의 어플리케이션들은 CMT 코드에서 scroll() 또는 iterate()를 사용하는 것을 쉽게 피할 수 있다.)
만일 Session이 (어떤 SQLException을 포함하는) 예외상황을 던질 경우, 당신은 데이터베이스 트랜잭션을 즉시 롤백시키고, Session.close()를 호출하고 Session 인스턴스를 폐기시켜야한다. Session의 어떤 메소드들은 그 세션을 일관된 상태로 남겨두지 않을 것이다. Hibernate에 의해 던져진 예외상황은 복구가능한 것으로 취급될 수 없다. 그 Session이 finally 블록 내에서 close()를 호출하여 닫혀지도록 확실히 하라.
Hibernate 영속 계층에서 발생할 수 있는 대부분의 오류들을 포장하는 HibernateException은 체크되지 않은 예외상황이다(그것은 Hibernate의 이전 버전에는 없었다). 우리의 의견으로, 우리는 낮은 계층에서 복구불가능한 예외상황을 캐치하도록 어플리케이션 개발자를 강제하지 않을 것이다. 대부분의 시스템들에서, 체크되지 않은 치명적인 예외상황들은 (예를 들어 더 높은 계층에서) 메소드 호출 스택의 첫 번째 프레임들 중 하나 속에서 처리되고, 한 개의 오류 메시지가 어플리케이션 사용자에게 표시된다(또는 어떤 다른 적절한 액션이 취해진다). Hibernate는 또한 HibernateException이 아닌, (예를 들면 버전 체크에서 실효성이 없는 데이터를 검출할 때) 다른 체크되지 않은 예외상황들을 던질 수도 있음을 노트하라. 다시 이것들은 복구가능하지 않고 적절한 액션이 취해져야 한다.
Hibernate는 데이터베이스와 상호작용하는 동안에 던져진 SQLException들을 하나의 JDBCException 속에 포장한다. 사실, Hibernate는 그 예외상황을 JDBCException의 보다 의미있는 서브클래스로 변환하려고 시도할 것이다. 기본 SQLException은 항상 JDBCException.getCause()를 통해 이용 가능하다. Hibernate는 SessionFactory에 첨부된 SQLExceptionConverter를 사용하여 SQLException을 적당한 JDBCException 서브클래스로 변환시킨다. 디폴트로 SQLExceptionConverter는 구성된 dialect에 의해 정의된다; 하지만 또한 맞춤 구현 속에 플러그인 시키는 것이 가능하다(상세한 것은 SQLExceptionConverterFactory 클래스에 관한 javadocs를 보라). 표준 JDBCException 서브타입은 다음과 같다:
- JDBCConnectionException - 기본 JDBC 통신에 대한 오류를 나타낸다.
- SQLGrammarException - 생겨난 SQL에 대한 문법 또는 구문 문제점을 나타낸다.
- ConstraintViolationException - 무결성 제약 위반에 관한 어떤 형식을 나타낸다.
- LockAcquisitionException - 요청된 오퍼레이션을 실행하는데 필수적인 잠금 레벨을 획득하는 오류를 나타낸다.
- GenericJDBCException - 다른 카테고리들 중 어떤 것으로 분류되지 않았던 일반적인 예외상황.
고도의 동시성과 고도의 가용성을 일치시키는 유일한 접근법은 버전화를 가진 optimistic동시성 제어이다. 버전 체킹은 업데이트 충돌을 검출하기 위해(그리고 업데이트 손실을 방지하기 위해) 버전 번호들 또는 timestamp들을 사용한다. Hibernate는 optimistic동시성을 사용하는 어플리케이션 코드를 작성하는데 세 가지 가능한 접근법들을 제공한다. 우리가 보여주는 쓰임새들은 긴 어플리케이션 트랜잭션들의 상황 속에 있지만 버전 체킹 또한 단일 데이터베이스 트랜잭션들에서 업데이트 손실을 방지하는 이점을 갖고 있다.
Hibernate의 많은 도움이 없는 구현에서, 데이터베이스에 대한 각각의 상호작용은 새로운 Session 내에서 일어나고, 개발자는 그것들을 처리하기 전에 데이터베이스로부터 모든 영속 인스턴스들을 다시 로드시키는 책임이 있다. 이 접근법은 어플리케이션 트랜잭션을 확실히 격리시키기 위해 그것 자신의 버전 체킹을 수행하는 것을 어플리케이션에게 강제시킨다. 이 접근법은 최소한 데이터베이스 접근의 관점에서 효율성이 가장 적다. 그것은 엔티티 EJB들과 가장 유사한 접근법이다.
// foo는 이전 Session에 의해 로드된 인스턴스이다
session = factory.openSession();
Transaction t = session.beginTransaction();
int oldVersion = foo.getVersion();
session.load( foo, foo.getKey() ); // 현재 상태를 로드시킨다
if ( oldVersion!=foo.getVersion ) throw new StaleObjectStateException();
foo.setProperty("bar");
t.commit();
session.close();
version 프로퍼티는 <version>을 사용하여 매핑되고, Hibernate는 만일 엔티티가 dirty일 경우 flush 동안에 그것을 자동적으로 증가시킬 것이다.
물론, 당신이 낮은 데이터 동시성 환경에서 작업하고 있고 버전 체킹을 필요로 하지 않을 경우에, 당신은 이 접근법을 사용할 수 도 있고 단지 버전 체크를 생략할 수도 있다. 그 경우에, last commit wins 는 당신의 긴 어플리케이션 트랜잭션들에 대한 디폴트 방도가 될 것이다. 이것이 어플리케이션의 사용자들을 혼동시킬 수 있음을 염두에 두라. 왜냐하면 사용자들은 오류 메시지들 또는 충돌 변경들을 병합시킬 기회 없이 업데이트들 손실을 겪을 수도 있기 때문이다.
명료하게 수작업 버전 체킹은 매우 사소한 환경들에서도 공포적이고 대부분의 어플리케이션들에 대해 실제적이지 않다. 흔히 단일 인스턴스 뿐만 아니라 변경된 객체들의 전체 그래프들이 체크되어야 한다. Hibernate는 설계 패러다임으로서 긴 Session 또는 detached 인스턴스들에 대해 자동적인 버전 체킹을 제공한다.
하나의 Session 인스턴스와 그것의 영속 인스턴스들은 전체 어플리케이션 트랜잭션에 사용된다. Hibernate는 flush 할 때 인스턴스 버전들을 체크하고 만일 동시성 변경이 검출될 경우에 예외상황을 던진다. 이 예외상황을 잡아내고 처리하는 것을 개발자의 몫이다(공통된 옵션들은 변경들을 병합시키거나 또는 쓸모가 없지 않은 데이터로 비지니스 프로세스를 다시 시작하는 기회를 사용자에게 주는 것이다).
Session은 사용자 상호작용을 기다릴 때 어떤 기본 JDBC 커넥션으로부터 연결해제된다. 이 접근법은 데이터베이스 접근의 관점에서 보면 가장 효율적이다. 어플리케이션은 버전 체킹 또는 detached 인스턴스들을 재첨부하는 것에 그 자체 관계할 필요가 없거나 그것은 모든 데이터베이스 트랜잭션에서 인스턴스들을 다시 로드시킬 필요가 없다.
// foo는 Session에 의해 로드된 인스턴스이다
session.reconnect(); // 새로운 JDBC 커넥션을 얻는다
Transaction t = session.beginTransaction();
foo.setProperty("bar");
t.commit(); // 변경을 flush 시키고 버전을 체킹하여 데이터베이스 트랜잭션을 종료시킨다
session.disconnect(); // JDBC 커넥션을 반환한다
foo 객체는 그것이 로드된 Session이 어느 것이였는지를 여전히 알고 있다. Session.reconnect()은 새로운 커넥션을 획득하고(또는 당신이 커넥션을 제공할 수 있다) 그리고 그 세션을 다시 시작시킨다. Session.disconnect() 메소드는 JDBC 커넥션으로부터 세션을 연결해제하고 (당신이 커넥션을 제공하지 않는 한) 그 커넥션을 풀(pool)로 반환할 것이다. 재연결 후에, 당신이 업데이트하고 있는 데이터에 대한 버전 체킹을 강제시키기 위해, 당신은 또 다른 트랜잭션에 의해 업데이트 되었던 어떤 객체들에 대해 LockMode.READ로 Session.lock()을 호출할 수도 있다. 당신은 당신이 업데이트하고 있는 어떤 데이터에 대한 잠금을 필요로 하지 않는다.
만일 사용자가 생각하는시간 동안 Session이 저장되기에 너무 큰 경우에 이 패턴은 문제가 있다. 예를 들어 HttpSession은 가능한 작은 것으로 유지되어야 한다. 또한 Session은 (위임된) 첫 번째 레벨 캐시이고 모든 로드된 객체들을 포함하기 때문에, 우리는 아마 적은 요청/응답 주기들에 대해서만 이 방도를 사용할 수 있다. Session이 곧 실효성이 없는 데이터를 갖게 될 것이므로 이것이 진정으로 권장된다.
또한 당신이 연결해제된 Session을 영속 계층에 가깝게 유지해야함을 노트하라. 달리말해, Session을 보관하는데 EJB stateful session bean을 사용하고 HttpSession 내에 그것을 저장하기 위해서 그것을 웹 계층에 전송하지 말라(또는 그것을 별도의 티어에 직렬화 시키지도 말라).
11.3.3. Detached 객체들과 자동적인 버전화
영속 저장소와의 각각의 상호작용은 새로운 Session에서 일어난다. 하지만 동일한 영속 인스턴스들은 데이터베이스와의 각각의 상호작용에 재사용된다. 어플리케이션은 원래 로드되었던 detached 인스턴스들의 상태를 또 다른 Session 내에서 처리하고 나서 Session.update(), Session.saveOrUpdate(), Session.merge()를 사용하여 그것들을 다시 첨부시킨다.
// foo는 이전 Session에 의해 로드된 인스턴스이다
foo.setProperty("bar");
session = factory.openSession();
Transaction t = session.beginTransaction();
session.saveOrUpdate(foo); // "foo"가 이미 로드되었다면 merge()를 사용하라
t.commit();
session.close();
다시, Hibernate는 flush 동안에 인스턴스 버전들을 체크할 것이고 업데이트 충돌이 발생할 경우에 예외상황을 던질 것이다.
당신은 또한 update()대신에 lock()을 호출할 수도 있고 만일 그 객체가 변경되지 않았음을 당신이 확신하는 경우에 (버전 체킹을 수행하고 모든 캐시들을 무시하는) LockMode.READ를 사용할 수 있다.
11.3.4. 자동적인 버전화를 맞춤화 시키기
당신은 optimistic-lock 매핑 속성을 false로 설정함으로써 특정 프로퍼티들과 콜렉션들에 대한 Hibernate의 자동적인 버전 증가를 불가능하도록 할 수도 있다. 그때 Hibernate는 그 프로퍼티가 dirty 일 경우에 더 이상 버전을 증가시키지 않을 것이다.
리거시 데이터베이스 스키마들은 자주 static이고 변경될 수 없다. 또는 다른 어플리케이션들은 또한 동일한 데이터베이스에 접근하고 버전 번호들 또는 심지어 timestamp들을 처리하는 방법을 모를 수도 있다. 두 경우들에서, 버전화는 테이블 내의 특정 컬럼에 의지할 수 없다. version 또는 timestamp 프로퍼티 매핑 없이 행 내의 모든 필드들에 대한 상태를 비교하여 버전 체크를 강제시키기 위해서, <class> 매핑 속에 optimistic-lock="all"을 표시하라. 만일 Hibernate가 이전 상태와 새로운 상태를 비교할 수 있을 경우에, 예를 들면 당신이 하나의 긴 Session을 사용하고 session-per-request-with-detached-objects을 사용하지 않을 경우 이것은 개념적으로만 동작함을 노트하라.
때때로 행해졌던 변경들이 중첩되지 않는 한 동시적인 변경이 허용될 수 있다. 만일 <class>를 매핑할 때 당신이 optimistic-lock="dirty"을 설정하면, Hibernate는 flush 동안에 dirty 필드들을 비교만 할 것이다.
두 경우들에서, 전용 version/timestamp 컬럼의 경우 또는 full/dirty 필드 비교의 경우, Hibernate는 법전 체크를 실행하고 정보를 업데이트하는데 엔티티 당 (적절한 WHERE 절을 가진) 한 개의 UPDATE 문장을 사용한다. 만일 당신이 연관된 엔티티들에 대한 재첨부를 케스케이드 하는데 transitive 영속을 사용할 경우, Hibernate는 불필요하게 업데이트들을 실행할 수도 있다. 이것은 대개 문제가 아니지만, 심지어 변경들이 detached 인스턴스들에 대해 행해지지 않았을 때에도 데이터베이스 내에서 on update 트리거들이 실행될 수도 있다. 행을 업데이트하기 전에 변경들이 실제로 일어났음을 확인하기 위해 인스턴스를 SELECT 하는 것을 Hibernate에게 강제시키는, <class> 매핑 속에 select-before-update="true"를 설정함으로써 당신은 이 특징을 맞춤화 시킬 수 있다.
사용자들은 잠금 방도에 대해 걱정하는데 많은 시간을 할애하하려고 생각하지 않는다. 대개 JDBC 커넥션들에 대한 격리 레벨을 지정하는 것으로 충분하고 그런 다음 단순히 데이터베이스로 하여금 모든 작업을 행하도록 한다. 하지만 진일보한 사용자들은 때때로 배타적인 pessimistic 잠금들을 얻거나 또는 새로운 트랜잭션의 시작 시에 잠금들을 다시 얻고자 원할 수도 있다.
Hibernate는 결코 메모리 내에 있는 객체들이 아닌 데이터베이스의 잠금 메커니즘을 항상 사용할 것이다!
LockMode 클래스는 Hibernate에 의해 획득될 수 있는 다른 잠금 레벨들을 정의한다. 잠금은 다음 메커니즘들에 의해 얻어진다:
- LockMode.WRITE 는 Hibernate가 한 행을 업데이트 하거나 insert 할 때 자동적으로 획득된다.
- LockMode.UPGRADE 는 SELECT ... FOR UPDATE 구문을 지원하는 데이터베이스 상에서 SELECT ... FOR UPDATE를 사용하여 명시적인 사용자 요청 상에서 얻어질 수 있다.
- LockMode.UPGRADE_NOWAIT 는 오라클에서 SELECT ... FOR UPDATE NOWAIT를 사용하여 명시적인 사용자 요청 상에서 얻어질 수도 있다.
- LockMode.READ 는 Hibernate가 반복 가능한 읽기(Repeatable Read) 또는 Serialization 격리 레벨에서 데이터를 읽어들일 때 자동적으로 얻어질 수도 있다. 명시적인 사용자 요청에 의해 다시 얻어질 수도 있다.
- LockMode.NONE 은 잠금이 없음을 나타낸다. 모든 객체들은 Transaction의 끝에서 이 잠금 모드로 전환된다. update() 또는 saveOrUpdate()에 대한 호출을 통해 세션과 연관된 객체들은 또한 이 잠금 모드로 시작된다.
"명시적인 사용자 요청"은 다음 방법들 중 하나로 표현된다:
- LockMode를 지정한, Session.load()에 대한 호출.
- Session.lock()에 대한 호출.
- Query.setLockMode()에 대한 호출.
만일 Session.load()가 UPGRADE 또는 UPGRADE_NOWAIT 모드로 호출되고 ,요청된 객체가 아직 이 세션에 의해 로드되지 않았다면, 그 객체는 SELECT ... FOR UPDATE를 사용하여 로드된다. 만일 요청된 것이 아닌 다소 제한적인 잠금으로 이미 로드되어 있는 객체에 대해 load()가 호출될 경우, Hibernate는 그 객체에 대해 lock()을 호출한다.
만일 지정된 잠금 모드가 READ, UPGRADE 또는 UPGRADE_NOWAIT 일 경우에 Session.lock() 은 버전 번호 체크를 수행한다.( UPGRADE 또는 UPGRADE_NOWAIT 인 경우에, SELECT ... FOR UPDATE 가 사용된다.)
만일 데이터베이스가 요청된 잠금 모드를 지원하지 않을 경우, (예외상황을 던지는 대신에) Hibernate는 적절한 대체 모드를 사용할 것이다. 이것은 어플리케이션이 이식 가능할 것임을 확실히 해준다.
어플리케이션이 Hibernate 내부에서 발생하는 어떤 이벤트들에 대해 반응하는 것이 자주 유용하다. 이것은 어떤 종류의 일반적인 기능, 그리고 Hibernate의 확장 기능의 구현을 허용해준다.
Interceptor 인터페이스는 영속 객체가 저장되고, 업데이트되고, 삭제되거나 로드되기 전에 영속 객체의 프로퍼티들을 조사하고/하거나 처리하는 것을 어플리케이션에 허용해줌으로써 세션으로부터 어플리케이션으로의 콜백들을 제공한다. 이것에 대한 한 가지 가능한 사용은 감사 정보를 추적하는 것이다. 예를 들어, 다음 Interceptor은 Auditable이 생성될 때 createTimestamp를 자동적으로 설정하고고 Auditable이 업데이트될 때 lastUpdateTimestamp 프로퍼티를 업데이트 한다.
package org.hibernate.test;
import java.io.Serializable;
import java.util.Date;
import java.util.Iterator;
import org.hibernate.Interceptor;
import org.hibernate.type.Type;
public class AuditInterceptor implements Interceptor, Serializable {
private int updates;
private int creates;
public void onDelete(Object entity,
Serializable id,
Object[] state,
String[] propertyNames,
Type[] types) {
// do nothing
}
public boolean onFlushDirty(Object entity,
Serializable id,
Object[] currentState,
Object[] previousState,
String[] propertyNames,
Type[] types) {
if ( entity instanceof Auditable ) {
updates++;
for ( int i=0; i < propertyNames.length; i++ ) {
if ( "lastUpdateTimestamp".equals( propertyNames[i] ) ) {
currentState[i] = new Date();
return true;
}
}
}
return false;
}
public boolean onLoad(Object entity,
Serializable id,
Object[] state,
String[] propertyNames,
Type[] types) {
return false;
}
public boolean onSave(Object entity,
Serializable id,
Object[] state,
String[] propertyNames,
Type[] types) {
if ( entity instanceof Auditable ) {
creates++;
for ( int i=0; i<propertyNames.length; i++ ) {
if ( "createTimestamp".equals( propertyNames[i] ) ) {
state[i] = new Date();
return true;
}
}
}
return false;
}
public void postFlush(Iterator entities) {
System.out.println("Creations: " + creates + ", Updates: " + updates);
}
public void preFlush(Iterator entities) {
updates=0;
creates=0;
}
...
}
인터셉터는 세션이 생성될 때 지정될 것이다.
Session session = sf.openSession( new AuditInterceptor() );
당신은 또한 Configuration을 사용하여 인터셉터를 전역 레벨 상에 설정할 수도 있다:
new Configuration().setInterceptor( new AuditInterceptor() );
당신의 영속 계층에서 특별한 이벤트들에 대해 반응해야 한다면, 당신은 또한 Hibernate3 이벤트 아키텍처를 사용할 수도 있다. 이벤트 시스템은 부가물로 사용될 수 있거나 인터셉터들에 대한 대체물로 사용될 수 있다.
본질적으로 Session 인터페이스의 모든 메소드들은 이벤트와 서로 관련되어 있다. 당신은 LoadEvent, FlushEvent, 등을 갖는다 (정의된 이벤트 타입들의 전체 리스트에 대해서는 XML 구성 파일 DTD 또는 org.hibernate.event 패키지를 참조하라). 하나의 요청이 이들 메소드들 중 하나에 의해 만들어질 때, Hibernate Session은 적절한 이벤트를 생성시키고 그것을 그 타입에 대해 구성된 이벤트 리스너에게 전달한다. 박싱없이, 이들 리스너들은 그들 메소드들이 항상 귀결되는 동일한 프로세싱을 구현한다. 하지만 당신이 리스너 인터페이스들 중 하나에 대한 맞춤화 구현하는 것은 자유롭고(예를 들어 LoadEvent는 LoadEventListener 인터페이스의 등록된 구현에 의해 처리된다), 그 경우에 그들 구현은 Session에 의해 만들어진 임의의 load() 요청들을 처리할 책임이 있을 것이다.
리스너들은 효과적으로 싱글톤(singleton)들로 고려되어야 할 것이다; 이것은 그것들이 요청들 사이에서 공유되고, 따라서 임의의 상태를 인스턴스 변수들로서 저장하지 말아야 함을 의미한다. 하지만 이벤트 객체들 그 자체들은 그것들이 각각의 요청에 대해 유일하므로 처리에 필요한 많은 컨텍스트를 보관한다. 맞춤형 이벤트 리스너들은 또한 어떤 필요한 처리 변수들의 스토리지로서 이벤트의 컨텍스트를 사용할 수도 있다. 컨텍스트는 간단한 맵이지만, 디폴트 리스너들은 컨텍스트 맵을 전혀 사용하지 않아서, 내부적으로 필요한 컨텍스트 변수들을 덮어쓰는 것을 걱정하지 않는다.
맞춤형 리스너는 그것이 편의적인 기저 클래스들(또는 리스너들이 이 용도로 final이 아닌 것으로 선언되므로 Hibernate out-of-the-box에 의해 사용된 디폴트 이벤트 리스너들) 중 하나를 처리하고자 그리고/또는 확장하고자 원하는 이벤트들에 대해 적절한 인터페이스를 구현해야 한다. 맞춤형 리스너들은 Configuration 객체를 통해 프로그램 상으로 등록될 수 있거나, Hibernate 구성 XML 속에 지정될 수 있다 (프로퍼티들을 통한 선언적인 구성은 지원되지 않는다). 다음은 맞춤형 load 이벤트 리스너에 대한 예제이다:
public class MyLoadListener extends DefaultLoadEventListener {
// this is the single method defined by the LoadEventListener interface
public Object onLoad(LoadEvent event, LoadEventListener.LoadType loadType)
throws HibernateException {
if ( !MySecurity.isAuthorized( event.getEntityClassName(), event.getEntityId() ) ) {
throw MySecurityException("Unauthorized access");
}
return super.onLoad(event, loadType);
}
}
당신은 또한 Hibernate에게 디폴트 리스너 대신에 해당 리스너를 사용하도록 알려주는 구성 엔트리를 필요로 한다:
<hibernate-configuration>
<session-factory>
...
<listener type="load" class="MyLoadListener"/>
</session-factory>
</hibernate-configuration>
대신에 당신은 그것을 프로그램 상으로 등록할 수도 있다:
Configuration cfg = new Configuration();
cfg.getSessionEventListenerConfig().setLoadEventListener( new MyLoadListener() );
선언적으로 등록된 리스너들은 인스턴스들을 공유할 수 없다. 만일 동일한 클래스 이름이 여러 <listener/> 요소들에서 사용될 경우, 각각의 참조는 그 클래스에 대한 별도의 인스턴스로 귀결될 것이다. 만일 당신이 리스너 타입들 사이에서 리스너 인스턴스들을 공유할 가용성을 필요로 할 경우 당신은 프로그램 상의 등록 접근법을 사용해야 한다.
왜 인터페이스를 구현하고 컨피그레이션 동안 특정 타입을 지정하는가? 물론 리스너 구현은 여러 개의 이벤트 리스너 인터페이스들을 구현할 수 있다. 등록 동안에 타입을 추가적으로 정의하는 것은 컨피그레이션 동안에 맞춤형 리스너들의 사용 여부를 전환시키는 것을 더 쉽게 해준다.
대개 Hibernate 어플리케이션들에서 선언적인 보안은 session facade 계층에서 관리된다. 이제, Hibernate3는 어떤 액션들이 JACC를 통해 퍼미션을 주어지고, JAAS를 통해 인가되는 것을 허용해준다. 이것은 모든 아키텍처의 상단에 빌드된 옵션 기능이다.
먼저, 당신은 JAAS authorization 사용을 이용 가능하도록 하기 위해 적절한 이벤트 리스터들을 구성해야 한다.
<listener type="pre-delete" class="org.hibernate.secure.JACCPreDeleteEventListener"/>
<listener type="pre-update" class="org.hibernate.secure.JACCPreUpdateEventListener"/>
<listener type="pre-insert" class="org.hibernate.secure.JACCPreInsertEventListener"/>
<listener type="pre-load" class="org.hibernate.secure.JACCPreLoadEventListener"/>
다음으로, 여전히 hibernate.cfg.xml 내에서 퍼미션들을 role들에 바인드 시킨다 :
<grant role="admin" entity-name="User" actions="insert,update,read"/>
<grant role="su" entity-name="User" actions="*"/>
역할(role) 이름들은 당신의 JACC 프로바이더에 의해 이해된 role들이다.
Hibernate를 사용하여 100 000 개의 행들을 삽입시키는 naive 접근법은 다음과 같다:
Session session = sessionFactory.openSession();
Transaction tx = session.beginTransaction();
for ( int i=0; i<100000; i++ ) {
Customer customer = new Customer(.....);
session.save(customer);
}
tx.commit();
session.close();
이것은 50 000번째 행 가까운 곳에서 OutOfMemoryException 상으로 떨어질 것이다. 그것은 Hibernate가 session-level 캐시 속에 모든 새로이 삽입된 Customer 인스턴스들을 캐시시키기 때문이다.
이 장에서 우리는 이 문제를 피하는 방법을 당신에게 보여줄 것이다. 하지만 먼저 당신이 배치 처리를 행할 경우, 당신이 적당한 퍼포먼스를 성취하려고 할 경우에 당신이 JDBC 배치 사용을 가능하게 하는 것은 절대적으로 필요하다. JDBC 배치 사이즈를 적당한 숫자(10-50)로 설정하라.
hibernate.jdbc.batch_size 20
당신은 또한 second-level 캐시를 가진 상호작용이 완전하게 불가능한 프로세스 내에서 이런 종류의 작업을 행하고 싶어할 수도 있다:
hibernate.cache.use_second_level_cache false
새로운 객체들을 영속화 시킬 때, 당신은 first-level 캐시의 사이즈를 제어하기 위해 세션을 정기적으로 flush()시키고 나서 clear() 시켜야 한다.
Session session = sessionFactory.openSession();
Transaction tx = session.beginTransaction();
for ( int i=0; i<100000; i++ ) {
Customer customer = new Customer(.....);
session.save(customer);
if ( i % 20 == 0 ) { //20, JDBC 배치 사이즈와 같다
// insert들의 batch를 flush 시키고 메모리를 해제시킨다:
session.flush();
session.clear();
}
}
tx.commit();
session.close();
데이터 검색과 업데이트의 경우 동일한 아이디어들이 적용된다. 게다가 당신은 많은 데이터 행들을 반환하는 질의들에 대해 서버-측 커서들의 장점을 취하는데 scroll()을 사용할 필요가 있다.
Session session = sessionFactory.openSession();
Transaction tx = session.beginTransaction();
ScrollableResults customers = session.getNamedQuery("GetCustomers")
.setCacheMode(CacheMode.IGNORE)
.scroll(ScrollMode.FORWARD_ONLY);
int count=0;
while ( customers.next() ) {
Customer customer = (Customer) customers.get(0);
customer.updateStuff(...);
if ( ++count % 20 == 0 ) {
// update들의 batch를 flush 시키고 메모리를 해제시킨다:
session.flush();
session.clear();
}
}
tx.commit();
session.close();
이미 논의했듯이, 자동적이고 투명한 객체/관계형 매핑은 객체 상태의 관리에 관계된다. 이것은 객체 상태가 메모리 내에서 이용 가능함을 의미하므로, 데이터베이스 내에서 직접 데이터를 업데이트하거나 삭제하는 것(SQL을 사용하는 UPDATE와 DELETE)는 메모리 내 상태에 영향을 주지 않을 것이다. 하지만 Hibernate는 Hibernate Query Language를 통해 수행되는 대량 SQL-스타일의 UPDATE와 DELETE 문장 실행을 위한 메소드들을 제공한다.(제14장,HQL: Hibernate Query Language).
UPDATE와 DELETE 문장들의 유사-구문은 다음과 같다: ( UPDATE | DELETE ) FROM? ClassName (WHERE WHERE_CONDITIONS)?. 노트할 몇 가지 :
- from-절에서, FROM 키워드는 옵션이다
- from-절 내에 명명된 한 개의 클래스가 오직 존재할 수 있고,alias를 가질 수 없다.
- join들은 (함축적이든 명시적이든) 대량 HQL 질의 속에 지정될 수 없다. 서브-질의들은 where-절 속에서 사용될 수 있다.
- where-절 또한 옵션이다.
하나의 예제로서, 한 개의 HQL UPDATE를 실행하기 위해, Query.executeUpdate() 메소드를 사용하라:
Session session = sessionFactory.openSession();
Transaction tx = session.beginTransaction();
String hqlUpdate = "update Customer set name = :newName where name = :oldName";
int updatedEntities = s.createQuery( hqlUpdate )
.setString( "newName", newName )
.setString( "oldName", oldName )
.executeUpdate();
tx.commit();
session.close();
한 개의 HQL DELETE를 실행하기 위해, 동일한 Query.executeUpdate() 메소드를 사용하라(그 메소드는 JDBC의 PreparedStatement.executeUpdate()에 친숙한 사람들을 위해 명명된다):
Session session = sessionFactory.openSession();
Transaction tx = session.beginTransaction();
String hqlDelete = "delete Customer where name = :oldName";
int deletedEntities = s.createQuery( hqlDelete )
.setString( "oldName", oldName )
.executeUpdate();
tx.commit();
session.close();
Query.executeUpdate() 메소드에 의해 반환되는 int 값은 그 오퍼레이션에 의해 영향받은 엔티티들의 개수를 나타낸다. 이것이 데이터베이스 내에서 영향받은 행들의 개수와 상관이 있는지 없는지 여부를 살펴보자. HQL 대량 오퍼레이션은 예를 들어 joined-subclass의 경우에 실행 중인 여러 개의 실제 SQL 문장들로 귀결될 수 있다. 반환되는 숫자는 그 문장에 의해 영향받은 실제 엔티티들의 개수를 나타낸다. joined-subclass 예제로 되돌아가면, 서브 클래스들 중 하나에 대한 삭제는 단지 그 서브클래스가 매핑되어 있는 테이블에 대한 삭제 뿐만 아니라 또한 "루트" 테이블과 상속 계층에서 더 내려온 잠정적으로 조인된-서브클래스 테이블들에 대한 삭제들로 귀결될 수 있다.
장래의 배포본들에서 전달될 대량 HQL 오퍼레이션들에 대한 몇 가지 제한들이 현재 존재함을 노트하라; 상세한 것은 JIRA 로드맵을 참조하라.
14 장. HQL: Hibernate Query Language
Hibernatesms (매우 의도적으로) SQL과 매우 유사한 극히 강력한 질의 언어를 구비하고 있다. 그러나 그 구문에 의해 우롱당하지 말라; HQL은 상속, 다형성 그리고 연관과 같은 개념들을 이해하는, 전체적으로 객체 지향적이다.
질의들은 Java 클래스들과 프로퍼티들의 이름들을 제외하면 대소문자를 구분하지 않는다. 따라서 SeLeCT는 sELEct과 같고 SELECT과도 같지만 org.hibernate.eg.FOO는 org.hibernate.eg.Foo와 같지 않고 foo.barSet은 foo.BARSET과 같지 않다.
이 매뉴얼은 소문자 HQL 키워드를 사용한다. 몇몇 사용자들은 보다 나은 가독성을 위해 대문자 키워드들을 가진 질의들을 찾지만, 우리는 자바 코드 속에 삽입할 때 이 컨벤션이 추하다는 점을 발견한다.
가장 간단한 가능한 Hibernate 질의는 다음 형식이다:
from eg.Cat
이것은 eg.Cat 클래스의 모든 인스턴스들을 간단하게 반환한다. 우리는 대개 클래스 이름을 수식할 필요가 없다. 왜냐하면, auto-import 가 디폴트이기 때문이다. 따라서 우리는 대개 항상 단지 다음과 같이 작성한다:
from Cat
대개 당신은 한 개의 alias를 할당할 필요가 있을 것이다. 왜냐하면 당신은 질의의 다른 부분들에서 Cat을 참조하고자 원할 것이기 때문이다.
from Cat as cat
이 질의는 alias cat을 Cat 인스턴스들에 할당하여서, 우리는 나중에 질의 속에서 그 alias를 사용할 수 있을 것이다. 키워드 as 는 옵션이다; 우리는 또한 다음과 같이 작성할 수 있다:
from Cat cat
여러 개의 클래스들은 cartesian product(카티젼 곱) 또는 "cross" join 으로 귀결되어 나타날 수도 있다.
from Formula, Parameter
from Formula as form, Parameter as param
로컬 변수들에 대한 Java 네이밍 표준들과 일치되게, 첫 소문자를 사용하여 질의 alias들을 명명하는 것은 좋은 습관으로 간주된다(예를 들면 domesticCat).
우리는 또한 join을 사용하여 , alias들을 연관된 엔티티들 또는 값들을 가진 콜렉션의 요소들에도 할당할 수도 있다.
from Cat as cat
inner join cat.mate as mate
left outer join cat.kittens as kitten
from Cat as cat left join cat.mate.kittens as kittens
from Formula form full join form.parameter param
지원되는 join 타입들은 ANSI SQL로부터 빌려왔다
- inner join
- left outer join
- right outer join
- full join (대개 유용하지 않음)
inner join, left outer join , right outer join 구조체들이 약칭될 수 있다.
from Cat as cat
join cat.mate as mate
left join cat.kittens as kitten
게다가, "fetch" join은 값들을 가진 콜렉션들이나 연관관계들이 하나의 select를 사용하여, 그것들의 부모 객체들과 함께 초기화 되는 것을 허용해준다. 이것은 콜렉션의 경우에 특히 유용하다. 그것은 연관관계들과 콜렉션들에 대한 매핑 파일의 outer join과 lazy 선언들을 효율적으로 오버라이드 시킨다. 추가 정보는 19.1절, “페칭 방도들”를 보라.
from Cat as cat
inner join fetch cat.mate
left join fetch cat.kittens
fetch join은 대개 alias를 할당할 필요가 없다. 왜냐하면, 연관된 객체들이 where 절(또는 어떤 다른 절) 속에 사용되지 않을 것이기 때문이다. 또한 연관된 객체들은 질의 결과들 속에 직접 반환되지 않는다. 대신에 그것들은 부모 객체를 통해 접근될 수 있다.
현재의 구현에서는 오직 한 개의 콜렉션 role 만이 질의 속에 join 페치될 수 있음을 노트하라(하나 이상의 role은 대개 한 개의 cartesian product로 귀결될 것이다). 또한 fetch 구조체는 scroll() 또는 iterate()를 사용하여 호출된 질의들 속에서 사용될 수 없음을 노트하라. 마지막으로 full join fetch 와 right join fetch는 무의미함을 노트하라.
만일 당신이 (바이트코드 방편으로) property-레벨 lazy 페칭을 사용할 경우, Hibernate로 하여금 fetch all properties를 사용하여 (첫 번째 질의 속에서) lazy 프로퍼티들을 즉시 페치하도록 강제시키는 것이 가능하다.
from Document fetch all properties order by name
from Document doc fetch all properties where lower(doc.name) like '%cats%'
select 절은 질의 결과 셋 속에서 어느 객체들과 프로퍼티들을 반환할 것인지를 골라낸다. 다음을 검토하자:
select mate
from Cat as cat
inner join cat.mate as mate
질의는 다른 Cat들의 mate들을 select 할 것이다. 실제로 당신은 이 질의들을 다음과 같이 보다 축약형으로 표현할수도 있다:
select cat.mate from Cat cat
질의들은 컴포넌트 타입의 프로퍼티들을 포함하여 임의의 값 타입의 프로퍼티들을 반환할 수도 있다:
select cat.name from DomesticCat cat
where cat.name like 'fri%'
select cust.name.firstName from Customer as cust
Family 클래스가 적당한 생성자를 갖고 있음을 가정하면, 질의들은 여러 객체들 과/또는 프로퍼티들을 Object[] 타입의 배열로서
select mother, offspr, mate.name
from DomesticCat as mother
inner join mother.mate as mate
left outer join mother.kittens as offspr
또는 List로서
select new list(mother, offspr, mate.name)
from DomesticCat as mother
inner join mother.mate as mate
left outer join mother.kittens as offspr
또는 실제 typesafe 자바 객체로서
select new Family(mother, mate, offspr)
from DomesticCat as mother
join mother.mate as mate
left join mother.kittens as offspr
반환할 수도 있다.
당신은 as를 사용하여 select되는 표현식들에 alias들을 할당할 수 있다:
select max(bodyWeight) as max, min(bodyWeight) as min, count(*) as n
from Cat cat
다음은 select new map과 함께 사용될 때 가장 유용하다:
select new map( max(bodyWeight) as max, min(bodyWeight) as min, count(*) as n )
from Cat cat
이 질의는 select된 값들에 대한 alias로부터 한 개의 Map을 반환한다.
HQL 질의들은 프로퍼티들에 대한 집계(aggregate) 함수들의 결과들을 반환할수도 있다:
select avg(cat.weight), sum(cat.weight), max(cat.weight), count(cat)
from Cat cat
지원되는 집계 함수들은 다음과 같다
- avg(...), sum(...), min(...), max(...)
- count(*)
- count(...), count(distinct ...), count(all...)
당신은 select 절 속에 산술 연산자들, 연결 연산자, 그리고 인지된 SQL 함수들을 사용할 수 있다:
select cat.weight + sum(kitten.weight)
from Cat cat
join cat.kittens kitten
group by cat.id, cat.weight
select firstName||' '||initial||' '||upper(lastName) from Person
distinct 키워드와 all 키워드가 사용될 수 있고 SQL과 같은 의미를 갖는다.
select distinct cat.name from Cat cat
select count(distinct cat.name), count(cat) from Cat cat
다음과 같은 질의:
from Cat as cat
은 Cat의 인스턴스들 뿐만 아니라, 또한 DomesticCat과 같은 서브클래스들 또한 반환한다. Hibernate 질의들은 from 절 내에 임의의 자바 클래스나 인터페이스를 명명할 수 있다. 질의는 그 클래스를 확장하거나 그 인터페이스를 구현하는 모든 영속 클래스들의 인스턴스들을 반환할 것이다. 다음 질의는 모든 영속 객체들을 반환할 것이다:
from java.lang.Object o
인터페이스 Named는 여러 가지 영속 클래스들에 의해 구현될 수도 있다:
from Named n, Named m where n.name = m.name
이들 마지막 두 개의 질의들은 하나 이상의 SQL SELECT를 필요로 할 것임을 노트하라. 이것은 order by 절이 정확하게 전체 결과 셋을 순서지우지 않음을 의미한다.(그것은 또한 당신이 Query.scroll()을 사용하여 이들 질의들을 호출할 수 없음을 의미한다).
where 절은 반환된 인스턴스들의 목록을 제한시키는 것을 당신에게 허용해준다. 만일 alias가 존재하지 않을 경우, 당신은 name에 의해 프로퍼티들을 참조할 수도 있다:
from Cat where name='Fritz'
만일 한 개의 alias가 존재할 경우, 하나의 수식어가 붙은 프로퍼티 이름을 사용하라:
from Cat as cat where cat.name='Fritz'
는 'Fritz'로 명명된 Cat 의 인스턴스들을 반환한다.
select foo
from Foo foo, Bar bar
where foo.startDate = bar.date
는 Foo의 startDate 프로퍼티와 동일한 date 프로퍼티를 가진 bar의 인스턴스가 존재하는 Foo의 모든 인스턴스를 반환할 것이다. 합성 경로 표현식들은 where 절을 매우 강력하게 만들어준다. 다음을 검토하자:
from Cat cat where cat.mate.name is not null
이 질의는 테이블 (inner) join을 가진 SQL 질의로 번역된다. 만일 당신이 다음과 같은 어떤 것을 작성했다면
from Foo foo
where foo.bar.baz.customer.address.city is not null
당신은 SQL에서 네 개의 테이블 join들을 필요로 하는 하나의 질의로 끝낼 것이다.
= 연산자는 프로퍼티들 뿐만 아니라 또한 인스턴스들을 비교하는데 사용될 수 있다:
from Cat cat, Cat rival where cat.mate = rival.mate
select cat, mate
from Cat cat, Cat mate
where cat.mate = mate
특별한 프로퍼티(소문자) id는 객체의 유일 식별자를 참조하는데 사용될 수 있다.(당신은 또한 그것의 프로퍼티 이름을 사용할 수도 있다.)
from Cat as cat where cat.id = 123
from Cat as cat where cat.mate.id = 69
두 번째 질의가 효율적이다. 테이블 join이 필요 없다!
composite identifier(합성 식별자)들의 프로퍼티들이 또한 사용될 수 있다. Person이 country와 medicareNumber로 구성된 composite identifier를 갖는다고 가정하자.
from bank.Person person
where person.id.country = 'AU'
and person.id.medicareNumber = 123456
from bank.Account account
where account.owner.id.country = 'AU'
and account.owner.id.medicareNumber = 123456
다시 한번, 두 번째 질의는 테이블 join을 필요로 하지 않는다.
마찬가지로, 특별한 프로퍼티 class는 다형성 영속성(polymorphic persistence)의 경우에 인스턴스의 판별자(discriminator) 값에 액세스한다. where 절 속에 삽입된 Java 클래스 이름은 그것의 판별자(discriminator) 값으로 번역될 것이다.
from Cat cat where cat.class = DomesticCat
당신은 또한 컴포넌트들 또는 composite 사용자 정의 타입들 (그리고 컴포넌트들의 컴포넌트들, 기타)의 프로퍼티들을 지정할 수도 있다. (컴포넌트의 프로퍼티와은 반대로) 컴포넌트 타입의 프로퍼티로 끝나는 경로-표현식을 사용하려고 결코 시도하지 말라. 예를 들어, 만일 store.owner가 컴포넌트 address 를 가진 엔티티일 경우
store.owner.address.city // 옳다
store.owner.address // 오류!
"임의의" 타입은 다음 방법으로 join을 표현하는 것을 우리에게 허용해주는, 특별한 프로퍼티들 id와 class를 갖는다(여기서 AuditLog.item은 <any>로 매핑된 프로퍼티이다).
from AuditLog log, Payment payment
where log.item.class = 'Payment' and log.item.id = payment.id
log.item.class 과 payment.class 는 위의 질의 내에서 완전히 다른 데이터베이스 컬럼들의 값들을 참조할 것임을 노트하라.
where 절 속에 허용되는 표현식들은 당신이 SQL로 작성할 수 있는 대부분의 종류의 것들을 포함한다:
- 산술 연산자들 +, -, *, /
- 바이너리 비교 연산자들 =, >=, <=, <>, !=, like
- 논리 연산들 and, or, not
- 그룹핑을 나타내는 괄호들( )
- in, not in, between, is null, is not null, is empty, is not empty, member of 그리고 not member of
- "간단한" 경우, case ... when ... then ... else ... end, 그리고 "검색된" 경우, case when ... then ... else ... end
- 문자열 연결 ...||... 또는 concat(...,...)
- current_date(), current_time(), current_timestamp()
- second(...), minute(...), hour(...), day(...), month(...), year(...),
- EJB-QL 3.0에 의해 정의된 임의의 함수 또는 오퍼레이터: substring(), trim(), lower(), upper(), length(), locate(), abs(), sqrt(), bit_length()
- coalesce() 그리고 nullif(), 기본 데이터베이스에 의해 지원될 경우
- cast(... as ...), 여기서 두번 째 아규먼트는 Hibernate 타입의 이름이고, ANSI cast()와 extract()가 기본 데이터베이스에 의해 지원될 경우에는 extract(... from ...)
- sign(), trunc(), rtrim(), sin()과 같이 임의의 데이터베이스-지원 SQL 스칼라 함수
- JDBC IN 파라미터들 ?
- 명명된 파라미터들 :name, :start_date, :x1
- SQL 리터럴들 'foo', 69, '1970-01-01 10:00:01.0'
- Java public static final 상수들. 예를 들면.Color.TABBY
in 과 between 은 다음과 같이 사용될 수 있다:
from DomesticCat cat where cat.name between 'A' and 'B'
from DomesticCat cat where cat.name in ( 'Foo', 'Bar', 'Baz' )
그리고 부정형들은 다음과 같이 작성될 수 있다
from DomesticCat cat where cat.name not between 'A' and 'B'
from DomesticCat cat where cat.name not in ( 'Foo', 'Bar', 'Baz' )
마찬가지로, is null 과 is not null 은 null 값들을 테스트하는데 사용될 수 있다.
Boolean들은 Hibernate 구성에서 HQL 질의 치환들을 선언하여 표현식들 내에 쉽게 사용될 수 있다:
<property name="hibernate.query.substitutions">true 1, false 0</property>
이것은 키워드 true와 false 키워드들을 이 HQL로부터 번역된 SQL에서 리터럴 1과 0로 대체될 것이다:
from Cat cat where cat.alive = true
당신은 특별한 프로퍼티 size로서 또는 특별한 size() 함수로서 콜렉션의 사이즈를 테스트할 수 있다.
from Cat cat where cat.kittens.size > 0
from Cat cat where size(cat.kittens) > 0
인덱싱된 콜렉션들에 대해, 당신은 minIndex과 maxIndex를 사용하여 최대 인덱스과 최소 인덱스를 참조할 수 있다. 유사하게 당신은 minelement 함수와 maxelement 함수를 사용하여 기본 타입을 가진 콜렉션의 최소 요소 및 최대 요소를 참조할 수 있다.
from Calendar cal where maxelement(cal.holidays) > current date
from Order order where maxindex(order.items) > 100
from Order order where minelement(order.items) > 10000
SQL 함수들 any, some, all, exists, in 은 콜렉션의 요소 또는 인덱스 세트(elements 함수와 indices 함수), 또는 서브질의의 결과를 전달했을 때 지원된다(아래를 보라).
select mother from Cat as mother, Cat as kit
where kit in elements(foo.kittens)
select p from NameList list, Person p
where p.name = some elements(list.names)
from Cat cat where exists elements(cat.kittens)
from Player p where 3 > all elements(p.scores)
from Show show where 'fizard' in indices(show.acts)
이들 구조체들-size, elements, indices, minindex, maxindex, minelement, maxelement-은 Hibernate3에서 where 절 내에서만 사용될 것임을 노트하라.
인덱싱 된 콜렉션들의 요소들(배열들, 리스트들, map들)은 인덱스에 의해 참조될 수 있다(where 절 안에서만):
from Order order where order.items[0].id = 1234
select person from Person person, Calendar calendar
where calendar.holidays['national day'] = person.birthDay
and person.nationality.calendar = calendar
select item from Item item, Order order
where order.items[ order.deliveredItemIndices[0] ] = item and order.id = 11
select item from Item item, Order order
where order.items[ maxindex(order.items) ] = item and order.id = 11
[] 내부의 표현식은 산술 표현실일 수 있다.
select item from Item item, Order order
where order.items[ size(order.items) - 1 ] = item
HQL은 또한 one-to-many 연관 또는 값들을 가진 콜렉션의 요소들에 대해 미리 만들어진 index()함수를 제공한다.
select item, index(item) from Order order
join order.items item
where index(item) < 5
기본 데이터베이스에 의해 제공되는 Scalar SQL 함수들이 사용될 수도 있다
from DomesticCat cat where upper(cat.name) like 'FRI%'
당신이 아직 이 모든 것을 납득하지 못한다면, 다음 질의가 SQL 내에서 가독성이 얼마나 많고 적은지를 생각해보라:
select cust
from Product prod,
Store store
inner join store.customers cust
where prod.name = 'widget'
and store.location.name in ( 'Melbourne', 'Sydney' )
and prod = all elements(cust.currentOrder.lineItems)
힌트 : 다음과 같은 어떤 것
SELECT cust.name, cust.address, cust.phone, cust.id, cust.current_order
FROM customers cust,
stores store,
locations loc,
store_customers sc,
product prod
WHERE prod.name = 'widget'
AND store.loc_id = loc.id
AND loc.name IN ( 'Melbourne', 'Sydney' )
AND sc.store_id = store.id
AND sc.cust_id = cust.id
AND prod.id = ALL(
SELECT item.prod_id
FROM line_items item, orders o
WHERE item.order_id = o.id
AND cust.current_order = o.id
)
질의에 의해 반환된 리스트는 반환된 클래스 또는 컴포넌트들의 프로퍼티에 의해 순서(ordering)지워질 수 있다:
from DomesticCat cat
order by cat.name asc, cat.weight desc, cat.birthdate
asc 옵션 또는 desc 옵션은 각각 오름차순 또는 내림차순 정렬을 나타낸다.
aggregate 값들을 반환하는 질의는 반환된 클래스나 컴포넌트들의 프로퍼티에 의해 그룹지워질 수 있다:
select cat.color, sum(cat.weight), count(cat)
from Cat cat
group by cat.color
select foo.id, avg(name), max(name)
from Foo foo join foo.names name
group by foo.id
또한 having 절이 허용된다.
select cat.color, sum(cat.weight), count(cat)
from Cat cat
group by cat.color
having cat.color in (eg.Color.TABBY, eg.Color.BLACK)
SQL 함수들과 aggregate 함수들은 기본 데이터베이스에 의해 지원될 경우(예를 들어 MySQL은 지원되지 않는다) having 절과 order by 절 속에 허용된다.
select cat
from Cat cat
join cat.kittens kitten
group by cat
having avg(kitten.weight) > 100
order by count(kitten) asc, sum(kitten.weight) desc
group by 절도 order by 절 어느 것도 산술 표현식들을 포함할 수 없다는 점을 노트하라.
subselect들을 지원하는 데이터베이스들의 경우, Hibernate는 질의들 내에 서브질의들을 지원한다. 서브질의는 괄호에 의해 감싸져야 한다(자주 SQL 집계함수 호출에 의해). 심지어 서로 관련된 서브질의들(outer 질의 내에서 alias를 참조하는 서브질의들)이 허용된다.
from Cat as fatcat
where fatcat.weight > (
select avg(cat.weight) from DomesticCat cat
)
from DomesticCat as cat
where cat.name = some (
select name.nickName from Name as name
)
from Cat as cat
where not exists (
from Cat as mate where mate.mate = cat
)
from DomesticCat as cat
where cat.name not in (
select name.nickName from Name as name
)
select 리스트 내에 있는 하나 이상의 표현식을 가진 서브질의들의 경우에 당신은 tuple 생성자를 사용할 수 있다:
from Cat as cat
where not ( cat.name, cat.color ) in (
select cat.name, cat.color from DomesticCat cat
)
(Oracle 또는 HSQL이 아닌) 몇몇 데이터베이스들 상에서 당신은 다른 컨텍스트들 내에서, 예를 들면 component들이나 composite 사용자 타입들을 질의할 때 tuple 생성자들을 사용할 수 있음을 노트하라:
from Person where name = ('Gavin', 'A', 'King')
이것을 더 풀어쓰면 다음과 동일하다:
from Person where name.first = 'Gavin' and name.initial = 'A' and name.last = 'King')
당신이 이런 종류의 것을 행하는 것을 원하지 않을 수 있는 두 가지 좋은 이유들이 존재한다: 첫 번째로 데이터베이스 플랫폼들 사이에 완전하게 이식성이 없다; 두 번째로 그 질의는 이제 매핑 문서 속에 있는 프로퍼티들의 순서에 의존한다.
Hibernate 질의들은 매우 강력하고 복잡할 수 있다. 사실, 질의 언어의 힘은 Hibernate의 주요 판매 포인트들 중 하나이다. 다음은 내가 최근의 프로젝트에서 사용했던 질의들과 매우 유사한 몇몇 예제 질의들이다. 당신이 작성하게 될 대부분의 질의들은 이것들보다 훨씬 간단하다는 점을 노트하라!
다음 질의는 결과들을 전체 합계값에 따라 순서지워서 특정 고객에 대한 모든 지불되지 않은 주문들의 주문 id, 항목들의 개수, 그리고 주문의 전체 합계값 그리고 주어진 최소 전체 합계를 반환한다. 가격 결정에 있어, 그것은 현재의 카타록을 사용한다. 귀결되는 SQL 질의는 ORDER, ORDER_LINE, PRODUCT, CATALOG, PRICE 테이블들에 대한 네 개의 inner 조인들과 한 개의(상관지워지지 않은) subselect를 갖고 있다.
select order.id, sum(price.amount), count(item)
from Order as order
join order.lineItems as item
join item.product as product,
Catalog as catalog
join catalog.prices as price
where order.paid = false
and order.customer = :customer
and price.product = product
and catalog.effectiveDate < sysdate
and catalog.effectiveDate >= all (
select cat.effectiveDate
from Catalog as cat
where cat.effectiveDate < sysdate
)
group by order
having sum(price.amount) > :minAmount
order by sum(price.amount) desc
괴물 같은 것! 실제로 실 생활에서, 나는 서브질의들을 매우 좋아하지 않아서, 나의 질의는 실제로 다음과 같았다:
select order.id, sum(price.amount), count(item)
from Order as order
join order.lineItems as item
join item.product as product,
Catalog as catalog
join catalog.prices as price
where order.paid = false
and order.customer = :customer
and price.product = product
and catalog = :currentCatalog
group by order
having sum(price.amount) > :minAmount
order by sum(price.amount) desc
다음 질의는 현재 사용자에 의해 가장 최근의 상태 변경이 행해졌던 AWAITING_APPROVAL 상태에 있는 모든 지불들을 제외한, 각각의 상태에 있는 지불들의 개수를 카운트 한다. 그것은 PAYMENT, PAYMENT_STATUS, PAYMENT_STATUS_CHANGE 테이블들에 대한 두 개의 inner 조인들과 하나의 상관관계 지워진 subselect를 가진 SQL 질의로 변환된다.
select count(payment), status.name
from Payment as payment
join payment.currentStatus as status
join payment.statusChanges as statusChange
where payment.status.name <> PaymentStatus.AWAITING_APPROVAL
or (
statusChange.timeStamp = (
select max(change.timeStamp)
from PaymentStatusChange change
where change.payment = payment
)
and statusChange.user <> :currentUser
)
group by status.name, status.sortOrder
order by status.sortOrder
만일 내가 statusChanges 콜렉션을 set가 아닌 list로 매핑했다면, 그 질의는 작성하기가 훨씬 더 간단했을 것이다.
select count(payment), status.name
from Payment as payment
join payment.currentStatus as status
where payment.status.name <> PaymentStatus.AWAITING_APPROVAL
or payment.statusChanges[ maxIndex(payment.statusChanges) ].user <> :currentUser
group by status.name, status.sortOrder
order by status.sortOrder
다음 질의는 현재의 사용자가 속해 있는 조직의 모든 계정들과 지불되지 않은 지불들을 반환하는데 MS SQL Server isNull() 함수를 사용한다. 그것은 ACCOUNT, PAYMENT, PAYMENT_STATUS, ACCOUNT_TYPE, ORGANIZATION, ORG_USER 테이블들에 대한 세 개의 inner 조인들, 하나의 outer 조인, 그리고 하나의 subselect를 가진 SQL 질의로 번역된다.
select account, payment
from Account as account
left outer join account.payments as payment
where :currentUser in elements(account.holder.users)
and PaymentStatus.UNPAID = isNull(payment.currentStatus.name, PaymentStatus.UNPAID)
order by account.type.sortOrder, account.accountNumber, payment.dueDate
몇몇 데이터베이스들의 경우, 우리는 (상관관계 지워진) subselect를 없앨 필요가 있을 것이다.
select account, payment
from Account as account
join account.holder.users as user
left outer join account.payments as payment
where :currentUser = user
and PaymentStatus.UNPAID = isNull(payment.currentStatus.name, PaymentStatus.UNPAID)
order by account.type.sortOrder, account.accountNumber, payment.dueDate
당신은 실제로 질의 결과들을 반환하지 않고서 그것들(질의 결과들)의 개수를 카운트할 수 있다:
( (Integer) session.iterate("select count(*) from ....").next() ).intValue()
콜렉션의 크기에 따라 결과를 순서(ordering)지우려면, 다음 질의를 사용하라:
select usr.id, usr.name
from User as usr
left join usr.messages as msg
group by usr.id, usr.name
order by count(msg)
만일 당신의 데이터베이스가 subselect들을 지원할 경우, 당신은 당신의 질의의 where 절 내에 selection 사이즈에 대한 조건을 위치지울 수 있다:
from User usr where size(usr.messages) >= 1
만일 당신의 데이터베이스가 subselect를 지원하지 않을 경우, 다음 질의를 사용하라:
select usr.id, usr.name
from User usr.name
join usr.messages msg
group by usr.id, usr.name
having count(msg) >= 1
inner 조인 때문에 이 해결책이 0개의 메시지를 가진 User 를 반환할 수 없으므로, 다음 형식이 또한 유용하다:
select usr.id, usr.name
from User as usr
left join usr.messages as msg
group by usr.id, usr.name
having count(msg) = 0
하나의 JavaBean의 프로퍼티들은 명명된 질의 파라미터들에 바인드될 수 있다:
Query q = s.createQuery("from foo Foo as foo where foo.name=:name and foo.size=:size");
q.setProperties(fooBean); // fooBean has getName() and getSize()
List foos = q.list();
콜렉션들은 필터를 가진 Query 인터페이스를 사용하여 쪼매김하는 것이 가능하다:
Query q = s.createFilter( collection, "" ); // the trivial filter
q.setMaxResults(PAGE_SIZE);
q.setFirstResult(PAGE_SIZE * pageNumber);
List page = q.list();
콜렉션 요소들은 질의 필터를 사용하여 순서(ordering)지워지거나 그룹지워질 수 도 있다:
Collection orderedCollection = s.filter( collection, "order by this.amount" );
Collection counts = s.filter( collection, "select this.type, count(this) group by this.type" );
당신은 콜렉션을 초기화 하지 않고서 그것(콜렉션)의 크기를 찾을 수 있다:
( (Integer) session.iterate("select count(*) from ....").next() ).intValue();
Hibernate는 직관적인, 확장 가능한 criteria query API를 특징 짓는다.
org.hibernate.Criteria 인터페이스는 특정 영속 클래스에 대한 질의를 표현한다. Session은 Criteria 인스턴스들에 대한 팩토리이다.
Criteria crit = sess.createCriteria(Cat.class);
crit.setMaxResults(50);
List cats = crit.list();