안녕하세요,
오늘은 ORM(Object-Relational Mappging)에 대한 이야기를 해볼까 합니다.
ORM이란 용어가 익숙하시 분도 계시겠지만, 생소한 분도 있으실 것이라고 생각합니다.
사실 스프링 프레임워크를 쓰고 계신다면, 여러분은 이미 ORM 개념을 사용하고 계신 겁니다.
스프링 프레임워크가 곧 ORM 개발 방식을 구현한 것이니까요.
자세한 내용은 아래의 링크를 참조하시면 더 도움이 될 것입니다.
[출처] ORM의 기본적인 개념 및 활용방안
Table of Contents
Introduction
본 문서는 최근 persistence layer를 처리하기 위한 다양한 개발방법 중에 ORM(Object-Relational Mappging)의 기본적인 개념과 실무에서 활용할 수 있는 방안을 고민해 본다.
먼저 ORM 프로그램 기법이 왜 생기게 되었는지를 등장배경으로 정리해 보고, ORM의 기본적인 개념을 대해서 살펴본 후, ORM 기법을 이용하여 실제적으로 활용할 수 있는 방안이 있는지를 계속해서 살펴보도록 한다.
ORM
ORM 등장배경
ORM이 등장하기 까지의 배경을 살펴보면 다음 그림과 같다. 각 개발 방식에 따라서 장/단점을 기술해 두었다.
Java 진영에서 웹 애플리케이션을 개발할 때 처음에는 JSP/Servlet에 View와 Business logic, 데이터 베이스 처리 모듈까지 모두 포함하여 개발함으로써, 유지보수의 어려움과 같은 많은 문제점이 발견되어 진다.
따라서 MVC 모델에 입각한 각 Layer를 구분하여 개발자로 하여금 business logic에 집중할 수 있는 컴포넌트와 같은 형태들이 제공되어 지는데 SUN에서는 J2EE 스팩안에 EJB를 넣음으로서 persisten layer를 컨테이너에게 담당하게 함으로써 이와 같은 문제점을 해결하고자 했다.
하지만 Model 2 방식에서 이와 같은 접근은(특히 EJB-CMP를 두고 말함) 각 WAS(Web Application Server) 제품에 따른 deploymement descriptor 파일을 각각 수정해야 하고, 낮은 개발자 생산성, 테스트의 어려움 등 다양한 단점이 발생하게 된다. 뿐 만 아니라 각 WAS에 따른 컨테이너의 처리 방식이 상이함에 따라 개발자들은 EJB 뿐 만 아니라, WAS에 대한 이해도 요구하게 되었다.
이러한 하나의 대안으로 DAO 클래스를 두고 데이터 베이스와의 처리를 DAO에서 처리하게 된다. DAO 클래스에서는 개발자의 특성에 따라서 트랙잭션 처리, Connection Pool 처리 등을 상이하게 개발하게 된다.
그리고, 현재 ORM 프로그래밍 기법에 따른 persistence layer를 처리하는 Hibernate, iBatis, Oracle TopLink 와 같은 Framework가 등장하게 된다. EJB의 대안으로 등장하게 된 이와 같은 Framework는 각 Framework에 구현 방식에 따라 Query, Connection, transacation 처리 등을 Framework가 처리해 줌으로서 개발자로 하여금 business logic에 집중할 수 있도록 제공해 주게 된다.
하지만 이러한 프레임워크를 공부해야 하고 적응하는 데에도 시간이 필요하고, 경우에 따라서 유지보수의 어려움이 따르게 된다. 그러므로 각 개발환경에 맞는 적절한 개발방식을 선택할 수 있는 능력이 필요하게 된다.
ORM의 기본적인 개념
ORM의 기본적인 개념은 현재 살펴본 바로는 자바지기 스터디 1,2기에서 장진달님과 유성희님께서 수고한 덕분으로 이미 잘 정리된 문서가 존재함으로 먼저 소개하는 것으로 한다.
[Hibernate 프레임워크를 이용한 효율적인 개발 전략|http://wiki.javajigi.net/pages/viewpage.action?pageId=297#장진달] : 두 번째 목차인 "ORM - Obejct Relation Mappging"는 꼭 참조
[Object-Relational Mapping Strategies|http://wiki.javajigi.net/pages/viewpage.action?pageId=3081#유성희] : 두 번째 목차인 "Object-Relational Mapping이란?"는 꼭 참조
실제 Hibernate 문서와 위의 두 wiki 내용은 문서를 다시금 쉽게 정리해 보면 Object-Orientied programming에 있어서 관계형 데이터베이스와의 연계성을 분명히 하자는 것이다. 즉 Object-Relational Mapping은 다음과 같이 생각해 볼 수 있다. 우리가 '객체(Object)로 실세계의 현상(또는 구현하고자 하는 바)를 객체로 표현하여 좀 더 분명하고 쉽게 개발을 하자'라고 하는 게 객체지향 프로그래밍이라고 한다. 즉 개발자는 관계형 데이터베이스에 제약을 최대한 받지 않으면서(또는 한정되지 않고), 객체를 클래스(Bean)으로 표현하는 것과 같이 관계형 데이터베이스를 객체처럼 쉽게 표현 또는 사용하자는 것이다. 때문에 ORM이란 용어처럼 객체(Object)와 관계(Relation)을 연결(Mapping)해 주는 개념이 된다라고 생각할 수 있다.
다음 그림은 실세계에 존재하는 바를 왼쪽은 객체(Object) 형태로, 오른쪽은 데이터베이스의 테이블로 표현한 것이다.
위의 그림은 실세계를 표현하는 데 있어 객체와 테이블(관계형 데이터베이스)는 서로 다른 형태를 가지는 것을 볼 수 있다. 실제로 Object를 그대로 관계형 데이터베이스에 표현하면 좋다거나 관계형 데이터베이스의 테이블을 객체로 표현하면 좋은 경우가 있다. 하지만 서로다른 형태의 구조를 가진 객체와 테이블은 서로 배타적인 표현 때문에 개발자로 하여금 데이터베이스 프로그래밍을 처리하게 하는 것이다.
ORM에서의 mapping concept 중 'one-to-one' 개념 형태를 살펴보면 실제 데이터베이스와 객체를 1:1로 속성(객체의 attribute)와 row를 매핑해 주는 개념이 존재한다. 이러한 형태는 EJB Entity Bean이 가지고 있는 개념으로 데이터베이스의 테이블을 객체로 매핑해 주고 있는 것이다.
여기서 생각해 볼 문제는 위의 그림 예제와는 거꾸로 두 개의 테이블이 있다고 가정하면 각 테이블을 두 개의 객체로 표현해야 한다. 하지만 이를 OOP에서는 하나의 객체로 표현하는 게 좋을 경우도 있을 것이다. 이와 같은 경우 실제 객체와 데이터베이스와의 연결 구조를 어떻게 가져갈 것이냐? 라고 생각해 볼 수 있다.
이에 대해서는 이미 ORM에서 세 개의 mapping concept을 가지고 설명해 주고 있다. mapping concept에 대한 설명은 1기 장진달님께서 너무나도 잘 설명해 준 자료가 있어 소개한다.
Mapping Concept (Persistent class와 table사이의 mapping방법)
- one-to-one : 테이블의 모든 컬럼에 대하여 persistent class의 모든 변수가 표현되며 하나의 비지니스 클래스는 테이블의 할 row에 해당 한다.
- SUBSET Mapping : 하나의 테이블을 하나이상의 class로 mapping 하며 그 세부적으로 vertical mapping, horizontal mapping, filtered mapping이 있을 수 있다.
- 간단 한 예로 각각의 설명을 해보겠다. 어떤 매장의 고객과 종업원을 관리하는 업무라고 할때 고객과 종업원은 아래와 Class Diagram으로 객체가 구성되었다면.
- Vertical mapping: 하나의 테이블당 하나의 객체로 구현 되는 방식으로 아래와 같은 DB구조를 가지게 된다.
Person Table Employee Table Customer Table PersonID ID ID LastName PersonID PersonID FirstName Since Terms Title PayRate Owes - Horizontal mapping : 하나의 테이블을 하나이상의 객체로 표현 하는 방식으로 위의 아래와 같은 DB구조를 위의 그림과 같은 객체로 mapping하는 방식이다.
Employee Table Customer Table ID ID LastName LastName FirstName FirstName Title Title Terms Since Owes PayRate - Filter mapping : Horizontal 방식과 동일한 계념이나 어떤 data의 값을 가지고 filtering 하여 객체를 mapping하는 방식으로 아래와 같이 Type 컬럼의 값을 가지고서 고객과 종업원을 구분 하는 형식이 될 수 있다.
Person Table ID Type LastName FirstName Title Since PayRate_Owes Terms
- 간단 한 예로 각각의 설명을 해보겠다. 어떤 매장의 고객과 종업원을 관리하는 업무라고 할때 고객과 종업원은 아래와 Class Diagram으로 객체가 구성되었다면.
- SUPERSET Mapping : 여러 테이블을 묶어서 하나의 class로 mapping 하는 방식으로 view의 편의성을 위하여 많이 구현 된다.
- 간략하게 덧붙이자면 위의 경우에서 고객의 주소, 전화번호 같은 정보를 담고 있는 테이블 PersonInfo table이 존재할 경우 Person Table과 PersonInfo Table의 colum값을 모두 표현한는 객체로 mapping하는 방식이라 할 수 있는데 DB구조와 Class는 아래와 같은 구조로 보일 것이다.
- 객체 구조
Person LastName,FirstName,Address,Phone,Age - DB구조
Person Table PersonInfo Table ID ID LastName Address FirstName Phone Title Age Since LastVisit PayRate_Owes Terms
Transparent persistence
ORM의 기본적인 개념에서 객체와 관계에 대한 이야기를 토대로 했을 때, transparent persistence가 비로써 이해가 된다. Transparent persistence는 O-R Mapping tool이 관계형 데이터베이스에 저장된 데이터를(오브젝트 프로그래밍 언어를 통해) 직접 조작하는 기능이다.
Transparent persistence에 대한 자세한 설명과 샘플예제는 다음의 링크를 참조하면 된다.
Transparent persistence : http://www.service-architecture.com/object-oriented-databases/articles/transparent_persistence.html
Transparent persistence sample : http://www.service-architecture.com/object-oriented-databases/articles/java_and_object_databases.html
ORM 활용방안
가이드 라인
ORM을 이용한 활용 방안은 특별한 정답이 없는 듯 하다. 무조건적으로 해당 프로젝트에 ORM을 도입할 수도 없다. 하지만 적절한 환경, 적절한 애플리케이션에 ORM을 적용함으로써 얻을 수 있는 장점이 있음으로 이를 이해하지 않고 있을 수도 없는 일이다. 다음은 참조문헌에 있는 내용을 발췌한 ORM을 도입하는 데 있어 가이드라인을 제시하고 있다. 이를 한 번 살펴보자.
토의
- 대한민국 IT 환경에서의 ORM은 적합한가?
- 작은 규모의 프로젝트에서 ORM이 적합한가?
- 대형 규모의 프로젝트이거나 솔루션 기반의 프로젝트일 경우 데이터베이스의 비중과 ORM 도입에 있어서의 고민을 해 보아야 한다.
참조문헌
Persistence와 POJO: 오브젝트 및 관계형 모델의 통합
[ORM에 대한 기본적인 이해(Hibernate를 중심으로 한)|http://blog.naver.com/jdkim528/140010947478#김종대]
Hibernate 프레임워크를 이용한 효율적인 개발 전략 - 장진달
Object-Relational Mapping Strategies - 유성희
도메인모델을 이용한 엔터프라이즈 애플리케이션 구축 - 김형준
http://www.service-architecture.com/object-oriented-databases/articles/transparent_persistence.html
http://www.service-architecture.com/object-oriented-databases/articles/java_and_object_databases.html
'Dev. Java > Dev.Spring' 카테고리의 다른 글
[SPRING] 타일즈3 TILES3 세팅하기 (0) | 2016.10.29 |
---|---|
web.xml url-pattern / 와 /* 의 차이점 (0) | 2016.09.27 |
자동 의존성 주입 - @Autowired, @Resource, @Inject 차이 (0) | 2016.03.19 |
스프링 pom.xml에 ojdbc14 추가하기 (0) | 2016.02.29 |
STS로 프로젝트 생성 후, resources 폴더에 새로운 폴더를 생성할 때, 패키지로 보이는 문제 해결 방법 (0) | 2015.03.04 |