** 본 요약은 한빛미디어 '하이버네이트 프로그래밍' 책을 기준으로 작성되어져 있다.

** 본 설명은 바로 따라하기 하기엔 어려움이 있다. 책을 보지 않은 사람에게는 간단히 어떤 기능을 제공하는지 이해하는데 목적을 둔다.

** 하이버네이트 3.2.5.ga 버전, Ant 빌드를 기본적으로 사용한다.


5장 다양한 연관

이번 장에서는 5개의 주제를 가지고 설명한다. 앞장에서 다룬 연관을 조금 더 세밀하게 설정하는 방법을 배운다. 간단히 요약하면 다음과 같다. 

  1) 연관의 초기화 형태 : 느슨한(Lazy), 열성적인(Eager) 

  2) 순서가 있는 컬렉션 사용하기

  3) 순서가 있는 컬렉션 설정 보강하기

  4) 순서있는 연관 컬랙션 삭제에 따른 자동처리

  5) 재귀 연관

  

연관의 초기화 형태 : 느슨한(Lazy), 열성적인(Eager) 


객체내에 다른 객체를 참조하는 값을 가질 수 있는데, 참조되는 객체를 처음 생성때부터 초기화 할것인지(eager), 아니면 사용되기 전까지 초기화를 미룰 것인지(lazy) 설정하는 방법을 설명한다.


하이버네이트3부터는 연관 객체에 대한 생성은 늦은 초기화가 기본적으로 사용된다. 즉, 실제로 데이터를 사용하기 전까지 데이터베이스에서 로드하지 않는다는 것이다. 그러나 주의할 것은 하이버네이트 세션을 닫게 되면 연관 객체의 늦은 초기화가 작동하지 않아 LazyInitializationException을 던진다. 즉, Lazy를 사용하려면 하이버네이트 세션을 닫으면 안된다는 것이다. 


그러면 기본적인 늦은 초기화를 사용하지 않으려면 어떻게 하는지 알아보면 다음 코드 처럼 작성하면 된다.


<set name="association_Obj" table="AAA_BBB" lazy="false">

  <key column="AAA_ID"/>

  <many-to-many class="com.billy.sample.BBB" column="BBB_ID"/>

</set>


위와 같이 lazy="false"를 추가해 빠른(eagerly) 초기화를 설정할 수 있다.


다양한 연관에서 위와 같이 늦은 초기화를 사용안하게 할 수 있지만 특정 클래스 참조의 대한 부분 모두를 빠른 초기화 설정을 하는 방법은 매핑문서의 class 태그에 lazy="false"를 직접 사용하면 된다.



순서가 있는 컬렉션 설정 방법


만약 다른 연관된 테이블 데이터에 순서가 있을때, 순서를 지정하기 위한 방법을 다룬다. 일단 List 컬랙션과 맵핑해야 하고, 리스트의 순서를 설정하는 list_index 태그를 추가해야 한다. 그리고 순서 조정 값을 갖는 컬럼을 데이터 베이스에 추가해야 한다. 코드를 보면 다음과 같다.


<list name="bbbList" table="AAA_BBB" lazy="false">

  <key column="AAA_ID"/>

  <list-index column="LIST_POS"/>

  <many-to-many class="com.billy.sample.BBB" column="BBB_ID"/>

</list>


하이버네이트에서는 LIST_POS를 관리하여 나중에 데이터베이스에서 꺼낸 리스트 내용으 우리가 저장한 컬럼과 동일한 순서로 나오게 된다. 


순서가 있는 컬렉션 설정 보강하기


* 데이터베이스 스키마에 테이블을 넣는 방법

1) 테이블의 컬럼에 자바 객체의 매핑 프로퍼티를 명시하는 것

2) 컬렉션(컬렉션의 값 또는 연관)을 정의하고, 그 컬렉션을 관리하는데 사용할 테이블과 컬럼을 지정하는 것


* 엔티티와 컴포넌트

- 엔티티 : 다른 객체에 대해 독립적으로 생성, 질의, 삭제될 수 있는 것

- 컴포넌트 : 다른 엔티티의 종속적인 부분으로서 데이터베이스에 저장되거나 데이터베이스에서 검색 될 수 있는 것


* 컴포넌트 맵핑

<list name="tracks" table="ALBUM_TRACK">

  <meata attribute="use-in-tostring">true</meta>

  <key column="ALBUM_ID"/>

  <index column="LIST_POS"/>

  <composite-element class="com.oreilly.hh.data.AlbumTrack">

    <many-to-one class="com.oreilly.hh.data.Track" name="track" cascade="all">

      <meta attribute="use-in-tostring">true</meta>

      <column name="TRACK_ID" />

    </many-to-one>

    <property name="disc" type="integer" />

    <property name="positionOnDisc" type="integer" />

  </composite-element>

</list>


위 컴포넌트 맵핑은 새로운 AlbumTrack라는 클래스를 소개한다. 3장에서 소개한 N:N 맵핑의 경우 조인 테이블을 사용한다고 했는데, 그 조인 테이블을 사용하는 클래스가 없다고 설명했다. 그런데 이와 같이 N:N 조인테이블에 부가정보를 추가해 새로운 관리 클래스를 생성할 수 있다. 이렇게 생성된 코드를 살펴보면 다음과 같다.


public class AlbumTrack  implements java.io.Serializable {

  private Track track;

  private Integer disc;

  private Integer positionOnDisc;


  public AlbumTrack() {

  }


  public AlbumTrack(Track track, Integer disc, Integer positionOnDisc) {

    this.track = track;

    this.disc = disc;

    this.positionOnDisc = positionOnDisc;

  }

 

  public Track getTrack() {

    return this.track;

  }

  

  public void setTrack(Track track) {

    this.track = track;

  }

  public Integer getDisc() {

    return this.disc;

  }

  

  public void setDisc(Integer disc) {

    this.disc = disc;

  }

  public Integer getPositionOnDisc() {

    return this.positionOnDisc;

  }

  

  public void setPositionOnDisc(Integer positionOnDisc) {

    this.positionOnDisc = positionOnDisc;

  }


  /**

   * toString

   * @return String

   */

  public String toString() {

 StringBuffer buffer = new StringBuffer();


    buffer.append(getClass().getName()).append("@").append(Integer.toHexString(hashCode())).append(" [");

    buffer.append("track").append("='").append(getTrack()).append("' ");

    buffer.append("]");

     

    return buffer.toString();

  }

}


위 코드는 id 프로퍼티가 빠져있다. 컴퍼넌트 클래스는 식별 필드나 특별한 인터페이스 구현이 필요 없다. 


순서있는 연관 컬랙션 삭제에 따른 자동처리


- 부모 객체와 자식 객체의 종속성 설정 : 하이버네이트에서는 "부모"객체가 실행되면 "자식" 혹은 "의존" 객체까지 전이되는 연산을 cascade 어트리뷰트로 처리할 수 있다. 이 기능은 모든 종류의 컬렉션과 연관에 적용된다. 

- 사용 방법 : 객체 맵핑 프로퍼티에 <cascade> 태그를 추가한다.

  -> cascade 값 : none, save-update, delete, all(save-update+delete)

  -> 전체 default cascade 변경 : <hibernate-mapping> 태그에 default-cascade 어트리뷰트 추가

  -> 부모가 삭제되면 자식도 동시에 삭제 가능하도록 설정 가능

  -> 중간에 order관련 값들을 자동으로 변경해줌

  

재귀 연관


객체나 테이블에서 자체적으로 연관을 맺을 수 있다. 트리같은 객체 구조를 생성할 수 있다는 것이다. 이는 단순하다. 

Artist 클래스 맵핑 문서야 다음과 같이 자기 자신을 참조하는 필드를 추가하면 된다. 


<many-to-one name="actualArtist" class="com.oreilly.hh.data.Artist">

 <meta attribute="use-in-tostring">true</meta>

</many-to-one>


  

  

  

Posted by 빌리 :

** 본 요약은 한빛미디어 '하이버네이트 프로그래밍' 책을 기준으로 작성되어져 있다.

** 본 설명은 바로 따라하기 하기엔 어려움이 있다. 책을 보지 않은 사람에게는 간단히 어떤 기능을 제공하는지 이해하는데 목적을 둔다.

** 하이버네이트 3.2.5.ga 버전, Ant 빌드를 기본적으로 사용한다.


4장 컬렉션과 연관

본 장에서는 객체간 관계나 그룹이 처리되는 방법은 알아본다.


테이블 간의 관계 표현하기


두개의 클래스 용 맵핑 문서를 만들고, 그 안에 'set'이라는 태그를 이용해 연결 정보를 추가하여 사용한다.


N:N 연결 관계는 다음 코드를 연결할 양쪽 클래스 맵핑 문서에 다음 내용을 추가해 구현한다.


<set name="[필드명]" table="[조인 테이블명]">

  <key column="[현재 테이블의 아이디 키]"/>

  <many-to-many class="[연결되는 대상 클래스]" column="[연결되는 테이블의 아이디 키]"/>

</set>


양쪽에 <many-to-many> 태그를 사용하면, 하이버네이트에서는 자동으로 현재 테이블의 아이디 키와 연결되는 테이블의 아이디 키가 컬럼을 가진 조인 테이블을 생성한다. 이 테이블과 실제로 맵핑되는 객체는 존재하지 않고 N:N 관계를 유지하기 위한 테이블인 조인 테이블로 사용하게 된다.


실제 전체 코드를 보면 다음과 같이 작성될 수 있다.


AAA 클래스용 매핑 문서

<?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="com.billy.sample.data.AAA" table="AAA">

<meta attribute="class-description">

A 샘플 클래스 매핑 예제

</meta>

<id name="id" type="int" column="AAA_ID">

<meta attribute="scope-set">protected</meta>

<generator class="native"/>

</id>

    

<property name="name_a" type="string" not-null="true"/>

    

    <set name="bset" table="AAA_BBB" inverse="true">

      <meta attribute="field-description">연관된 B 클래스</meta>

      <key column="AAA_ID"/>

      <many-to-many class="com.billy.sample.data.BBB" column="BBB_ID"/>

    </set>

  </class>

</hibernate-mapping>


BBB 클래스용 매핑 문서

<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"

"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">

<hibernate-mapping>

<class name="com.billy.sample.data.BBB" table="BBB">

<meta attribute="class-description">

B 샘플 클래스 매핑 예제

</meta>

<id name="id" type="int" column="BBB_ID">

<meta attribute="scope-set">protected</meta>

<generator class="native"/>

</id>

    

<property name="name_b" type="string" not-null="true"/>

    

    <set name="aset" table="AAA_BBB">

      <meta attribute="field-description">연관된 A 클래스</meta>

      <key column="BBB_ID"/>

      <many-to-many class="com.billy.sample.data.AAA" column="AAA_ID"/>

    </set>

  </class>

</hibernate-mapping>


위 코드 중에 AAA 클래스용 매핑 문서의 <set> 태그 안에 inverse(역매핑)이 설정 되어 있다. 이는 AAA클래스의 bset이 변경이 일어나면 AAA_BBB 테이블이 변경되지 않지만, BBB클래스의 aset이 변경이 일어났을때 AAA_BBB 테이블이 자동 갱신되도록 설정한다.


이렇게 맵핑 문서를 작성하고 난 후 Hibernate tool을 이용해 소스코드를 생성하면 해당 객체에 다음과 같은 코드가 추가 된다.


public class AAA implements java.io.Serializable {

  ...

  private Set bset = new HashSet(0);

  

  public Set getBset() {

    return this.bset;

  }

  

  public void setBset(Set bset) {

    this.bset = bset;

  }

  ...

}


그런데 위의 코드는 Java 5 버전에서는 Typesefe 경고가 발생한다. 이를 처리하기 위해서는 자바 코드 생성시 다음과 같은 코드를 추가해준다.(Ant task기준)


<hbm2java jdk5="true">

위 코드를 추가 후에 자바코드를 추가 하면 


private Set<BBB> bset = new HashSet<BBB>(0); 

과 같이 타입이 명시된 제너릭 코드가 생성된다.


 

객체에서 관계된 테이블(컬렉션) 데이터 입력하고, 데이터 읽어오기


위 단계에서 N:N 설정된 데이터를 사용하는 방법은 간단한다. 자동생성된 클래스의 Set 부분에 해당 값을 추가 해서 저장하면 자동으로 조인테이블에 데이터가 생성된다. 데이터 읽어오는 방법도 기존의 앞에서 해당 객체를 불러오는 방법과 동일하게 사용하면 데이터를 가져올 수 있다.



1:N 관계 표현하기


N:N 관계 설정하는 방법과 비슷하지만 조금 더 간단한다. N:N에서 설정한 맵핑 파일에 주석을 단다고 가정하면 다음과 같이 맵핑 코드를 작성할 수 있다.


<set name="comments" table="AAA_COMMENTS">

  <key column="AAA_ID"/>

  <element column="COMMENT" type="string"/>

</set>


그러면 AAA_COMMENTS 테이블이 생성되고, 자바 코드상에도 Set<String> comments가 추가되어 사용할 수 있게 된다.



Posted by 빌리 :