달력

1

« 2025/1 »

  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
2007. 3. 14. 13:44

패턴과 프레임워크 그거/Tech2007. 3. 14. 13:44

패턴과 프레임워크

패턴과 프레임워크, 다소 어려워 보일 수 있는 주제이지만 프로그래머로써 한 단계 거듭나기 위해 거쳐야 할 과제입니다. 이제까지 기본기로만 싸워왔다면 이제부터는 좀 고난이도의 기술과 새로운 무기들을 써 봅시다. 지금부터가 정말 즐거운 프로그래밍의 세계로 들어가는 길입니다.

탈무드에는 배고픈 자식에게 고기를 잡아주기보다 고기를 잡는 방법을 가르쳐주라는 말이 있습니다. 좋은 프로그램을 고기라고 본다면 좋은 프로그램을 만들어주는 것보다 좋은 프로그램을 만드는 방법을 가르쳐야 한다는 말이 되겠죠. 하지만 그러기엔 마소 주니어의 지면은 넉넉지 않습니다. 사실 초급자는 잡지에서 새로운 것을 익히려 하기보다 입문서에서 배워야 합니다.

 

그렇다면 이 한정된 지면에서 할 수 있는 최선은 자세한 내용을 기술하는 것이 아니라 무엇을 공부해야 하는지, 어디서 정보를 얻을 수 있는지, 요즘 추세는 어떤지를 알려주는 것이겠지요. 말하자면 좋은 프로그램이 아니라 그것을 만드는 방법을 고기로 보고 이것을 잡는 방법을 제시하겠다는 뜻입니다.

 

필자의 짧은 세 번의 연재에는 모두 이런 생각이 담겨 있고 그래서 기술적인 상세 내용 설명보다는 무엇을 공부해야 하는가에 대한 내용에 치중했습니다. 이번 글 역시 디자인 패턴, 프레임워크, XP 세 가지를 다루고 있지만 각각의 상세한 설명보다는 이런 개념들을 어떻게 이해하고 활용해야 하는가에 초점을 맞출 것입니다.

디자인 패턴과 프레임워크는 먼저 용어에 대한 설명부터 시작합니다. 그리고 그것들의 목적과 가치에 대해 살펴보고 몇 가지 예를 살펴볼 것입니다. XP는 제목에 포함시키긴 했지만 이것은 디자인 패턴과 프레임워크를 XP의 시각에서 바라본다는 의미이고 개별적인 주제로는 다루지 않습니다.


용어로서의 디자인 패턴
디자인 패턴이란 프로그래밍에서 발생하는 여러 가지 문제 영역에 대한 해결 방법들을 모아서 정리해놓은 것입니다. GoF(Gang of Four)가 여러 가지 패턴들을 모아 『디자인 패턴』이라는 책을 내면서부터 디자인 패턴이라는 용어가 일반화되기 시작했습니다. GoF의 디자인 패턴 중 하나를 예로 들면 데코레이터(Decorator)라는 패턴이 있습니다. 랩퍼(Wrapper)라는 말로도 쓰이는데 이것은 어떤 객체에 기본적인 동작은 유지하면서 부가적인 기능을 넣고 싶을 때 사용합니다.

이 패턴의 간단한 예로 JDBC에서 수행한 쿼리를 로그로 남기고 싶을 때 로깅을 위한 데코레이터를 사용할 수 있습니다. 보통 JDBC에서는 Connection 객체를 이용해서 Statement 객체를 생성하고 이것을 이용해서 쿼리를 수행하게 되는데 이 때 Statement 대신 LoggableStatement를 사용하는 것입니다.

 

LoggableStatement는 Statement를 상속하며 생성자에서 Connection 객체가 만든 Statement 객체를 받아서 멤버로 갖고 있습니다. 그리고 쿼리 요청이 오면 먼저 로깅을 하고 실제 쿼리는 생성자에서 받은 Statement 객체에 위임합니다. 이런 방식으로 Statement의 동작은 그대로 유지하면서 로그를 남기는 기능을 추가할 수 있습니다.

 

이런 것이 디자인 패턴입니다. 수학 공식처럼 같은 종류의 문제들에 대해 이미 정리된 해결책을 이용함으로써 매번 같은 문제를 해결하는데 중복으로 투자되는 노력을 줄이려는 것이죠.

하지만 디자인 패턴을 그대로 사용하는 것이 좋은 것만은 아닙니다. 오용의 위험성이 크기 때문입니다. 사실 어떤 문제 상황에 어떤 패턴을 적용하면 되는가를 판단하기란 쉬운 일이 아니기 때문에 필요하지도 않은 복잡한 패턴을 도입하는 경우가 많습니다. 그 때문에 간단하게 해결될 문제가 더 복잡해지기도 하죠. 그래서 안티 패턴 같은 이야기도 나오고 있습니다. XP(eXtreme Programming)에서도 디자인 패턴을 의식하지 말라고 합니다.

 

그보다 원하는대로 동작할 수 있는 가장 간단한 방법으로 코딩을 하다가 중복이 생기거나 코드가 지저분해지면 리팩토링을 합니다. 그러면서 자연스럽게 코드가 디자인 패턴의 모습을 갖추어 나가는 것이 바람직합니다.

 

이처럼 디자인 패턴의 원래 목적은 문제 해결 시간을 단축시키는 것이었지만 문제 영역과 패턴을 정확히 매칭시키기 어렵고 Over-Engineering이 나오는 경우가 많기 때문에 실제 개발에서는 디자인 패턴을 직접적으로 사용하지 않는 것이 좋습니다.

 

디자인 패턴의 가치는 그보다도 의사소통 비용을 줄여줄 수 있다는 데 있습니다. 서로가 패턴에 대해 잘 알고 있다면 복잡한 설명 없이도 아주 간단하게 의사소통이 이루어질 수 있죠. 이를테면 이 예를 설명하기 위해 복잡하게 LoggableStatement의 구조를 설명하는 것 대신에 ‘이 LoggableStatement 클래스는 Statement의 데코레이터로 로깅을 추가로 하게 되어 있어’라고 말할 수 있다는 것입니다. 즉 패턴은 해결책으로서보다 의사소통을 위한 용어로서 활용하는 거라고 생각하는 것이 좋습니다.

 

자바빈즈의 해악
패턴 오용의 한 예로 자바빈즈를 들 수 있습니다. ‘해악’이라는 표현까지 써가면서 자바빈즈를 언급한 것에 대해 놀라는 사람도 많을 것입니다. 자바를 배우면서 보통 자바빈즈의 장점에 대한 이야기를 많이 듣게 되니까요. 하지만 점점 자바빈즈를 해로운 것으로 인식하는 사람들이 많아지고 있습니다. 먼저 자바빈즈가 무엇인지부터 다시 살펴봅시다.

 

자바빈즈는 원래 컴포넌트 아키텍처를 자바에서 구현하기 위해 도입된 기술입니다. 객체의 프로퍼티의 조작 메커니즘을 제공하여 재사용성이 높고 유연한 컴포넌트를 만드는 것이 목적이죠. 자바빈즈의 특징으로 언급되는 것이 여러 가지가 있지만 다른 자바 객체와 구별되는 특징은 프로퍼티 관리 방식입니다. 프로퍼티로 private 필드를 두고 그 필드에 대한 public getter/setter를 만들어서 이를 이용해서 프로퍼티에 접근을 합니다.

 

사실 이것은 일반적인 필드 인캡슐레이션(field encapsulation)과 크게 다르지 않습니다. 다른 점은 이 프로퍼티의 이름과 getter/setter의 이름에 네이밍 룰을 부여해서 빈즈 API를 통해서 프로퍼티의 이름으로 이 프로퍼티를 읽거나 쓸 수 있다는 점입니다.

 

예를 들어 name이라는 프로퍼티가 있다면 getName, setName이라는 이름으로 getter/setter를 만들고 빈즈 API에서는 name이라는 이름으로 프로퍼티에 접근할 수 있습니다. 즉, 일반적인 필드 인캡슐레이션을 하면 프로퍼티에 접근하기 위해 메쏘드명을 코드에 직접 써야 하지만 빈즈 규약에 맞게 getter/setter를 만들고 빈즈 API를 이용하면 String으로 주어진 프로퍼티에 접근하는 것이 가능해진다는 것입니다.

 

이 점을 이용하면 복잡한 애플리케이션에서 자바빈즈 객체의 프로퍼티를 동적으로 쉽게 조작할 수 있습니다. 그래서 정보 은폐(information hiding, data hiding)와 동적인 프로퍼티 접근의 두 마리 토끼를 동시에 잡는 것이 자바빈즈의 목적입니다.

 

그러면 무엇이 문제일까요? 우선 첫 번째 목적인 정보 은폐가 제대로 되지 않습니다. 사실 필드 인캡슐레이션은 정보 은폐를 위한 것이라고 하지만 이것은 허구에 불과합니다. 실질적으로 프로퍼티에 대한 접근성은 public field와 private field, public getter/setter가 완전히 동일합니다. 똑같이 읽고 쓸 수 있는데 뭐가 다르겠습니까. 단지 좀 더 조심스럽게 쓰지 않을까라고 추측하는 것뿐입니다. 그래서 자바빈즈를 사용할 때 꼭 필요한 경우가 아니라면 setter는 만들지 않기를 권하기도 합니다.

하지만 일반적인 필드 인캡슐레이션이라면 setter를 만들지 말라는 조언을 할 수 있겠지만 자바빈즈는 그 목적이 프로퍼티를 동적으로 조작하는 것이므로 컴포넌트로서의 역할을 제대로 하려면 setter도 당연히 필요합니다. 결국 필드 인캡슐레이션으로 인해 코드는 길어졌지만 정보 은폐는 달성하지 못했으므로 실질적인 이득이 없습니다.

 

또 다른 문제는 자바빈즈의 한계에 관한 것입니다. 동적으로 프로퍼티의 내용에 접근할 수는 있지만 프로퍼티를 추가하거나 없애는 정도의 유연성은 소화할 수 없습니다. 많은 경우 자바빈즈를 데이터 전송 객체(Data Transfer Object)로 사용하는데 이런 경우는 프로퍼티의 내용 뿐 아니라 프로퍼티 종류 자체가 자주 변합니다. 그런데 자바빈즈로 만들게 되면 프로퍼티가 늘어날 때마다 필드와 메쏘드를 추가해야 합니다. 그리고 데이터의 종류가 달라질 때마다 새로운 클래스를 만들어야 하구요. 자바 컬렉션에 Map이 있는데 이런 비효율을 감당해야 할 이유는 별로 없습니다.

 

Map에서는 프로퍼티에 동적으로 접근하는 것은 물론이고 프로퍼티가 자유롭게 늘어나거나 줄어들 수 있습니다. 게다가 빈즈 API는 String으로 주어진 프로퍼티명에 대한 getter/setter를 호출하기 위해 내부적으로 Introspection이라는 기술을 사용하는데 이것은 자바의 Reflection API를 이용합니다. 때문에 성능상에서도 해싱(Hashing)을 이용하는 HashMap에 비해 나은 것이 없습니다.

 

그래서 자바빈즈의 단점에 대해 많은 사람들이 공감하기 시작했고 최근에 등장하는 각종 프레임워크들도 이런 경향을 반영하고 있습니다. 초기의 프레임워크들은 자바빈즈를 활용하는 기능들을 제공했으나 최신 프레임워크들은 자바의 컬렉션을 적극적으로 활용하고 있습니다.

 

일반적으로 데이터 전송 객체의 경우는 Map을 그냥 사용하는 것이 좋고 그 외에 복잡한 로직을 수행하는 객체인데 프로퍼티를 많이 사용해야 한다면 Map을 데코레이터로 사용하면 편리합니다. 그리고 일반적으로 클래스를 설계할 때 setter는 되도록 만들지 않는 것이 좋습니다.

 

프레임워크의 가치
프레임워크(framework)란 단어는 여러 가지 의미로 사용되지만 보통은 어떤 영역의 API들을 사용하기 편리한 형태로 포장해놓은 것을 말합니다. 이를테면 자바 컬렉션 프레임워크는 자바에서 각종 컬렉션을 사용하기 쉽게 모아놓은 것이고 자카르타 프로젝트의 BSF(Bean Scripting Framework)는 자바 애플리케이션에서 자바빈즈나 스크립팅을 하기 쉽게 만든 API의 집합입니다. 그리고 지금 논의하려는 웹 애플리케이션 프레임워크는 웹에서 서블릿 API들을 좀 더 편리하고 객체지향적으로 사용할 수 있게 해주는 것입니다.

 

현재 웹 애플리케이션 프레임워크의 주류는 단연 스트럿츠(struts)입니다. 전 세계 자바 웹 개발자의 60%가 스트럿츠를 사용하고 있다고 하죠. 하지만 스트럿츠가 있음에도 계속해서 새로운 프레임워크가 등장하고 있습니다. 현재까지 공개된 프레임워크의 숫자는 이미 60개를 넘습니다. 스트럿츠에는 어떤 부족함이 있길래, 그리고 다른 프레임워크들은 뭐가 부족하길래 개발자들이 새로운 프레임워크를 계속 만들까요.

 

* 프레임워크의 정의

스트럭처(structure), 아키텍처(architecture), 프레임워크(framework)란 용어는 기술과 시대가 변하면서 조금씩 그 의미를 달리해 가는 것 같습니다. 스트럭처라는 용어가 Tree와 같은 계층적인 탄탄한 기반 구조를 말하는 반면, 프레임워크는 다소 수평적인 의미를 가지는 하부 구조를 나타냅니다. 또한 아키텍처라는 것은 이 두 부분을 모두 포함하는 더 포괄적인 개념 체계적인 기반 구조를 나타냅니다.

 

그러나 프레임워크이란 용어는 스트럭처나 아키텍처보다 더 낮은 레벨의 의미를 지닙니다. 즉, 프레임워크의 실체는 때론 API의 집합으로 나타나기도 한다는 것이지요. 썬 마이크로시스템즈의 Java Activation Framework가 그러하고 Java Media Framework가 그러합니다. 그러나 최근에 와서 IBM의 샌프란시스코 프레임워크(San Francisco Framework)라는 용어가 등장하면서 ‘반제품’의 의미를 강하게 띄는 것 같습니다.

 

알다시피 샌프란시스코 프레임워크는 정형화된 업무를 위한 비즈니스 컴포넌트를 미리 만들어 두고, 이를 조립함으로써 생산성을 극대화시키자는 것이 요지입니다. 어쨌거나 현재의 프레임워크란 것은 ‘기반 틀 구조’라는 모호한 추상적인 개념이기보다는 물리적인 실체이면서 반제품 성격의 구체적이고 체계화된 API를 제공하는 개념이라고 봐야 할 것입니다.

- javaservice.net 이원영님의 글

프레임워크의 가치는 기술적인 영역에서 소모적이고 반복적인 코딩을 프레임워크가 소화하여 개발자가 비즈니스 로직에만 전념할 수 있도록 해주는 데 있습니다. 이를테면 웹 프로그래밍을 한다면 HTTP 프로토콜도 어느 정도 알아야 하고 서블릿 API를 익히고 쿠키도 쓸 줄 알아야 합니다.

 

하지만 JSF(Java Server Faces) 같은 프레임워크를 이용하면 일반 자바 GUI 프로그래밍을 하듯이 프로그래밍할 수 있습니다. 서블릿 API를 몰라도 웹 애플리케이션을 잘 만들 수 있는 것이죠. 로깅(logging) 프레임워크가 없다면 개발자가 로그를 남기고 싶을 때 직접 파일을 열어서 파일에 내용을 쓰는 등의 복잡한 코드를 써나가야 하지만 log4j와 같은 프레임워크를 쓰면 단순히 Logger 객체에 로그 메시지를 넘겨주는 것으로 로깅을 할 수 있습니다.

 

EJB도 이런 프레임워크의 일종입니다. EJB가 없다면 분산 트랜잭션이 생길 때마다 개발자가 직접 분산 트랜잭션 관리를 해야 하지만 EJB 엔진이 이를 대신해주면 개발자는 비즈니스 로직의 구현에만 집중할 수 있죠. 이처럼 기술적인 영역이 달라질 때마다 개발자는 새로운 API를 익히고 또 소모적이고 반복적인 코드를 써야 하는데 해당 분야에 좋은 프레임워크가 있으면 중복 코드도 쉽게 제거할 수 있고 기술적인 세부사항에 신경쓰지 않아도 됩니다.

 

결국 프레임워크의 가치는 개발자의 할 일을 줄여주는 데 있는 것입니다.

하지만 간혹 프레임워크의 가치를 유지보수의 용이함에서 찾는 경우가 있습니다. 프레임워크가 꽉 짜놓은 틀 안에서만 코딩을 하면 개발자들이 모두 일관된 방식으로 개발하기 때문에 다른 사람이 작성한 코드를 파악하기 쉬워서 유지보수하기도 쉽다는 것이죠.

 

그러나 이렇게 일관성을 필요 이상으로 강조하면 때때로 문제가 될 수 있습니다. PEAA(Patterns of Enterprise Application Architecture)에서는 기업용 애플리케이션(Enterprise Application)에는 모두 각각의 어려움을 가지고 있는데 이 어려움은 모두 종류가 다르기 때문에 이 모든 어려움을 해결하는 하나의 아키텍처는 불가능하다고 말합니다. 다른 문제 영역에는 다른 해결책을 써야한다는 것이죠. 그런데 프레임워크를 통해서 지나치게 틀을 고정시켜 놓게 되면 그 틀에서 수용할 수 없는 문제를 만났을 때 아주 비효율적인 코딩을 하게 됩니다. 물론 이것은 프레임워크 자체가 가져오는 직접적인 문제는 아닙니다.

 

그보다도 프레임워크의 장점을 ‘일관성’에서 찾는 사람들이 일반적으로 범하는 오류입니다. 일관성의 가치를 지나치게 높게 평가한 나머지 코딩 패턴에 너무 많은 구속을 하게 되고 그 결과로 문제 영역에 맞는 패턴을 사용하는 것이 아니라 패턴에 코드를 끼워 맞추는 코드를 만들게 됩니다. 그래서 결과적으로 프레임워크를 유연하게 활용하지 못하고 오히려 코드가 더 길고 복잡해지는 경우가 많습니다. 앞서 언급한 디자인 패턴의 오용과 같은 것이죠.

 

사실 일관성의 문제는 프레임워크보다도 컨벤션을 통해 해결해야 하는 것입니다. XP에서는 코드에 대한 소유권을 개발자 한 명이 아니라 모두가 갖게 되는 코드 공동 소유(collective code ownership)를 권장합니다. 그래서 다른 사람의 코드를 읽고 수정하는 일은 아주 일상적인 일입니다. 이런 상황에서 개발자 각각의 개성이 강해서 서로의 코드를 읽기 어렵다면 문제가 크겠죠.

그럼에도 불구하고 XP는 프레임워크에 대해 약간의 부정적인 입장을 취하고 있습니다. 이것은 프레임워크가 디자인 패턴과 같은 문제를 발생시킬 수 있다고 보기 때문이기도 하지만 고정적인 패턴에 따라 코딩된 코드보다도 코드 컨벤션이 통일되고 리팩토링이 잘된 코드가 더 읽기 쉽다고 보기 때문입니다. 실질적으로 코딩 패턴의 강제를 통해 얻을 수 있는 것은 그리 많지 않습니다. 어차피 이해하기 가장 어려운 부분은 비즈니스 로직인데 이 부분까지 패턴이나 프레임워크로 강제할 수는 없기 때문입니다.

 

결국 아무리 다양한 기능을 제공한다 해도 결과적으로 개발자가 작성해야 하는 코드를 줄여주지 못한다면 좋은 프레임워크라고 할 수 없습니다. 스트럿츠에 대한 대안이 계속 쏟아져 나오고 있는 것도 실제적으로 스트럿츠가 개발자의 일을 많이 줄여주지 못하기 때문입니다.

 

스트럿츠의 강점으로 컨트롤러(controller)를 꼽는데 사실 스트럿츠의 컨트롤러는 HttpServlet의 역할을 Action으로, web.xml의 역할을 struts-config.xml로 옮겨온 것 밖에 없습니다. 사실 스트럿츠의 컨트롤러 기능 자체는 서블릿 API의 래퍼 몇 개로 완전히 대체할 수 있습니다. 이런 점을 깨달은 개발자들이 정말로 개발자의 일을 줄여주는 것을 목표로 다양한 프레임워크를 쏟아내고 있는 것입니다.

 

프레임워크의 구성 요소
실제로 어떤 프레임워크를 사용하든 그대로 사용하는 경우는 그다지 많지 않습니다. 다른 프레임워크와 결합해서 쓰기도 하고 상속을 통해서 확장하거나 혹은 직접 소스코드를 고쳐서 사용하기도 합니다. 어차피 모든 영역에 맞는 프레임워크는 없기 때문에 각자의 환경에 맞게 커스터마이징하는 것은 바람직한 일입니다. 그래서 이번에는 프레임워크를 구성하는 요소들에 대해 살펴보겠습니다.

 

제어 구조의 반전
요즘 나오는 프레임워크들은 공통적으로 제어 구조의 반전(IoC, Inversion of Control)를 내세우고 있습니다. 제어 구조의 반전이라는 말의 의미는 개발자가 제어하던 것을 프레임워크로 옮겨서 프레임워크에서 제어한다는 것입니다.

 

예를 들면 GUI로 입력을 받는 프로그램을 개발자가 직접 모든 부분을 만든다면 입력 항목을 출력하고 사용자의 입력을 기다린 다음 사용자의 입력을 판단해서 로직을 실행하는 코드를 모두 순차적으로 쓰게 됩니다. 그런데 이걸 GUI 프레임워크를 이용하면 화면과 입력에 관한 부분은 프레임워크가 처리하고 사용자는 단지 사용자가 입력했을 때 반응하는 이벤트 핸들러만 작성합니다. 즉 입력 제어가 개발자에서 프레임워크로 이동하는 것입니다.

 

윈도우 프로그래밍에서 콜백(callback)이라고 부르는 것도 비슷한 개념입니다. 사실 서블릿 자체도 일종의 프레임워크이고 이미 IoC가 일어나고 있습니다. HTTP 요청을 대기하고 받아들여서 서블릿을 실행하는 코드를 개발자가 작성하는 것이 아니라 서블릿 엔진이 제공하고 개발자는 단지 서블릿만을 작성하게 되죠. 결국 이런 컨트롤의 반전을 통해서 중복 코드를 효과적으로 제거할 수 있습니다. 프레임워크의 목적이 소모적이고 반복적인 코드를 제거하는 것인 만큼 대부분의 프레임워크에는 프레임워크 개발자가 의도했든 아니든 IoC가 사용되고 있습니다.

 

하지만 IoC 역시 주의해서 사용해야 합니다. 일상적인 프로그램과는 달리 제어 구조가 거꾸로 되기 때문에 디버깅하기도 어렵고 컨트롤이 반전된 부분에 대한 선행 지식이 없으면 이해하기 어려운 코드가 됩니다. 그래서 IoC는 꼭 필요한 경우가 아니면 사용하지 않는 것이 좋습니다.


MVC 패턴
요즘 프레임워크가 IoC를 강조한다면 스트럿츠처럼 좀 오래된 프레임워크들은 MVC(Model-View-Controller)를 내세웠습니다. UI가 포함된 프레임워크에는 거의 필수적으로 사용되는 패턴입니다. 원래 MVC는 웹보다도 일반 GUI 애플리케이션을 개발할 때 UI(User Interface)와 비즈니스 로직을 효과적으로 분리하기 위해서 고안되었습니다. 현재는 웹에서도 그 효과가 입증되었기 때문에 널리 쓰이고 있습니다.

 

MVC의 구조는 MFC 프로그램에서 등장했던 Document-View 구조를 한 차원 더 발전시킨 것으로 프로그램의 구성 요소를 모델, 뷰, 컨트롤러로 나누어서 각각 다른 역할을 맡게 하는 것입니다. 모델은 비즈니스 로직을 담는 객체를 말하며 이 모델은 PEAA에서 말하는 도메인 모델이 됩니다.

데이터베이스에 접근하는 것도 모델 객체의 몫입니다. 모델 객체들 자체로 UML 클래스 다이어그램을 그린다면 그 자체로 비즈니스 로직의 표현이 될 수 있습니다. 뷰는 화면 UI를 구성하는 요소이며 일반적으로 웹에서는 JSP가 뷰의 역할을 맡습니다.

 

그리고 컨트롤러는 사용자의 요청을 받아서 모델 객체를 실행하고 그 결과를 뷰로 전달하는 역할을 맡게 되는데 일반적으로 이 부분은 프레임워크에서 담당하며 개발자는 어떤 모델을 실행하고 어떤 뷰를 선택할 것인지를 컨트롤러에 알려주기만 하면 됩니다. 이런 구조가 이상적인 MVC 패턴의 구조입니다.

 

MVC 패턴의 가장 큰 장점은 모델과 뷰의 분리를 통해 화면 UI를 위한 코드와 비즈니스 로직을 위한 코드가 섞이지 않는다는 것입니다. 그래서 때때로 화면 UI 개발자와 비즈니스 로직 개발자를 따로 두는 것도 가능합니다. 그리고 이 패턴을 통해 프로그램 디자인의 기본 원칙인 low coupling, high cohesion을 자연스럽게 달성할 수 있죠.

 

무언가 변경하고 싶을 때 여러 컴포넌트들을 같이 변경해야 하는 coupling은 줄이면서 컴포넌트간의 협력(cohesion)은 컨트롤러를 통해 자유롭게 할 수 있는 것입니다. 또한 모델과 뷰가 분리되면 한 모델에 여러 가지 다양한 뷰를 붙이는 것도 가능하고 그 반대의 경우도 가능합니다. 그래서 요구사항의 복잡도는 높지만 규격이 잘 정해진 애플리케이션을 만들 때는 개발자의 일을 획기적으로 줄일 수 있습니다. MVC 프레임워크란 결국 이런 모델과 뷰의 분리를 효과적으로 할 수 있는 컨트롤러를 제공한다는 데에 그 가치가 있습니다.


XML 설정 파일의 역할
프레임워크에서 또 하나 중요한 것은 설정 파일의 활용입니다. 유행처럼 번져나가기 시작한 XML 설정 파일은 요즘 거의 프레임워크의 필수사항처럼 되어가고 있습니다. 초기에 설정 파일이 등장하기 시작한 이유는 소스의 내용 중 상수에 해당하는 값들을 설정 파일로 빼 놓으면 컴파일을 다시 하지 않아도 동작을 변경할 수 있기 때문입니다. C++ 등의 언어에서는 컴파일 후에 링크까지 해야 하기 때문에 이 차이는 적지 않습니다.

 

하지만 자바에서는 필요한 부분만 간단하게 컴파일하면 바로 동작하게 할 수 있고 XP의 지속적인 통합(Continuous Integration)이 일반화되면서 소스를 변경하는 비용이 크게 줄어들었죠. 그래서 사실 지금은 XML 설정 파일에 상수를 넣어두는 것이 자바 코드에 넣는 것보다 개발시의 변경 비용이 적다고 하기 힘듭니다.

 

하지만 여전히 애플리케이션 사용자의 입장에서는 XML이 훨씬 변경하기 쉽습니다. 클라이언트로 배포된 프로그램 혹은 서버에 배치된 프로그램의 설정을 바꾸기 위해 다시 빌드를 할 수는 없는 일이니까요.

 

하지만 이런 비용적인 측면 외에도 XML 설정 파일은 자바 코드보다 좀 더 정보를 서술적(descriptive)으로 담아놓을 수 있다는 장점이 있습니다.

XML은 구조적인 정보와 메타 정보를 포함한 모든 정보를 한 눈에 알아보기 쉽게 정리할 수 있습니다. 또한 자바 코드에서 여러 클래스로 분산될 수 있는 설정 내용을 개발자가 관리하기 편한 단위로 손쉽게 통합해서 관리할 수 있죠.

 

자바의 프로퍼티도 설정 파일 용도로 많이 사용되어 왔습니다만 구조적인 정보를 제대로 정의하기 어렵기 때문에 설정 파일로는 부적합합니다. 게다가 프로퍼티는 한글 문제도 있기 때문에 앞으로 프로퍼티는 쓰지 않는 것이 좋다고 봅니다.

 

뭐든지 잘못 쓰면 해가 된다는 점에서 XML 역시 예외가 아닙니다. XML 설정 파일을 오용하는 대표적인 사례로 Jelly 스크립트가 있습니다. 자카르타 commons 프로젝트의 하나인데 XML로 스크립팅을 할 수 있게 해주는 엔진입니다. Jelly 스크립트의 파워는 여전히 진짜 프로그래밍 언어에 미치지 못하면서 Jelly로 스크립트를 짜면 가독성도 떨어지고 리팩토링하기 어렵기 때문에 점점 지저분한 코드가 되고 맙니다. Ant도 이와 비슷한 문제점이 있습니다. 그래서 복잡한 제어 흐름이 필요한 부분에서는 자바 코드를 직접 이용하고 이것이 무겁다면 좀더 가벼운 스크립트 언어를 자바에 임베딩해서 사용하는 것이 좋습니다.

 

단, SQL의 경우는 언어 자체가 결과물에 대한 정의를 표현하는 것이기 때문에 복잡한 제어 구조가 필요하지 않아 XML로 별 무리 없이 소화가 가능합니다. 실제로 XML과 JDBC를 결합시킨 iBatis에서 XML로 SQL들을 관리해보면 실제 자바 코드보다 훨씬 쉽고 보기 좋게 코딩할 수 있습니다.

또 한 가지 주의해야 할 것은 XML 설정 파일을 이용하게 만들더라도 XML 없이도 프레임워크가 동작하게 만들어야 한다는 것입니다. 앞서의 iBatis는 구현된 기능들이 정말 편리하고 좋지만 각 컴포넌트들이 지나치게 강하게 결합되어 있고 XML 설정 파일 없이는 아무 것도 하지 못합니다. 그래서 추가 기능을 넣고 싶다거나 개선하고 싶은 부분이 있을 때 상속 등의 방법으로는 해결하기 어렵고 거의 대부분 소스를 수정해야 해결할 수 있죠. 그래서 프레임워크를 만들더라도 일단 자바 코드로 모두 만든 다음 마지막에 XML로 설정 내용들을 빼는 작업을 하는 게 좋습니다.

 

살펴볼만한 프레임워크
그림도 코드도 없는 지루한 텍스트를 읽어 내려오느라 지루했을 독자들을 위해 구체적인 프레임워크들을 몇 가지 소개할까 합니다. 웹 애플리케이션 프레임워크 뿐 아니라 여러 영역에서 유용하다고 생각되는 것들을 뽑았습니다. 단지 필자의 경험 속에서 유용했던 것들이므로 한 번 살펴볼 가치가 있다는 정도로 받아들이면 될 것입니다.

 

스트럿츠와 JSF
웹 애플리케이션 프레임워크로는 스트럿츠가 현재의 주류이며 JSF가 일부에서 미래의 주류로 꼽힙니다. 모두 한 사람이 주도한 프레임워크입니다.

필자의 판단을 덧붙이면, 스트럿츠는 실제로 개발자의 일을 거의 줄여주지 못하며 오히려 늘이는 경우조차 간혹 있습니다. 스트럿츠를 쓸 바에는 직접 스트럿츠와 유사한 형태로 서블릿을 직접 사용하도록 재구성하는 것이 좋으리라 생각됩니다. 물론 유효성 검사 기능과 Tiles는 좋은 기능이지만 이들은 스트럿츠와 분리해서도 사용 가능합니다.

JSF는 미래의 주류라고 하기엔 부족함이 너무 많습니다. 썬에서 Java Studio Creator를 통해 멋진 가능성을 내보이긴 했지만 그런 IDE를 사용하지 않을 때의 JSF 코딩은 너무 번잡하고 복잡도도 큽니다. 이는 구조적으로 아직 개선의 여지가 많다는 것이죠. JSF 스펙 역시 꾸준히 발전하고 있으니 미래는 어떨지 모르겠지만 지금 JSF를 사용하는 것은 시기상조라 생각됩니다. 오히려 JSF와 비슷한 구조인 Tapestry가 더 완성도가 높다는 평가를 받고 있습니다.

 

Turbine
Turbine도 꽤 널리 쓰이는 프레임워크 중 하나입니다. 기본 틀은 스트럿츠와 비슷하나 훨씬 많은 부분들을 커버하고 있기 때문에 코딩량은 상당히 줄어듭니다. 개발자의 편의를 위한 클래스들도 많으며 Turbine 개발 과정에서 만들어진 컴포넌트들이 상당수 독립해서 별도의 프로젝트로 진행되고 있습니다. 전에는 유연성이 다소 떨어진다는 것이 단점이었는데 점점 나아지고 있습니다.

 

Spring
Spring은 요즘 주목 받는 프레임워크로 AOP(Aspect Oriented Programming)를 내세우고 있으며 low coupling, high cohesion이라는 디자인 원칙이 잘 지켜진 프레임워크입니다. 딱히 웹을 염두에 두었다기보다 종합적인 애플리케이션 프레임워크이기 때문에 부분별로 사용할 수도 있고 다른 프레임워크에 붙여서 사용하기도 합니다. 데이터베이스와 관련해서 풍부한 API를 제공하며, Mock Object를 프레임워크 자체에서 제공하고 있다는 것도 특이한 점입니다. 현재 가장 추천할 만한 프레임워크입니다.

 

iBatis
데이터베이스와 관련해서는 앞서의 Spring으로도 충분할 수 있지만 XML로 SQL을 관리하고자 한다면 iBatis가 최선의 선택일 것입니다. 내부적인 코드의 품질은 다소 떨어지지만 기능상으로는 별다른 부족함이 없습니다. Spring과 iBatis가 JDBC에 대한 단순 래퍼 수준의 API를 제공한다면 Hibernate는 OR(Object-Relation) 맵핑을 통해서 개발자는 객체만을 바라보고 데이터베이스를 간접적으로 이용할 수 있는 프레임워크입니다. 기본적으로 구조가 깔끔하고 편리하다는 평가를 받고 있으며 비즈니스 로직을 도메인 모델로 직접 다룰 수 있다는 점에서 높이 평가할만 합니다.

 

OSCache, Quartz, JUnit
이외에 웹에서 캐싱 기법으로 성능을 향상시키기 위한 OSCache, 스케쥴러를 만들기 위한 API인 Quartz도 아주 유용합니다. 물론 테스팅 프레임워크로 JUnit은 언급할 필요가 없을 것입니다. 이런 프레임워크들을 꼭 사용하지 않더라도 한 번 살펴보면 배울 만한 점들을 발견할 수 있고 이런 점들이 나중에 도움이 되는 경우가 적지 않을 것입니다.

 

많은 논란거리들
패턴이나 프레임워크는 특정 영역에 대해 어느 정도 검증된 해결책입니다. 그래서 때때로 이에 대한 맹신이 나타나기도 합니다. 검증된 방법이라는 논리로 너무 쉽게 문제 영역에 패턴을 적용하거나 프레임워크를 사용하곤 하죠. 하지만 그 문제 영역이 패턴이나 프레임워크를 활용할 수 있는 바로 그 영역이라는 점은 검증하기 힘든 것입니다. 때문에 패턴 중심적인 사고 방식이나 프레임워크에 끼워 맞추는 코딩을 지양하고 문제 영역의 해결에 충실하면서 패턴을 생각하는 것이 더 좋습니다.

 

바둑 격언 중에 ‘정석은 배우고, 잊어버려라’라는 말이 있습니다. 정석을 익혀서 알고는 있되 실전에서는 정석에 집착하지 말고 상황에 따라 유연하게 대처해야 좋은 바둑을 둘 수 있다는 뜻입니다. 패턴은 프로그래밍의 정석과도 같습니다. 상황에 대한 검증된 해결책이지만 그 상황은 계속 변하고 그 변화에 빠르게 대처해야 합니다.

 

패턴이나 프레임워크에 대한 맹신으로 빠져드는 것을 경계하기 위해 어떤 사람들은 무엇이든지 장단점이 있으니 너무 집착하지 말라고 말합니다. 이것은 옳은 말이지만 사실 큰 의미가 있는 말은 아닙니다. 무엇이든 장단이 있다는 사실은 누구도 부인할 수 없지만 또한 누구도 부인할 수 없는 자명한 사실이기에 의미 없는 말이기도 합니다. 실제 상황에 맞게 잘 쓰려면 구체적으로 장점이 무엇인지, 단점이 무엇인지를 파악하는 것이 중요합니다. 앞서 패턴과 프레임워크를 다루면서 각각의 단점들을 구구절절이 따지고 들어간 것도 그런 이유입니다.

 

나날이 신기술이 쏟아져 나오고 있지만 그에 못지않게 패러다임도 많이 변화하고 있습니다. 유연성 높은 컴포넌트 기술로 주목받던 자바빈즈가 해로운 것이라는 평가를 받기 시작하고, 많이 써야 좋은 것이라고 하던 주석을 이제는 적게 쓸수록 좋은 것이라고 합니다.

 

신기술을 따라가는 것도 필요하지만 이러한 패러다임의 변화를 읽는 것이 더 중요합니다. 현재의 방식이 지금은 가장 효율적인 방식일 수 있겠지만 지금보다 훨씬 더 효율적인 방식이 나오는데 모르고 있으면 도태될 수밖에 없습니다. 흔히 프로그래머는 워낙 기술이 빨리 발전하기 때문에 후배들을 당할 수가 없다는 말을 하곤 합니다.

 

하지만 현실은 결코 그렇지 않습니다. 기술의 변화는 단지 API가 변하는 것 뿐 그렇게 중요한 것은 아닙니다. 프로그래밍의 주요한 패러다임은 그렇게 빠른 속도로 변하지 않으며 또한 그런 패러다임의 변화를 주도하는 것은 반짝이는 아이디어가 아니라 경험 속에서 축적된 고민들입니다. 끊임없이 고민하면서 프로그래밍을 해왔다면 이런 변화를 따라잡는 것은 어려운 일이 아닙니다.

 

좋은 프로그래머가 되기 위해서 폭넓은 지식과 빠른 학습 능력은 물론 필수적인 것입니다만 그보다 더 중요한 것은 늘 고민하는 자세와 깊이 있는 사고 능력입니다. 변화를 따라잡는 힘은 빠른 학습 능력에서 나오는 것이 아니라 변화의 중요성을 인식하고 발전을 위해 고뇌하는데서 나오는 것입니다.

이번 연재는 아마 지난 두 번보다 훨씬 더 많은 논란거리들을 담고 있을 것입니다. 패러다임은 한 사람의 고민에 의해서도 발전하지만 많은 사람들의 머리를 맞댄 토론에서도 발전합니다. 필자의 글에 반론이 있거나 더 깊은 논의를 원하는 독자들은 youngrok.com이나 javaservice.net, 혹은 c2.com에 제시해 주세요. 서로를 발전시킬 수 있는 좋은 기회가 있길 바랍니다.@

 

* 이 기사는 ZDNet Korea의 제휴매체인 마이크로소프트웨어에 게재된 내용입니다.

'그거 > Tech' 카테고리의 다른 글

Velocity  (0) 2007.03.29
DOM(Document Object Model)  (0) 2007.03.20
Crossing borders: JavaScript의 특징 (한글)  (0) 2007.03.13
JSON vs. XML  (0) 2007.03.13
JSON (JavaScript Object Notation) 개요  (0) 2007.03.13
:
Posted by 뽀기
2007. 3. 14. 10:29

Recommended Documents for Beginners 그거/Java2007. 3. 14. 10:29


    서블렛 + JDBC 연동시 코딩 고려사항 -제1탄-
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=servlet&c=r_p&n=968185187

    서블렛 + JDBC 연동시 코딩 고려사항 -제2탄-
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=servlet&c=r_p&n=968522077

    Re: DB Connection Pool: Orphan and Idle Timeout
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=servlet&c=r_p&n=1005294960

    Connection/Statement 최대 동시 Open 수
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=jdbc&c=r_p&n=972287002
    '최대열기 커서수를 초과했습니다' 오류 튜닝법
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=servlet&c=r_p&n=976511779

    Connection Pool & PreparedStatement Cache Size
    http://www.javaservice.net/~java/bbs/read.cgi?m=appserver&b=was&c=r_p&n=995572195

    Understanding JavaServer Pages Model 2 architecture
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=servlet&c=r_p&n=944700949

    Re: 서블릿에서의 Global 변수
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=servlet&c=r_p&n=959175733

    Re: [팁] EUC-KR, KSC5601, euc-kr , etc.. 20010321  손님(guest)
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=servlet&c=r_p&n=985157298

    "/servlet/test?name=이원영&address=서울 도곡동"
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=servlet&c=r_p&n=960356845

    JSP에서 DB연결의 문제
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=servlet&c=r_p&n=954339077

    Re: DB Connection Pool & 그 외의 것들
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=servlet&c=r_p&n=960704622

    Oracle OSDK DB Connection Pool 사용 방법
    http://www.javaservice.net/~java/bbs/read.cgi?m=appserver&b=oas&c=r_p&n=965358223

    소스 Review 2 - ConnectionPool Q&A -
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=servlet&c=r_p&n=971498129

    73   Hans Bergsten Connectionpool 을 Thread간통신에 붙여보았습니다
    http://www.javaservice.net/~java/bbs/read.cgi?m=dbms&b=jdbc&c=r_p&n=1011919113

    AIX Java App. SIGSEGV 진단 방법에 관한 예
    http://www.javaservice.net/~java/bbs/read.cgi?m=appserver&b=was&c=r_p&n=1000371692

    (웹로직)엔티티 빈 자동 생성 프로그램입니다.
http://www.javaservice.net/~java/bbs/read.cgi?m=appserver&b=weblogic&c=r_p&n=1009036410

    Re: WebLogic JSP와 Servlet에서 한글 깨지는 문제
http://www.javaservice.net/~java/bbs/read.cgi?m=appserver&b=weblogic&c=r_p&n=957179071

    jspSmartUpload 패키지 사용을 금합니다
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=servlet&c=r_p&n=995622233

    파일 Download에 관한 참고
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=servlet&c=r_p&n=992449785

    JDBC 코딩시 Tips - 대량데이타 Select -
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=servlet&c=r_p&n=982581488

    소스 Review 7 - Servlet/JDBC 클래스 디자인
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=servlet&c=r_p&n=983500745

    Re: 특수한 기호를 HTML TAG로 변환하는 방법
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=www&c=r_p&n=934284565

    [자바의 숫자표현] 어디까지 가능할까 !!!
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=javatip&c=r_p&n=920434490

    Object Pool Contorl 기법
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=javatip&c=r_p&n=974750350

    PPC(Peek Point Control) 기법
    http://www.javaservice.net/~java/bbs/read.cgi?m=qna&b=consult&c=r_p&n=996642473

    Re: 각 환경별 Servlet & JNI 정리
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=servlet&c=r_p&n=956659322
    JNI - Native Method Pooling 기법
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=javatip&c=r_p&n=975913198
    41 JNI 2탄입니다. 
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=javatip&c=r_p&n=963972664
    Java에서 signal 처리 - JNI -
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=javatip&c=r_p&n=978373413

    26   JavaStackTrace(javacore/ThreadDump) 분석방법
    http://www.javaservice.net/~java/bbs/read.cgi?m=etc&b=jdk&c=r_p&n=1039372188

    TCP/IP Socket 프로그램 구현시 고려사항
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=javatip&c=r_p&n=1009171849
    Java Socket Utilities API : SocketUtil.java
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=javatip&c=r_p&n=976117970
    18   parse int and long type from byte[]
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=javatip&c=r_p&n=953537896
    99     Re: C <--> Java 임의의 데이타 타입 주고 받기
    http://www.javaservice.net/~java/bbs/read.cgi?m=qna&b=discussion&c=r_p&n=1029190704
    Re: [답변]자바와 C 간의 소켓 통신상 한글 문제
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=javatip&c=r_p&n=956208406
    109     Re: TCP Status & JNI & Socket Communication
    http://www.javaservice.net/~java/bbs/read.cgi?m=unix&b=unix&c=r_p&n=1022636033
    92   Re: TCP/IP ESTABLISHED & Connection Timeout
     http://www.javaservice.net/~java/bbs/read.cgi?m=unix&b=unix&c=r_p&n=1005195947
    Re: DB Connection Pool: Orphan and Idle Timeout
     http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=servlet&c=r_p&n=1005294960

    89 AIX 4.3.3 FIN_WAIT_1, FIN_WAIT_2, TIME_WAIT
     http://www.javaservice.net/~java/bbs/read.cgi?m=unix&b=unix&c=r_p&n=1002732057
    88 AIX4.3 "netstat -n" shows too many FIN_WAIT_2
     http://www.javaservice.net/~java/bbs/read.cgi?m=unix&b=unix&c=r_p&n=1002693727
    87 AIX 4.3 tcp keepalives & tcp_
     http://www.javaservice.net/~java/bbs/read.cgi?m=unix&b=unix&c=r_p&n=1002689463
    85 Solaris 2.x - Tuning Your TCP/IP Stack and 
     http://www.javaservice.net/~java/bbs/read.cgi?m=unix&b=unix&c=r_p&n=1002605172
    82 TCP/IP Normal Close Sequence
     http://www.javaservice.net/~java/bbs/read.cgi?m=unix&b=unix&c=r_p&n=994674560

    30   RMI 및 Object Serialization 세미나 자료 20011203  絶對高手(juldaegosu)
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=rmi&c=r_p&n=1007358212

    Orion Sever 에서 Apache SOAP 사용하기
    http://www.javaservice.net/~java/bbs/read.cgi?m=appserver&b=orion&c=r_p&n=985270192
    Orion Server에서 Apache SOAP Server와 Visual Basic SOAP Client 연동하기
    http://www.javaservice.net/~java/bbs/read.cgi?m=appserver&b=orion&c=r_p&n=985572196

    [세미나자료] 김덕태님의 "XML과 자바"
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=websvcs&c=r_p&n=955159322

    JSP에서 동적으로 생성된 xml보여주기
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=websvcs&c=r_p&n=985450162

    [필수]객체지향 개발자가 꼭 읽어야 할 책 4권 
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=javatip&c=r_p&n=988257203

    173   자바 프로그래밍 언어 정복(Garbage Collection 등 추가)  -출판초고,이덕하-
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=javatip&c=r_p&n=1012451983

    [필수]Design Pattern - javacompanion
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=javatip&c=r_p&n=977108336

    UML modeling 도구입니다.
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=javatip&c=r_p&n=977368627

    UML 개론 프리젠테이션 자료
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=javatip&c=r_p&n=979041227

    139   BROS - JavaBeans 의 변형입니다.
    http://www.javaservice.com/~java/bbs/read.cgi?m=devtip&b=javatip&c=r_p&n=999099977

    [온라인책] "Mastering EJB"- Ed Roman -
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=ejb&c=r_p&n=960552403
    41 EJB책한권 올립니다. (Mastering EJB)
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=ejb&c=r_p&n=1010784800

    EJB 사용을 위한 결정 요소
    http://www.javaservice.net/~java/bbs/read.cgi?m=devtip&b=ejb&c=r_p&n=973909127

    352  To EJB, or not to EJB
    http://www.javaservice.net/~java/bbs/read.cgi?m=qna&b=consult&c=r_p&n=1010454437

    [TIP]EJB 1.1 트랜잭션 속성에 관한 강좌
http://www.javaservice.com/~java/bbs/read.cgi?m=appserver&b=powertier&c=r_p&n=1002186458

    Usage & Limitations of Txn Control in WebLogic V5.1
http://www.javaservice.net/~java/bbs/read.cgi?m=appserver&b=weblogic&c=r_p&n=1000771631

    15 응용 S/W 아키텍춰 및 모델링
    http://www.javaservice.net/~java/bbs/read.cgi?m=appserver&b=was&c=r_p&n=930570297

    WEB개발 및 Application Server동향
    http://www.javaservice.net/~java/bbs/read.cgi?m=qna&b=consult&c=r_p&n=936757537

    Re: [답변] 자바 프로젝트 개발 방향
    http://www.javaservice.net/~java/bbs/read.cgi?m=qna&b=consult&c=r_p&n=955519088

    Re: 자바 프로젝트 개발 방향 - 답변2
    http://www.javaservice.net/~java/bbs/read.cgi?m=qna&b=consult&c=r_p&n=956198284

    Re: 자바 기술 Set 구성과 관련된 질문
    http://www.javaservice.net/~java/bbs/read.cgi?m=qna&b=consult&c=r_p&n=956676463

    RE: Applet에서 WAS의 Connection Pool 이용 ?
    http://www.javaservice.net/~java/bbs/read.cgi?m=qna&b=consult&c=r_p&n=1007007020

    대표적인 S/W 아키텍쳐 및 기술셋 예제
    http://www.javaservice.net/~java/bbs/read.cgi?m=qna&b=consult&c=r_p&n=957877865

    웹서비스(WebService) 관련 기술정보/예제 및 기사
    http://www.javaservice.net/~java/bbs/read.cgi?m=qna&b=consult&c=r_p&n=999508397

    Re: 세션클러스트링
    http://www.javaservice.net/~java/bbs/read.cgi?m=qna&b=consult&c=r_p&n=959681361

    어플리케이션서버와 데이타서버 분리의 장단점
    http://www.javaservice.net/~java/bbs/read.cgi?m=qna&b=consult&c=r_p&n=962379253

    Re: 어플리케이션서버와 데이타서버 분리의 장단점
    http://www.javaservice.net/~java/bbs/read.cgi?m=qna&b=consult&c=r_p&n=962585278

    Re: ◈ 대량의 동시사용자를 위한 조언
    http://www.javaservice.net/~java/bbs/read.cgi?m=qna&b=consult&c=r_p&n=974615872
   
    81   Re: [쇼핑몰] 대량의 고객정보 관리
    http://www.javaservice.net/~java/bbs/read.cgi?m=qna&b=consult&c=r_p&n=966762191

    [답변]쇼핑몰 퍼포먼스 관련
    http://www.javaservice.net/~java/bbs/read.cgi?m=qna&b=consult&c=r_p&n=980794006

    Oracle 8i Performance & Tuning
    http://www.javaservice.com/~java/bbs/read.cgi?m=dbms&b=dbms&c=r_p&n=1002277016

    응답시간 느린 HTTP 요청 두번이상 자동호출될 가능성
    http://www.javaservice.net/~java/bbs/read.cgi?m=qna&b=consult&c=r_p&n=982251775

    장애진단(Problem Determination) 절차 
    http://www.javaservice.net/~java/bbs/read.cgi?m=resource&b=consult&c=r_p&n=1023729776

    255   성능저하시, 모니터링 방법
    http://www.javaservice.net/~java/bbs/read.cgi?m=appserver&b=weblogic&c=r_p&n=1012075146

    106   RE: netstat & lsof
    http://www.javaservice.net/~java/bbs/read.cgi?m=appserver&b=engine&c=r_p&n=1007110331

    대량데이타 조회시 고려사항
    http://www.javaservice.net/~java/bbs/read.cgi?m=qna&b=consult&c=r_p&n=1000186035


◆  Web Application Servers

    어플리케이션서버
    http://www.javaservice.net/~java/bbs/index.cgi?m=appserver&b=appserver

    ** 이하 ABC 순입니다.

    ATG(Art Technology Group) Dynamo
     http://www.javaservice.net/~java/bbs/index.cgi?m=appserver&b=atg
    BEA/WebLogic Application Server
     http://www.javaservice.net/~java/bbs/index.cgi?m=appserver&b=weblogic
    Borland AppServer(Inprise Application Server)
     http://www.javaservice.net/~java/bbs/index.cgi?m=appserver&b=inprise
    Evermind Orion Server
     http://www.javaservice.net/~java/bbs/index.cgi?m=appserver&b=orion
    GemStone's GemStone/J
     http://www.javaservice.net/~java/bbs/index.cgi?m=appserver&b=gemstone
    HP Netaction Application Server
     http://www.javaservice.net/~java/bbs/index.cgi?m=appserver&b=eserver
    IBM WebSphere Application Server
     http://www.javaservice.net/~java/bbs/read.cgi?m=appserver&b=was&c=r_p&n=1010952419
    IONA iPortal Application Server (iPAS)
     http://www.javaservice.net/~java/bbs/index.cgi?m=appserver&b=ipas
    Marc Fleury's JBoss Application Server
     http://www.javaservice.net/~java/bbs/index.cgi?m=appserver&b=jboss
    Oracle 9iAS/OAS/OSDK
     http://www.javaservice.net/~java/bbs/index.cgi?m=appserver&b=oas
    Persistence PowerTier
     http://www.javaservice.net/~java/bbs/index.cgi?m=appserver&b=powertier
    Sun/Netscape iPlanet
     http://www.javaservice.net/~java/bbs/index.cgi?m=appserver&b=iplanet
    Sybase EAServer
     http://www.javaservice.net/~java/bbs/index.cgi?m=appserver&b=sybase
    TmaxSoft JEUS
     http://www.javaservice.net/~java/bbs/index.cgi?m=appserver&b=jeus
    서블렛엔진(JServ,Tomcat,JRun,..)
     http://www.javaservice.net/~java/bbs/index.cgi?m=appserver&b=engine


◆  Java Development Kit
    http://www.javaservice.net/~java/docs/jdk1.1.8/
    http://www.javaservice.net/~java/docs/jdk1.2.2/

    http://www.javaservice.net/~java/docs/j2sdk_1.3.1/
    http://www.javaservice.net/~java/docs/j2sdk_1.3.1/src/

◆  JAVA Specifications

   [Java 관련 Specifications]

   Java Language Specification -Second Edition-
      http://java.sun.com/docs/books/jls/second_edition/html/jTOC.doc.html

   Java Virtual Machine Specification
      http://www.java.sun.com/docs/books/vmspec/
      http://www.java.sun.com/docs/books/vmspec/2nd-edition/html/VMSpecTOC.doc.html
      http://www.java.sun.com/docs/books/vmspec/html/VMSpecTOC.doc.html

   Servlet Specification 2.3 ( Public Draft )
      ftp://ftp.javaservice.net/pub/java/spec/servlet23_PublicDraft1.pdf
      (1,475,499 bytes)

   Servlet Specification 2.2
      ftp://ftp.javaservice.net/pub/java/spec/servlet2_2-spec.pdf

   Servlet Specification 2.1
      ftp://ftp.javaservice.net/pub/java/spec/servlet-2.1.pdf

   JSP (Java Server Page) Specification 1.2 ( Public Draft )
      ftp://ftp.javaservice.net/pub/java/spec/jsp12_PublicDraft1.pdf

   JSP (Java Server Page) Specification 1.1
      ftp://ftp.javaservice.net/pub/java/spec/jsp1_1-spec.pdf

   JSP (Java Server Page) Specification 1.0
      ftp://ftp.javaservice.net/pub/java/spec/jsp1_0-spec.pdf
  
   JSP 0.91 programming reference : The IBM implementation version
http://www-4.ibm.com/software/webservers/appserv/doc/v30/ae/web/doc/whatis/icjspref.html

   OMG CORBA/IIOP 2.2 Specification : The Complete formal/98-07-01
      ftp://ftp.javaservice.net/pub/corba/spec/98-07-01.pdf (5,968,346 bytes)

   EJB(Enterprise JavaBeans) Specification Version 1.0
      ftp://ftp.javaservice.net/pub/java/spec/ejb_10.pdf  (330,603 bytes)

   Enterprise JavaBeans 1.0 CORBA Mapping
      ftp://ftp.javaservice.net/pub/java/spec/ejb-corba_10.pdf (35,377 bytes)

   JavaBeans Specification 1.01
      ftp://ftp.javaservice.net/pub/java/spec/beans_101.pdf (257,046 bytes)

   JDBC 2.0 Core API Specification
      ftp://ftp.javaservice.net/pub/java/spec/jdbc20.pdf (141,137 bytes)

   JDBC 2.0 Standard Extension API Specification (old)
      ftp://ftp.javaservice.net/pub/java/spec/jdbc20_stdext.pdf (154,667 bytes)

   JDBC 2.1 Core API Specification
      ftp://ftp.javaservice.net/pub/java/spec/jdbc2_1-spec.pdf (269,312 bytes)

   JDBC 2.0 Optional Package API specification: (new)
       NOTE: The javax.sql package is also called the JDBC 2.0 Optional Package API
             (formerly known as the JDBC 2.0 Standard Extension API).
      ftp://ftp.javaservice.net/pub/java/spec/jdbc20.stdext.pdf (327,005 bytes)


   Java Remote Method Invocation Specification (JDK 1.1, JDK 1.2)
     http://www.java.sun.com:80/products/jdk/1.1/docs/guide/rmi/spec/rmiTOC.doc.html
     http://www.java.sun.com:80/products/jdk/1.2/docs/guide/rmi/spec/rmiTOC.doc.html

   RMI Architecture and Functional Specification (JDK 1.2)
      ftp://ftp.javaservice.net/pub/java/spec/rmi-spec-JDK1_2.pdf
                                                    (495,423 bytes)

   Java Transaction API specification
      ftp://ftp.javaservice.net/pub/java/spec/jta-090.pdf

   Java IDL Documentation
      http://www.java.sun.com:80/products/jdk/1.2/docs/guide/idl/index.html

출처 : JavaServiceNet

:
Posted by 뽀기
2007. 3. 13. 16:14

Crossing borders: JavaScript의 특징 (한글) 그거/Tech2007. 3. 13. 16:14

프로그래밍 언어의 단점 분석 (한글)





난이도 : 초급

Bruce Tate, President, RapidRed

2007 년 2 월 27 일

JavaScript는 프로그래밍 언어의 골칫거리로 취급을 받곤 합니다. 지극히 많은 개발 툴, HTML 페이지에 대한 복잡하고 일관성 없는 문서 객체 모델, 그리고 일관성 없는 브라우저 구현 등 때문입니다.하지만 JavaScript는 우리가 생각하는 그저 그런 장난감이 아닙니다. 이 글의 저자인 Bruce Tate가 JavaScript의 기능들을 설명합니다.

웹 개발자라면 한번쯤은 JavaScript를 욕한 적이 있을 것이다. 문서 객체 모델(DOM:Document Object Model)이라고 하는 복잡한 프로그래밍 모델, 구현 및 디버깅에 사용되는 형편없는 툴, 일관성 없는 브라우저 구현 때문에 JavaScript는 더욱 욕을 먹고 있다. 최근까지도 많은 개발자들은 기껏해야 필요악으로써, 최악의 경우에는 장난감으로 취급하고 있을 정도이다.

하지만 JavaScript는 중요성이 더해가고 있고, 웹 개발에 광범위하게 사용할 수 있는 스크립팅 언어이다. 일부 업계 리더들은 JavaScript를 새로운 시각으로 바라보고 있다. Asynchronous JavaScript and XML(Ajax) 같은 프로그래밍 기술로 인해 인터랙티브(interactive) 웹 페이지 구현이 가능해졌다. Apache Cocoon 같은 완전한 웹 개발 프레임웍은 JavaScript를 사용하고 있다. ActionScript라고 하는 JavaScript 파생물은 Macromedia의 Flash의 클라이언트 측 프레임웍이다. 그리고 JVM에서 실행되는 Rhino는 JavaScript를 주 스크립팅 언어로 사용한다. (참고자료).

Ajax 전문가인 나의 동료 Stuart Halloway는 "2011년이 되면, 현대적 애플리케이션을 개발하는데 필요한 기능을 갖춘 언어로 JavaScript를 선택하게 될 것이다."라고 말하고 있다. JavaScript 프로그램들은 비슷한 자바 프로그램들 보다 10배 정도 밀도가 있고, 또 그럴 만한 언어 기능들을 증명했다.

시리즈 소개

In the Crossing borders 시리즈에서는 Bruce Tate가 자바 프로그래머들이 다른 방식과 언어를 배울 수 있는 기회를 제공한다. 모든 개발 프로젝트에 자바만이 최고의 선택이었던 시절 이후 프로그래밍 세계는 변했다. 다른 프레임웍들도 자바 프레임웍이 걸어왔던 길을 걷고 있고, 다른 언어에서 배운 개념들도 자바 프로그래밍에 접목시킬 수 있다. Python (또는 Ruby 또는 Smalltalk) 코드는 자바 코딩 방식을 변화시킬 수 있다.

이 시리즈에서는 자바 개발과는 매우 다르지만, 적용할 수 있는 프로그래밍 개념과 기술을 소개한다. 어떤 경우, 이를 활용하기 위해서는 기술을 통합해야 한다. 또는, 이 개념들을 직접 적용할 수도 있을 것이다. 툴과 같은 경우, 다른 언어들과 프레임웍들이 자바 커뮤니티의 개발자, 프레임웍, 기본 방식들에 영향을 미칠 수 있는 개념만큼은 중요하지는 않다.

이 글에서는 매우 매력적인 JavaScript의 기능을 살펴보도록 하겠다.

  • Higher-order functions. Higher-order function은 인자로서 함수들을 취하거나 함수를 리턴하는 함수이다. JavaScript 프로그래머들은 자바 언어가 할 수 없는 방식으로 함수들을 조작할 수 있다.

  • Dynamic typing. 바인딩을 지연시켜서 보다 정확하고 유연한 JavaScript로 만든다.

  • A flexible object model. JavaScript의 객체 모델은 일반적인 자바 클래스 기반의 객체 모델 대신, 프로토타입(prototypes)이라고 하는 상속에 대해 비교적 특이한 접근 방식을 사용한다.

여러분은 아마도 다른 Crossing borders 기술자료에서 설명한 동적 타이핑 모델(dynamic typing model), Higher-order function 형태의 함수 프로그래밍, 오픈 객체 모델(open object model)에 대해 알고 있을 것이다. 만일 여러분이 중요한 JavaScript 개발을 해본 적이 없다면, 이러한 기능들이 JavaScript 같은 "장난감" 언어가 아닌, Python, Lisp, Smalltalk, Haskell 같은 중요한 언어에 속한 기능이라고 생각할 것이다. 이제부터 이러한 개념들을 실제 코드 예제를 통해서 설명하도록 하겠다.

시작

JavaScript를 설치하기 위해 특별히 해야 할 일은 없다. 여러분의 브라우저에서 이 글을 읽는 것으로 충분하다. 이 글의 모든 프로그래밍 예제들은 대부분의 브라우저에서 실행된다. 나는 Firefox를 사용할 것이다.

나는 <script type='text/javascript'></script>로 래핑된 임베디드 JavaScript가 있는 간단한 웹 페이지를 로딩하겠다. Listing 1은 "Hello, World" 텍스트를 디스플레이 한다.


Listing 1. Hello, world
				
<script type='text/javascript'>
alert('Hello, World.')
</script>

코드를 실행하려면, example1.html이라는 파일을 생성하고, Listing 1을 이 파일에 복사하고, 브라우저에 파일을 로딩하면 된다. (HTML 파일 예제는 다운로드 섹션을 참조하라.) 이 코드는 페이지를 재 로딩 할 때마다 즉시 실행된다.

alert은 매개변수로서 하나의 스트링을 갖고 있는 함수 호출이다. 그림 1은 Listing 1의 코드가 "Hello, World"라는 박스를 띄우는 모습이다. 이 코드가 HTML 바디로 되어 있다면(여러분은 어떤 바디도 설정하지 않았지만, 브라우저는 엉성한 HTML라도 감수하고, 기본적으로 이 모든 페이지들을 바디로서 취급한다.) 페이지가 로딩되면 JavaScript가 실행된다.


그림 1. Hello, world
Hello, World.

실행을 지연해야 한다면, Listing 2와 같이 HTML <head> 엘리먼트에 JavaScript 함수를 선언할 수 있다.


Listing 2. 실행 지연
				
<head>
    
    <script type='text/javascript'>
        function hello() {
            alert('Hello, World.')
        }
    </script>
</head>
<body>
    <button onclick="hello();">Say Hello</button>
</body>

Listing 2의 코드를 HTML 파일에 입력하고, 브라우저에 파일을 로딩하고, Say Hello 버튼을 클릭하면, 그림 2와 같은 결과가 나온다.


그림 2. 실행 지연
Delaying execution



위로


Higher-order functions

Listing 2에서, 함수들을 조작하는 JavaScript이 기능을 보았다. 그 함수의 이름을 HTML button 태그로 전달하고, HTML의 빌트인 이벤트 모델을 활용한다. 나는 가끔씩 JavaScript를 사용하여, 변수 또는 어레이에 함수들을 저장한다. (아래 객체 모델(Object models) 섹션에서, JavaScript 객체 모델 전략이 이 기술을 사용하고 있는 것을 확인할 수 있다.) Listing 3을 보자.


Listing 3. 변수를 가진 함수 다루기
				
<head>
    
    <script type='text/javascript'>
        hot = function hot() {
            alert('Sweat.')
        }
        cold  = function cold() {
            alert('Shiver.')
        }
        
        function swap() {
            temp = hot
            hot = cold
            cold = temp    
            alert('Swapped.')
        }
    </script>
</head>
<body>
    <button onclick="hot();">Hot</button>
    <button onclick="cold();">Cold</button>
    <button onclick="swap();">Swap</button>
</body>

함수들은 JavaScript의 퍼스트 클래스(first-class) 객체들이고, 이들을 자유롭게 조작할 수 있다. 우선, 나는 두 개의 함수를 선언한다. hotcold이다. 각각을 하나의 변수에 저장한다. Hot과 Cold 버튼을 클릭하면 해당 기능이 호출되며 경고를 발생한다. 그런 다음, 나는 Hot과 Cold 버튼의 값을 바꾸는 함수를 선언하면서, 이 함수를 제 3의 버튼에 붙이면 그림 3과 같은 경고가 디스플레이 된다.


그림 3. 함수 다루기

이 예제는, 다른 변수들을 다루듯이 함수를 취급할 수 있다는 것을 보여준다. C 개발자들은 이 개념을 함수 포인터(function pointer)로 생각하겠지만, JavaScript의 higher-order function의 개념은 훨씬 더 강력한 것이다. 이 기능을 사용하면 JavaScript 프로그래머는 액션 또는 함수들을 다른 변수 유형을 다루는 것만큼 쉽게 다룰 수 있다.

함수를 함수 인자로서 사용하거나, 함수를 값으로서 리턴하면, 추상화를 Higher-order function의 경지로 끌어올릴 수 있다. Listing 4는 Listing 3을 약간 수정한 것으로, 함수를 리턴하는 higher-order function을 보여준다.


Listing 4. Higher-order functions
				
<head>

    <script type='text/javascript'>

        function temperature() {
            return current
        }

        hot = function hot() {
            alert('Hot.')
        }

        cold  = function cold() {
            alert('Cold.')
        }

        current = hot

        function swap() {
            if(current == hot) {
              current = cold
            } else {
              current = hot
            }
        }
    </script>
</head>
<body>
    <button onclick="funct = temperature()();">Temperature</button>
    <button onclick="swap();">Swap</button>
</body>

이 예제는 일반적인 문제를 해결한다. 변하는 작동을 사용자 인터페이스 이벤트에 어떻게 붙일 것인가? Higher-order function을 사용하면 쉽게 해결할 수 있다. temperature higher-order function은 current 값을 리턴하는데, 여기에는 hot 또는 cold 함수가 있다. 이상하게 보이는 함수 호출을 살펴보도록 하자. 바로 temperature()()이다. 첫 번째 괄호는 temperature 함수를 호출한다. 두 번째 괄호는 temperature에 의해 리턴 되는 함수를 호출한다. 그림 4는 결과이다.


그림 4. Higher-order function
higher-order functions

Higher-order function은 함수 프로그래밍의 기초가 되고, 거의 모든 사람들은 순수한 객체 지향 프로그래밍보다 고급 추상화를 나타낸다고 믿고 있다. 하지만, JavaScript의 장점은 Higher-order function이 전부가 아니다. JavaScript의 동적 타이핑(Dynamic Typing)도 UI 개발에 알맞다.




위로


동적 타이핑(Dynamic typing)

정적 타이핑(static typing)의 경우, 컴파일러는 인자, 변수 또는 주어진 연산에 사용된 값에 대한 리턴 값들을 검사할 수 있다. 장점은 컴파일러가 추가 에러 체크도 할 수 있다는 것이다. 또한 정적 타이핑은 IDE 같은 툴에 대한 추가 정보도 제공하고, 더욱 나은 코드 완성 같은 특징을 갖고 있다. 하지만 정적 타이핑은 여러 가지 단점들도 있다.

  • 의도(intention)를 일찍 선언해야 하기 때문에, 유연성이 떨어진다. 예를 들어, 자바 클래스를 변경하면 클래스의 유형을 변경해야 하고, 재 컴파일이 필요하다. 반대로, Ruby는 오픈 클래스를 허용하지만, 클래스를 변경하면 그 클래스의 유형도 변경해야 한다.

  • 같은 생각을 표현하는 데에도 더 많은 코드를 입력해야 할 경우가 종종 있다. 예를 들어, 인자들을 가진 유형 정보를 포함시키고, 함수를 가진 값들을 리턴해야 하고, 모든 변수들을 타이핑 해야 한다. 또한, 모든 변수들을 선언해야 하고, 유형들 간 변환도 필요하다.

  • 컴파일-전개 사이클은 비슷한 동적 언어의 개발 사이클 보다 길다. 가끔은 툴을 사용하여 이러한 문제들을 완화시킬 수 있지만 말이다.

정정적 타이핑은 미들웨어나 OS를 구현하는 언어에 알맞다. UI 개발은 더 나은 생산성과 유연성을 요하기 때문에 동적 타이핑이 선호된다. 물론, 위험 부담도 있다. JavaScript로 작업하는 웹 개발자들은 잘못된(mistyped) 변수 이름 때문에 골머리를 앓을 때가 있다. 하지만 장점도 만만치 않다. 두 가지 예제를 설명하겠다.

먼저, 하나의 객체를 생각해 보자. Listing 5에서, 나는 새로운 객체를 만들고, color라고 하는 존재하지 않는 애트리뷰트에 액세스 한다.


Listing 5. 애트리뷰트 가져오기
				
<script type='text/javascript'>
    blank_object = new Object();
    blank_object.color = 'blue'
    alert('The color is ' + blank_object.color)
</script>

애플리케이션을 로딩하고 실행하면 그림 5와 같은 결과가 생긴다.


그림 5. 애트리뷰트 가져오기
Introducing attributes

JavaScript는 blue 애트리뷰트가 존재하지 않는다고 해서 불평하지 않는다. 정적 언어 옹호론자들이라면 무수한 에러 잠재력을 가진 이 예제를 보고 놀랄 것이다. 위험해 보일지라도, 그 매력을 거부할 수 없다. 우리는 애트리뷰트를 가져올 수 있다. 이 예제를 이 글의 이전 예제들과 결합하여 작동을 가져올 수도 있다. 변수에는 함수가 포함될 수 있다는 것을 기억하라. Dynamic Typing과 Higher-order function에 기반하여 임의의 작동을 언제라도 클래스로 가져올 수 있다.

Listing 5를 Listing 6과 같이 쉽게 다시 작성할 수 있다.


Listing 6. 작동 가져오기
				
<script type='text/javascript'>
    blank_object = new Object();
    blank_object.color = function() { return 'blue'}
    alert('The color is ' + blank_object.color())
</script>

얼마나 빨리 JavaScript로 개념들을 변경할 수 있는지를 보고 있다. 데이터에 대한 작동을 가져오는 것과 마찬가지로, 의미를 변경하고 신택스를 아주 조금 변경하면 된다. 이것은 강점도 되고 가장 큰 약점도 된다. 사실, 이 언어 자체의 객체 모델은 JavaScript의 강점을 보여주는 하나의 예에 불과하다.




위로


객체 모델(Object model)

지금쯤이면, 여러분은 JavaScript가 장난감 그 이상이라는 것을 깨달아가고 있어야 한다. 사실 많은 사람들이 이 객체 모델을 사용하여 복잡하고, 잘 디자인 된, 객체 지향 소프트웨어를 만들고 있다. 하지만, 이러한 객체 모델은(특히 상속의 경우) 익숙한 것이 아니다.

자바 언어는 클래스 기반이다. 애플리케이션들을 구현할 때, 모든 객체들을 위한 템플릿으로서 새로운 클래스를 구현한다. 그런 다음, new를 호출하여 새로운 템플릿을 인스턴스로 만들면서 새로운 객체를 만든다. JavaScript에서, 여러분은 프로토타입을 만드는데, 이것은 모든 미래의 객체들을 만드는 인스턴스이다.

추상적인 설명은 집어치우고, 실제 코드를 보여주도록 하겠다. Listing 7은 name이라는 애트리뷰트와 speak 액션을 가진 Animal을 만든다. 그리고 나서 다른 동물들은 이 베이스(base)에서 상속을 받을 것이다.


Listing 7. 컨스트럭터 만들기
				
<script type='text/javascript'>        
function Animal() {
    this.name = "nobody"
    this.speak = function () {
        return "Who am I?"
    }
}

myAnimal = new Animal();
alert('The animal named ' + myAnimal.name + 
      ' says ' + myAnimal.speak());

</script>

Listing 7은 그림 6과 같은 모습으로 나타난다.


그림 6. 컨스트럭터 구현하기
constructor

자바 개발자에게 Listing 7의 코드는 예측이 불가능하고 이상해 보인다. 고유의 객체들을 구현하지 않는 대부분의 JavaScript 개발자들에게도 이 코드는 예측 불가능하고 이상해 보인다. 나는 이 그림에 어느 정도의 정상성을 부여해야 할 것이다.

세 조각의 정보가 필요하다. 첫 번째로, JavaScript는 객체들을 중첩된 함수들로서 나타낸다. Listing 7의 Animal의 정의가 유효한 신택스라는 것을 의미한다. 두 번째, JavaScript는 객체 구조를 프로토타입 또는 기존의 객체 인스턴스에 의존한다. 클래스 템플릿이 아닌. funct()는 호출이지만, new Animal()Animal의 프로토타입에 기반하여 객체를 구현한다. 마지막으로, JavaScript에서 객체들은 단순히 함수와 변수들의 컬렉션이다. 어떤 유형도 객체와 제휴되지 않으므로 이 구조를 자유롭게 수정할 수 있다.

Listing 7로 다시 돌아가보자. JavaScript가 Animal에서 지정한 프로토타입에 기반하여 새로운 객체인 myAnimal을 정의하는 것을 볼 수 있다. 여러분의 프로토타입에서 애트리뷰트와 함수들을 사용하거나, 함수나 애트리뷰트를 재정의 할 수 있다. 이러한 작동에 익숙하지 않은 자바 개발자들에게 이러한 유연성은 해가 될 수 있지만, 매우 강력한 모델이다.

prototype이라는 인스턴스 변수를 사용하여 당신이 사용한 객체의 껍데기(basis)를 선언 할 수 있다. prototype 인스턴스 변수를 설정하여 상속 체인에서 부모를 가리키도록 할 것이다. prototype을 설정하면, 여러분이 생성한 객체들은 애트리뷰트와 함수들을 상속할 것이다. 이러한 방식으로, 상속의 객체 지향 개념을 시뮬레이트 할 수 있다. Listing 8을 보자.


Listing 8. 상속
				
<script type='text/javascript'>        

Animal = function() {
    this.name = "nobody"
    this.speak = function () {
        return "Who am I?"
    }
}
Dog = function() {
  this.speak = function() {
    return "Woof!"
  }
}
Dog.prototype = new Animal();

myAnimal = new Dog();
alert('The animal named ' + myAnimal.name + 
      ' says ' + myAnimal.speak());
      </script>

Listing 8에서, Dog 프로토타입을 만든다. 그 프로토타입은 Animal에 기반하고 있다. Dogspeak() 메소드를 재 정의하지만, name() 메소드에 대해서는 어떤 것도 수행하지 않는다. 그런 다음, Dog의 프로토타입을 Animal로 설정한다. 그림 7은 결과 모습이다.


그림 7. 프로토타입과 상속
inheritance

다음은 JavaScript가 애트리뷰트나 메소드에 대한 레퍼런스를 해결하는 방법이다.

  • JavaScript는 생성자(Constructor)에 정의된, 원래 프로토타입에 기반하여 인스턴스를 만든다. 여러분이 참조하는 메소드나 애트리뷰트는 여러분이 생성한 원래 카피를 사용할 것이다.

  • 객체에서 변수들을 다른 변수들처럼 재정의 할 수 있다. 이렇게 해서, 객체를 변경할 수 있다. 따라서, 여러분이 분명하게 정의한 애트리뷰트나 함수들은 원래 프로토타입에서 정의된 것들 보다 우위를 차지한다.

  • prototype이라고 하는 인스턴스 변수를 설정하면, JavaScript는 정의되지 않은 인스턴스 변수나 애트리뷰트에 대한 인스턴스를 보게 될 것이다. 검색은 반복적이다.prototype에서 정의된 인스턴스가 애트리뷰트나 함수를 찾을 수 없다면 이것의 프로토타입을 볼 것이다.

그렇다면 JavaScript의 상속 모델은 무엇인가? 이 모든 것은 여러분이 정의하는 방식에 의존한다. 여러분은 상속 작동을 정의하여 이를 오버라이드 할 수 있다. JavaScript는 객체 지향 언어보다는 함수적이고, 똑똑한 신택스와 의미를 사용하여 매우 복잡한 작동들을 시뮬레이트 한다. 객체 모델은 매우 유연하고 개방적이며 매우 강력하다. 누군가는 이것이 너무 유연하다고 말한다. 각자의 작업에 알맞은 올바른 툴을 사용해야 한다.




위로


Wrap up

JavaScript 객체 모델은 Dojo 같은 거대한 라이브러리를 지원하는 언어 기능들을 기반으로 구현된다. (참고자료) 이러한 유연성 덕택에 각 프레임웍이 섬세한 방식으로 객체 모델을 변경할 수 있다. 어떤 면에서는 이러한 유형의 유연성이 큰 약점이다. 상호운용성 문제를 떠올리면 쉽게 이해된다. (언어의 유연성으로 이러한 문제가 완화되기도 하지만 말이다.)

또 다른 관점에서 볼 때, 유연성은 대단한 축복이 될 수도 있다. 자바 언어는 복잡함이 문제였다. 기본 객체 모델은 확장될 수 있을 정도로 유연하지 않다. Aspect 지향 프로그래밍, Spring 프로그래밍 프레임웍, 바이트코드 강화 라이브러리 같은 대단한 오픈 소스 프로젝트는 엔터프라이즈 개발자들이 자바 언어를 성공적으로 다루기 위해서 배워야 할 것들이다.

JavaScript의 유연성 덕택에 higher-order 언어의 힘을 맛볼 수 있었다. 모든 프로젝트에서 이를 선택할 수 없다. 정확한 정보에 기반하여 언어의 강점과 단점을 파악하면 더욱 잘 핸들할 수 있다. JavaScript 웹 위젯을 다룰 수 있을 정도라면 언어의 힘을 충분히 활용하는 것이다. 다음 시간을 기대해주기 바란다.

기사의 원문보기





위로


다운로드 하십시오

설명 이름 크기 다운로드 방식
Sample HTML files for this article j-cb12196.zip 3KB HTTP
다운로드 방식에 대한 정보


참고자료

교육

토론


필자소개

Bruce Tate

Bruce Tate는 베스트 셀러인 Jolt winner Better, Faster, Lighter Java의 저자이다. 최근에는 From Java to RubyRails: Up and Running를 출간했다. IBM에서 13년 동안 근무했으며, 자바와 Ruby on Rails에 기반한 경량 개발 전략과 아키텍트를 전문적으로 다루는 RapidRed를 창립했다. 현재 그는 WellGood LLC의 CTO이다.


출처 : IBM developerWorks

'그거 > Tech' 카테고리의 다른 글

DOM(Document Object Model)  (0) 2007.03.20
패턴과 프레임워크  (0) 2007.03.14
JSON vs. XML  (0) 2007.03.13
JSON (JavaScript Object Notation) 개요  (0) 2007.03.13
소프트웨어 개발 방법론 충격진화「MDA」  (0) 2007.03.05
:
Posted by 뽀기
2007. 3. 13. 15:22

JSON vs. XML 그거/Tech2007. 3. 13. 15:22

Ajax를 이용한 웹 애플리케이션이 늘어나면서부터 서버와의 통신에 있어 어떠한 데이터 교환 형식을 이용할지에 대한 관심이 부쩍 늘어난 것 같다. 이러한 관심의 예로 JSON(Javascript Object Notation)을 기존 XML을 대체할 데이터 교환 규약으로 보고 있는 사람들이 많으며 26일 InfoQ에서도 이러한 JSON과 XML에 대한 논쟁?을 기사로 올렸다. XML은 지난 몇년에 걸쳐 업계의 노력으로 이제는 완전히 시장이 포화상태에 있다. 반면 JSON은 그동안 사람들에게 많이는 알려지지 않았지만, 그 고유의 단순함 내지는 경량모델로 최근에 개발자들 사이에서 인기를 얻고있는 데이터 교환이다. 뭐 사실 구글에서 조금만 검색해도 JSON에 대한 이런 이야기는 많이 찾아볼 수 있다. -_-;

JSON은 Head Rush Ajax라는 책에서 처음 접했는데, 처음보는 것이고 또 아직은 XML이 더 눈에 익숙하고 더 연마해야 하는지라 자세히 살펴보지는 않았는데, JSON에 관한 ppt를 보니 사람들이 왜 JSON에 열광하는지 조금 이해가 갔다. 사실 XML은 그 규약 자체도 상당히 엄격한(XML의 well-formedness를 준수하는 데만도 10가지 규칙을 내세운다.)데다 네임스페이스나 그런것도 복잡하고...

아무튼 JSON을 지지하는 측에서 JSON이 데이터 교환에 있어서의 장점은 이렇단다. (대충 직역)
1. 사람과 동시에 기계도 읽을 수 있는 형식이다.
2. 유니코드를 지원하는데, 따라서 거의 모든 인간 언어가 소통 가능하다.
3. 자기문서화적이다(? 원문 - self-documenting). 즉, 특정 값 뿐만 아니라 구조와 필드명까지 서술한다.
4. 엄격한 문법과 필수적인 알고리즘을 허용하는 파싱 요구조건이 단순하며, 효율적이고, 일관성있다.
5. 대부분의 일반 컴퓨터 과학에서 제시하는 자료구조인 레코드, 리스트, 트리를 표현할 수 있는 능력을 지녔다.

반면에 JSON 사용에 대해 반박하는 일반적인 논쟁거리는 다음과 같다.
1. JSON은 네임스페이스를 갖지 않는다. 따라서 모든 객체 자체가 네임스페이스이며 키 집합은 다른 객체에 독립적이며 심지어 배타적으로 중첩된다. 또한 JSON은 애매모호함을 피하기 위해 다른 언어에서 하는 것과 같이 컨텍스트를 이용한다.
2. JSON은 검증기가 없다. 궁극적으로 모든 애플리케이션이 입력에 대한 검증을 책임져야 한다. 이는 위임될 수 없으나 YAML 검증기는 사용할 수 있다.
3. JSON은 확장될 수 없다. 사실 그럴필요가 없다. JSON은 유연하며 어떠한 비순환 데이터 구조를 표현할 수 있다. 새로운 필드가 기존 구조를 폐기시키지 않고도 추가될 수 있다.
4. JSON은 XML이 아니다. Douglas는 JSON이 XML에 비해 훨씬 더 심플하다고 주장한다.

읽고나서 보니 Douglas 이 사람 JSON을 발명한 아키텍트네.. -_-;;
아무튼 JSON이 XML에 비해 데이터 교환에 있어서는 상당히 간단한 것임에는 틀림없고, 어디에 어떻게 사용되는 지는 그 상황이 어떠한 상황이냐에 달린 것 같다.(어중간한 으로 마무리.. -_-v)

그나저나 JSON은 뭐라고 발음한다냐? '제이손'? '제이에스-온'? '제이에스오엔'?, '자바스크립트 오브젝트 노테이션~ 헉헉'?

출처 : decoder.egloos.com

:
Posted by 뽀기
2007. 3. 13. 15:19

JSON (JavaScript Object Notation) 개요 그거/Tech2007. 3. 13. 15:19


JSON
(JavaScript Object Notation)은 경량의 DATA-교환 형식이다. 이 형식은 사람이 읽고 쓰기에 용이하며, 기계가 분석하고 생성함에도 용이하다. JavaScript Programming Language, Standard ECMA-262 3rd Edition - December 1999의 일부에 토대를 두고 있다. JSON은 완벽하게 언어로 부터 독립적이지만 C-family 언어 - C, C++, C#, Java, JavaScript, Perl, Python 그외 다수 - 의 프로그래머들에게 친숙한 관습을 사용하는 텍스트 형식이다. 이러한 속성들이 JSON을 이상적인 DATA-교환 언어로 만들고 있다.

JSON은 두개의 구조를 기본으로 두고 있다:

  • name/value 형태의 쌍으로 collection 타입. 다양한 언어들에서, 이는 object, record, struct(구조체), dictionary, hash table, 키가 있는 list, 또는 연상배열로서 실현 되었다.
  • 값들의 순서화된 리스트. 대부분의 언어들에서, 이는 array, vector, list, 또는 sequence로서 실현 되었다.

이러한 것들은 보편적인 DATA 구조이다. 사실상 모든 현대의 프로그래밍 언어들은 어떠한 형태로든 이것들을 지원한다. 프로그래밍 언어들을 이용하여 호환성 있는 DATA 형식이 이러한 구조들을 근간에 두고 있는 것은 당연하다.

JSON 에서, 이러한 형식들을 가져간다:

object는 name/value 쌍들의 비순서화된 SET이다. object는 { (좌 중괄호)로 시작하고 } (우 중괄호)로 끝내어 표현한다. 각 name 뒤에 : (colon)을 붙이고 , (comma)로 name/value 쌍들 간을 구분한다.

array은 값들의 순서화된 collection 이다. array는 [ (left bracket)로 시작해서 ] (right bracket)로 끝내어 표현한다. , (comma)로 array의 값들을 구분한다.

value는 큰따옴표안에 string, number ,true ,false , null, object ,array이 올수 있다. 이러한 구조들을 포함한다.

string은 큰따옴표안에 둘러 싸인 zero 이상 Unicode 문자들의 조합이며, 쌍다옴표안에 감싸지며,backslash escape가 적용된다. 하나의 문자(character)도 하나의 문자열(character string)로서 표현된다. string은 C 또는 Java 문자열 처럼 매우 많이 비슷하다.

number는 8진수와 16진수 형식을 사용하지 않는것을 제외하면 C와 Java number 처럼 매우 많이 비슷하다.

토근들의 어떤 쌍 사이에 공백을 삽입할수 있다. 드물게 encode된 세부 항목을 제외하면, 이렇게 설명된 JSON의 형식은 완벽하게 그 언어를 설명한다.






Why JSON isn't just for JavaScriptI can't believe it's not XML!The JSON vs XML debate begins in earnestJSON and XMLWhy JSON vs XML is a yawnJSON vs. XML as a data interchange formatExamining JSON

json@JSON.org


출처 : www.json.org
:
Posted by 뽀기
2007. 3. 13. 15:04

객체 직렬화(Object Serialization) 그거/Java2007. 3. 13. 15:04

객체 직렬화(Object Serialization):

송 재승(angel1011@hanmail.net)
 
자바에서는 자바에서 제공하는 기본적인 데이터 유형 이외에도 여러 객체들을 스트림에 쓰고, 읽을 수 있는 기능을 제공하는데 이것이 바로 객체 직렬화를 통해서 가능하다.
이러한 객체 직렬화 기법은 원격 메소드 호출, 보안 및 암호화 등 실제 프로그래밍 시에 유용하게 사용되어 질 수 있다.

 객체 직렬화의 비밀

그러면 먼저 객체 직렬화가 어떠한 과정을 거쳐서 이루어 지는지 비밀을 벗겨보도록 하자.
먼저 객체들은 ObjectOutputStream에 의해서 직렬화가 되며, ObjectInputStream에 의해서 직렬화가 해제된다는 사실을 기억하도록 하자. 이러한 일련의 과정을 그림으로 나타내면 다음과 같다.
[그림 객체 직렬화]

사용자 삽입 이미지
여기에 사용되는 ObjectOutputStream과 ObjectInputStream은 모두 java.io 패키지의 일부분이고, 각각 DataOutput과 DataInput클래스를 implement한 ObjectOutput과 ObjectInput을 extends 해서 만들어 진것이다. 그리고 이 두 클래스에는 비원시 객체와 배열등을 스트림에 읽고 쓰기 위한 메쏘드들이 추가되어져 있다.
[그림 클래스 구조도]

객체 직렬화의 과정

지금부터는 보다 자세히 객체의 직렬화 과정에 대해 알아보도록 하자.
객체는 ObjectOutputStream의 writeObject() 메쏘드에 자신을 넘김으로써 직렬화 된다.
WirteObject()메쏘드는 private 필드와 super 클래스로부터 상속받은 필드를 포함, 객체의 모든 것을 기록하게된다.
직렬화 해제는 직렬화와 반대의 과정을 거치게 되는데 ObjectInputStream의 readObject()메쏘드를 호출함으로써 스트림으로부터 읽어 들이고 이를 직렬화가 되기전의 객체로 다시 만들게 된다.
다음은 객체가 직렬화 되고 해제되어 원래의 객체로 복원되는 과정에 대한 간단한 예제이다.
눈여겨 볼 부분은 ObjectInputStream을 생성해서 writeObject()를 사용해서 객체를 직렬화 하는것과 ObjectInputStream을 생성해서 readObject()를 통해서 객체를 복원하는 부분이다. 또한 SerializableClass가 Serializable을 implements한 것을 주의해서 보길 바란다.
 
import java.io.*;
public class ObjectSerializeTest {
    public static void main(String[] args) throws Exception {
/* 파일을 열어서 그곳에 객체를 직렬화 시켜서 저장한다. */
// 파일 출력 스트림을 연다.

    FileOutputStream fos = new FileOutputStream("_imsi.txt");
// 객체 스트림을 열고, 객체스트림을 통해 객체를 파일에 저장
    ObjectOutputStream oos = new ObjectOutputStream(fos);
    oos.writeObject(new SerializableClass("Serialize Test Program", 1004));
// 스트림을 닫는다.
    oos.close();
/* 직렬화 된 객체가 저장된 파일로 부터 객체를 해제시켜 원래의 객체를 복원*/
// 파일 입력 스트림으로부터 객체 입력 스트림을 연다.

    FileInputStream fis = new FileInputStream("_imsi.txt");
    ObjectInputStream ois = new ObjectInputStream(fis);
// 객체 입력 스트림으로부터 객체를 읽어온다.
    SerializableClass sc = (SerializableClass)ois.readObject();
// 스트림을 닫는다.
    ois.close();
/* 스트림으로부터 읽어들인 객체의 내용을 출력 원래 생성되었던 객체와 같은 값을 갖는다는 것을 알수가 있다. */
    System.out.println("String : " + sc.mString);
    System.out.println("Integer : " + sc.mInt);
   }
}
/* 하나의 문자열과 정수를 저장하고있는 클래스
Serializable을 implements 함으로써 스트림을 통해 직렬화되고 해제되어질수 있다. */
class SerializableClass implements Serializable {
    public String mString;
    public int mInt;
// 생성자
    public SerializableClass(String s, int i) {
        this.mString = s;
        this.mInt = I;
    }
}
  • 실횅방법
    javac.exe ObjectSerializeTest.java
    java.exe ObjectSerializeTest
  • 실행결과
String : Serialize Test Program
Integer : 1004
 
 

커스텀 직렬화

  • Serializable
스트림을 통해서 직렬화 또는 해제 될수 있는 객체는 Serialiazable을 implements한 객체만이 가능하다.자바 2에서는 java.awt, javax.swing, etc등의 클래스들이 Serializable을 implements하고 있으며, 이러한 상위 클래스들이 implements하므로 하위 클래스들 또한 같이 직렬화/해제의 범주에 들 수가 있다.
다음은 Serializable을 implements하고있어서 객체의 직렬화가 가능한 객체들의 리스트이다.
[직렬화 가능 객체 리스트]
 
패키지
Serializable
java.awt
BorderLayout, CardLayout, CheckboxGroup, Color, Component, ComponentOrientation, Cursor, Dimension, Event, FlowLayout, FocusManager, Font, FontMetrics, GraphicsConfigTemplate, GridBagConstraints, GridBagLayout, GridBagLayoutInfo, GridLayout, ImageMediaEntry, Insets, LightweightDispatcher, MediaTracker, MenuComponent, MenuShortcut, Point, Polygon, Rectangle, ScrollPaneAdjustable, SystemColor
java.awt.dnd
DropTarget
java.awt.font
TransformAttribute
java.awt.geom
AffineTransform
java.awt.image.renderable
ParameterBlock
java.beans
PropertyChangesSupport, VetoableChangeSupport, BeanContext, BeanContextChildSupport, BeanContextSupport
java.io
ExternalizableFile, FilePermission, FilePermissionCollection, ObjectStreamClass
java.net
InetAddress, SocketPermission, SocketPermissionCollection, URL
java.rmi
MarshalledObject
java.rmi.activation
ActivationDesc, ActivationGroupDesc, ActivationGroupID, ActivationID
java.rmi.dgc
Lease, VMID
java.rmi.server
ObjID, RemoteObject, UID
java.security
AllPermissionCollection, BasicPermission, BasicPermissionCollection, CodeSource, GuardedObject, Identity, Key, KeyPair, Permission, PermissionCollection, Permissions, PermissionsHash, SecureRandomSpi, SignedObject, UnresolvedPermission, UnresolvedPermissionCollection
java.text
BreakIterator, Collector, DateFormatSymbols, DecimalFormatSymbols, Format, SpecialMapping, TextBoundaryData, UnicodeClassMapping, WordBreakTable
java.util
ArrayList, BitSet, Calendar, Date, EventObject, HashMap, HashSet, Hashtable, LinkedList, Locale, PropertyPermissionCollection, Random, TimeZone, TreeMap, TreeSet, Vector
javas.swing.table
AbstractTableModel, DefaultTableCellRenderer, DefaultTableColumnModel, DefaultTableModel, TableColumn
javax.swing.text
AbstractDocument, EditorKit, GapContext, SimpleAttributeSet, StringContent, StyleContext, TabSet, TabStop
javax.swing.tree
DefaultMutableTreeNode, DefaultTreeModel, DefaultTreeSelectionModel, TreePath
 
Serializable은 Cloneable과 같은 마커 인터페이스로서, 어떠한 메쏘드들을 정의해놓은 것이 아닌, serizlVersionUDI라는 변수 하나만을 가지며, 이 객체가 직렬화가 가능하다는 것을 알려주는 역할만을 하는 인터페이스일 뿐이다.
  • transient
스트림을 이용해서 직렬화 하는데 있어서, 커다란 프로그램 전체가 직렬화된다면, 여러가지 면에서 많은 낭비일 것이다. 예를 들어, 한 객체가 마우스가 눌려진 위치를 알기 위해서 마우스 클릭시에 위치를 저장하는 필드를 가지고 있다고 가정하자, 이 경우 마우스의 위치값은 프로그램이 돌아가는 상태에서 마우스가 눌려졌을 당시에만 유효한 값으로, 객체가 직렬화 되었다가 해제 되었을 경우에는 쓸모없는 값이 되어버린다.
이런 객체 직렬화에 쓸모없는 값들은 transient로 선언해 줌으로써 객체 직렬화에서 제외되어질수 있다.
Private transient int x;
이러한 선언은 플랫폼에 따라 다른 값을 가지는 필드나, 현재의 환경에서만 유효한 필드등을 객체 직렬화에서 제외하는데 유용하게 쓰일 수가 있다.
transient선언이 적용된 예이다.
 
import java.io.*;
public class ObjectSerializeTest1 extends TestClass implements Serializable {
// 변수 선언, x는 객체직렬화에서 제외되도록 transient로 선언
    private int i;
    private transient int x;
// 생성자 Serializable을 implements한 i와 x는 받아온 인자를 그대로 대입하고, TestClass를 extends한 j는
// i값에 5배를 한 값을 대입했다.

    public ObjectSerializeTest1(int i, int x) {
        this.i = i;
        this.x = x;
        j = i*5;
    }
/* 객체직렬화를 하기전의 객체의 값들을 알아본뒤에, 해당 객체를 스트림에 직렬화 시킨다.*/
    private void writeObject(ObjectOutputStream out) throws IOException {
        System.out.println("writeObject");
        System.out.println("write ==> i = "+this.i+", j = "+j+", x = "+this.x);
        out.defaultWriteObject();
// 객체를 직렬화 한다.
        out.writeInt(j);
// 임시파일에 기록
    }
/* 스트림으로 부터 객체를 해제시킨다. */
    private void readObject(ObjectInputStream in) throws IOException, ClassNotFoundException {
        System.out.println("readObject");
        in.defaultReadObject();
// 객체 직렬화 해제 작업
        j=in.readInt();
// 임시파일에서 읽어옴
    }
/* 객체의 값을 알기 위해서 오버라이드 */
    public String toString() {
        return "i = "+i+", j = "+j+", x = "+x;
    }
    public static void main(String[] args) throws Exception {
// 객체직렬화에 사용될 화일을 열고 스트림을 얻는다.
        FileOutputStream fos = new FileOutputStream("temp.txt");
        ObjectOutputStream oos = new ObjectOutputStream(fos);
// 객체를 생성시킨뒤, 스트림에 객체를 쓴다.(10은 transient로 선언된 필드의 값임을 유의)
        oos.writeObject(new ObjectSerializeTest1(1, 10));
        oos.close();
// 정보를 저장한 화일을 열고 스트림을 통해 읽은뒤 객체직렬화를 해제시킨다.
        FileInputStream fis = new FileInputStream("temp.txt");
        ObjectInputStream ois = new ObjectInputStream(fis);
        ObjectSerializeTest1 ser = (ObjectSerializeTest1)ois.readObject();
        ois.close();
// print문 수행시 형 변환이 이루어져서 이미 오버라이드된 toString()메쏘드가 수행된다.
        System.out.println("read ==> "+ser);
    }
}
/* Serialize되지 않은 클래스 객체 직렬화 작업으로부터는 제외, 만약 사용자가 이 클래스를 객체 직렬화에 사용하고 싶다면 Serializable을 implements하면 된다. 만약 그렇게 할수 없을 경우에는 위와 같이 하도록 한다. */
class TestClass {
    int j;
}
  • 실횅방법
    javac.exe ObjectSerializeTest1.java
    java.exe ObjectSerializeTest1
  • 실행결과
writeObject
write ==> i = 1, j = 5, x = 10
readObject
read ==> i = 1, j = 5, x = 0
 
  • externalizable
객체직렬화의 또 다른 방법으로는 Externalizable 인터페이스를 들수 있는데, 이 방법은 Serializable보다 더 강력하게 객체 직렬화를 제어하고 싶을 경우에 사용되어진다. 어떤 클래스가 이 인터페이스를 구현하게 되면 readExternal()과 writeExternal()메쏘드를 사용하여 객체를 스트림으로부터 읽고 쓸수있다. Externalizable객체들은 자신의 상태를 직렬화 및 해제하는 과정에서 강력한 제어를 할수 있으며, Serializable 객체에서의 기본 데이터 포멧을 사용하고싶지 않을때에 사용한다.
다음은 externalizable을 사용한 객체 직렬화의 예이다.
 
import java.io.*;
public class ObjectExternalizeTest
   
 implements Externalizable {
    int i;
    String s;
/* 인자없는 생성자(직렬화된 데이터를 읽어들여서 객체를 만들기 위해 필요) */
    public ObjectExternalizeTest() { }
/* 생성자 */
    public ObjectExternalizeTest(String s, int i) {
        this.s = s;
        this.i = i;
    }
/* readObject()메쏘드로부터 호출이 되며, 직렬화된 객체를 해제시킨다. */
    public void
readExternal(ObjectInput oi) throws IOException, ClassNotFoundException {
        System.out.println("Execute readExternal()");
        s = (String)oi.readObject();
        i = oi.readInt();
    }
/* writeObject()메쏘드로부터 호출이 되며, 변수들을 직렬화한다.*/
    public void
writeExternal(ObjectOutput oo) throws IOException {
        System.out.println("Execute writeExternal()");
        oo.writeObject(s);
        oo.writeInt(i);
    }
/* 변수들의 값을 보여준다.*/
    public String toString() {
        return Integer.toString(i)+", "+s;
    }
    public static void main(String[] args) throws Exception {
        ObjectExternalizeTest oet1 = new ObjectExternalizeTest("Externalizable Test", 1000);

// 객체를 직렬화시켜 저장할 화일로부터 쓰기 스트림을 얻어온다.
        FileOutputStream out = new FileOutputStream("temp.txt");
        ObjectOutputStream oos = new ObjectOutputStream(out);

/* 객체를 직렬화 시킨다. 이경우 writeObject는 oet1의 writeExternal을호출하게되고writeExternal() 메쏘드의 코드들이 확장되어 실행이 된다. */
       oos.writeObject(oet1);
// 객체가 직렬화되어 저장된 파일로부터 스트림을 얻어온다.
        FileInputStream in = new FileInputStream("temp.txt");
        ObjectInputStream ois = new ObjectInputStream(in);
        ObjectExternalizeTest oet2 = (ObjectExternalizeTest)ois.readObject();
// 객체 직렬화 이전의 객체와 직렬화를 거친 뒤 다시 해제된 객체의 값을 출력한다.
        System.out.println(oet1);
        System.out.println(oet2);
// 만약 모든 과정이 제대로 이루어졌다면 oet1과 oet2객체는 같은 값을 가지고있을 것이다.
    }
}
  • 실횅방법
    javac.exe ObjectExternalizeTest.java
    java.exe ObjectExternalizeTest
  • 실행결과
Execute writeExternal()
Execute readExternal()
1000, Externalizable Test
1000, Externalizable Test
 
 

애플릿 직렬화

객체직렬화는 애플릿을 직렬화하는데에도 사용이 되어진다. 이러한 경우 <Applet>태그는 클래스 파일 대신에 직렬화 된 객체 파일을 지정하게 되고, 브라우저는 이러한 태그를 만나면 직렬화된 애플릿을 해제하여 화면에 보여주게 된다. 즉, 그래픽 유저 인터페이스를 생성하기 위한 코드를 전혀 가지고 있지 않으면서도 완벽한 그래픽 유저 인터페이스를 가지게 되는것이다.
일반적으로 직렬화된 애플릿은 *.ser의 확장자를 가진다.
직렬화된 애플릿을 호출하는 html 파일은 다음과 같은 내용을 가진다.
<Applet Object="SerializableApplet.ser" Width=300 Height=300></Applet>

소켓을 통한 객체 직렬화

소켓을 생성한 뒤 스트림을 얻어서 ObjectInputStream과 ObjectOutputStream을 얻어서 사용한다면, 파일이 아닌 네트웍 너머의 다른 컴퓨터와도 객체를 직접 주고 받을수 있다.
소켓을 통해 직렬화 되어 쓰고 읽혀질 객체는 역시 Serializable을 구현하고 있어야 하며, 사용법은 파일에 쓰고 읽는것과 동일하다.
다음은 서버에서 이미지를 포함하고 있는 Serializable을 구현한 객체를 생성하여 소켓을 이용하여 클라이언트 애플릿에게 직렬화된 객체를 보내고, 애플릿에서는 이를 해제하여 화면에 나타내는 예제이다.
여기서 주의깊게 살펴보아야 할 점은 이미지를 직렬화하기위한 방법이다.
이미지는 Serializable을 구현하고 있지 않기 때문에 직접적으로 객체 직렬화에 사용되어질 수는 없다. 하지만, 자바 2에서는 배열을 직렬화에 사용할 수가 있고, 이미지는 각각의 픽셀들이 배열에 저장되어질수 있으므로 직접적으로 이미지를 직렬화 할 수는 없지만 배열을 사용하여 간접적으로나마 이미지를 객체 직렬화에 사용할 수가 있다.
  • 서버측 프로그램
 
import java.awt.*;
import java.awt.image.*;
import java.net.*;
import java.io.*;
import java.util.*;
public class ImageSerializeServer {
// 서버는 3434 포트로 임의로 설정
    public static final int DEFAULT_PORT = 3434;
    protected int port;
    protected ServerSocket server;
// ImageSerializeServer를 실행한다.
    public static void main (String args[]) {
        new ImageSerializeServer ();
    }
/* 생성자. 서버의 소켓을 생성하고 클라이언트의 연결을 대기 */
    public ImageSerializeServer() {
// 포트를 지정(3434)
        port = DEFAULT_PORT;
        this.port = port;
// 지정된 포트로 서버의 소켓을 생성한다.
        try {
            server = new ServerSocket (port);
        } catch (IOException e) {
            System.err.println ("Error creating server");
            System.exit (1);
        }
// 클라이언트의 연결을 대기
        connectClient();
    }
/* 클라이언트로 부터의 연결을 기다리면서 무한루프를 돈다. */
    public void connectClient() {
        System.out.println ("Server Running");
        try {
// 클라이언트로 부터의 연결이 요청되면 클라이언트의 소켓을 생성 하고 이미지 픽셀을 클라이언트로 보내는 쓰레드를 생성한다.
            while (true) {
                Socket client = server.accept();
                System.out.println ("Connection from: " + client.getInetAddress().getHostName());
                appletConnection c = new appletConnection (client);
// 쓰레드 생성
            }
        } catch (IOException e) {
            System.err.println ("Exception listening");
            System.exit (1);
        }
        System.exit (0);
    }
}
/* 간단한 이미지를 이루는 픽셀을 만들어 이를 네트웍너머의 클라이언트로 보내는 클래스 */
class appletConnection extends Thread {
// 변수 선언
    private final static int b = Color.blue.getRGB();
    private final static int g = Color.green.getRGB();
    private final static int r = Color.red.getRGB();
    protected ObjectOutputStream oos;
// 간단한 이미지 픽셀
    protected int ipTemp[] = { b, b, b, b, b, b, b, b, b, b,
                                         b, b, b, b, b, b, b, b, b, b,
                                         b, b, g, g, g, g, g, g, b, b,
                                         b, b, g, g, g, g, g, g, b, b,
                                         b, b, g, g, r, r, g, g, b, b,
                                         b, b, g, g, r, r, g, g, b, b,
                                         b, b, g, g, g, g, g, g, b, b,
                                         b, b, g, g, g, g, g, g, b, b,
                                         b, b, b, b, b, b, b, b, b, b,
                                         b, b, b, b, b, b, b, b, b, b };
    public appletConnection (Socket client) {
// 객체 직렬화를 위한 출력 스트림을 생성
        try {
            oos = new ObjectOutputStream(client.getOutputStream());
        } catch (IOException e) {
            try {
                client.close();
            } catch (IOException e2) { }
            System.err.println ("Unable to connect.");
            return;
        }
        start();
    }
    public void run () {
/* Serializable을 구현한 imagePixel객체를 생성해서 Vector에 집어넣는다. 따라서 이미지가 하나가 아닌 여러개 일지라도 객체 직렬화를 통해 전달하는것이 가능하다.*/
        Vector vt = new Vector();
        ImagePixel ip = new imagePixel(ipTemp);
        vt.addElement(ip);
// 이미지 픽셀들을 포함하고 있는 벡터를 객체 직렬화 시켜 스트림에 쓴다.
        try {
         
   oos.writeObject(vt);
        } catch (Exception e) {
            System.out.println("Object write exception : " + e);
        }
    }
}
/* 객체의 직렬화를 위해 Serializable을 구현한 imagePixel 클래스 */
class imagePixel implements Serializable {
    protected int ip[];
// 생성자
    public imagePixel(int[] ip) {
        this.ip = ip;
    }
}
  •  실행방법
    javac.exe ImageSerializeServer.java
    java.exe ImageSerializeServer
    Server Running.....(서버가 대기하고 있는 상태)
 
  • 클라이언트측 프로그램
 
import java.applet.*;
import java.awt.*;
import java.awt.image.*;
import java.io.*;
import java.net.*;
import java.util.*;
public class ImageSerializeClient extends Applet {
// 기본 포트는 서버와 같이 3434로 하고 host는 localhost로 설정한다.
    private int defaultPort = 3434;
    private String host = "localhost";
    private Socket s;
    private ObjectInputStream oin;
// 변서 선언
    Vector vt;
    Image myImage;
    public void init() {
/* 서버에 접속하여 소켓을 구한뒤에 객체 직렬화를 해제하기 위한 스트림을 얻는다. 이것은 마치 파일로부터 스트림을 얻는 과정과 비슷하다. */
        try{
            s = new Socket(host, defaultPort);
            oin = new ObjectInputStream(s.getInputStream());
        } catch(IOException e) { }
    }
    public void start() {
// 객체를 스트림으로 부터 읽어들여 직렬화를 해제한다.
        try {
         
   vt = (Vector)oin.readObject();
        } catch(Exception e) {}
// 직렬화가 해제된 객체는 Vector에 들어가게 되고 Vector의 요소인 imagePixel을 구한다.
        imagePixel myPixel = (imagePixel)vt.elementAt(0);
// 이렇게 구해진 imagePixel의 이미지 픽셀을 가지고 실제 이미지를 생성
        myImage = createImage(new MemoryImageSource(10, 10, myPixel.ip, 0, 10));
    }
// 생성된 이미지로 그림을 그린다.
    public void paint(Graphics g) {
        g.drawImage(myImage, 0, 0, 100, 100, this);
    }
}
  •   실행방법
    javac.exe ImageSerializeClient.java
    java.exe ImageSerializeClient
     
  • 애플릿 포함 Html 내용
 
<HTML>
<TITLE>객체 직렬화</TITLE>
<BODY>
<APPLET CODE = "ImageSerializeClient.class" WIDTH = 100 HEIGHT = 100 ></APPLET>
</BODY>
</HTML>
 
  • 최종 결과 화면
사용자 삽입 이미지
 프로그램을 실행하기 위해서는 다음의 과정으로 실행하면 된다.
  1. 서버측 프로그램인 ImageSerializeServer.java를 컴파일하고, 서버 쪽에서 실행시면, "Server Running..."이라는 메시지가 나타나면서 서버측 프로그램이 제대로 작동을 하는 것이다.
  2. 다음으로는 클라이언트 애플릿 프로그램인 ImagSerializeClient.java를 컴파일한다.
  3. 브라우져를 실행시킨뒤에 클라이언트 애플릿 클래스를 실행시키기 위한 html파일이 있는 URL주소를 주소 입력란에 넣는다.
  4. 이때 서버프로그램이 돌아가는 곳과 애플릿을 위한 코드 및 html파일이 위치한 곳은 같은 Machine상에 있어야 한다.
참고로 이번 예제에서는 이미지 픽셀을 직접 그 값을 대입해서 이미지를 대신했는데, 실제 이미지를 이미지 픽셀로 바꾸는 기능이 자바에서 지원되고 있으므로 이를 간단히 소개토록 하겠다.
이미지는 여러개의 색깔을 표현하고 있는 픽셀들의 집합으로 볼 수 있고, 보다 정교하게 이미지를 다루는 작업을 하고 싶다면, 이미지의 픽셀들을 얻어서 조작을 해야한다. 그렇다면 어떻게 이미지의 픽셀들을 얻을 수 있을까? 그것은 PixelGrabber를 통해서 가능하다.
PixelGrabber는 이미지와 해당 이미지의 정보들을 기반으로 이미지 픽셀을 구해준다.
다음은 실제로 이러한 것을 보여주는 코드이다.
 
int imagePixel[];                         // 이미지의 픽셀들을 저장할 배열
int imageWidth, imageHeight;      
// 이미지의 폭과 높이를 가지게 될 변수들
Image imageObject;                  
// 이미지 객체
/* imageObject로부터 픽셀들을 얻어서 imagePixel배열에 저장한다. */
private void getImagePixel() {
    imagePixel = new int[imageWidth * imageHeight];    // imagePixel을 초기화 한다.
    PixelGrabber grabber = new PixelGrabber(imageObject, 0, 0, imageWidth, imageHeight, imagePixel, 0,                                                           imageWidth);
// grabber를 생성해서 픽셀들을 얻는다.
    try {
        grabber.grabPixels();
    } catch(Exception e) {
        return; }
    if((grabber.status() & ImageObserber.ABORT) != 0)
        return;
}

 
참고 문헌
  • Java Examples In a Nutshell (O'RELLY)
  • Java I/O (O'RELLY)
:
Posted by 뽀기

정지훈 (필자) ( 마이크로소프트웨어 )   2004/10/06

 

우리가 현재 살고 있는 산업사회에 있어서 가장 중요한 것은 무엇인가? 혹자는 믿음이라고 답할 것이고, 혹자는 밥이라고 답할 것이며, 어떤 사람은 그저 애니메이션이나 실컷 보면서 살 수 있으면 그만이라고 말할 것이다. 무엇이 되었든 개인에게 있어서 틀린 답은 아니겠으나 산업사회의 발전을 위해서 가장 중요한 사회학적인 요인을 하나만 선택하라고 하면 그것은 곧 ‘생산성(productivity)’이 될 것이다. 소위 우리가 말하는 ‘산업혁명’이라는 것도 기실은 생산성의 폭발적인 증대로 귀결될 수 있다.

소프트웨어를 개발하는 입장에서도 이 생산성의 문제는 무척이나 중요하다. 같은 수준의 완성품을 만드는 데 있어서 가능하면 적은 시간과 적은 노동력과 원자재만을 이용하는 방법을 연구하는 것은 그런 측면에서 커다란 의미를 가지는 것이다. 이번 호 특집에서 언급하는 MDA(Model Driven Architecture) 역시 소프트웨어의 생산성과 관련해서 중요한 의미를 가진다.

소프트웨어의 생산성은 어떻게 높아지는가
그렇다면 과연 어떻게 하면 소프트웨어의 생산성을 높일 수 있는가? 이 문제는 비단 최근에 제기되는 문제가 아니다. 물론 과거에 비해 복잡한 시스템 요구사항과 수많은 기술들이 등장하고 있는 근래에 그 중요성이 높아지고는 있으나 이미 소프트웨어 초창기부터 이 문제는 많은 사람들의 고민거리였다고 말할 수 있다. 이 문제를 해결하기 위해서 등장한 대표적인 학문이 바로 소프트웨어 공학이다.

소프트웨어 공학의 영역은 상당히 포괄적이다. 소프트웨어 개발에 투여되는 수많은 자원들의 효과적인 관리 방법과 평가, 그리고 조직과 의사결정 구조, 심지어는 개발자들의 심리상태에 대한 것도 대상이 될 수 있다. 소프트웨어 공학에 대해서 자세한 이야기를 하자면 소프트웨어의 위기에서부터 많은 방법론에 대한 것까지 커다란 연구 분야라고 할 수 있으며, 그 중요성은 최근의 복잡해진 환경에 의해 점점 더 높아가고 있다. 거창하게 소프트웨어 공학을 이야기했지만 소프트웨어 생산성을 높이기 위해 그 동안 시도됐던 몇 가지 방법에 대해서 간단히 이해하는 것이 MDA에 대한 이야기를 풀어나가는 데 도움이 될 것이다.

전통적인 방법론 vs. 창조적 프로그래밍
소프트웨어의 생산성을 높이기 위한 방법으로 그 동안 다양한 방법론과 이론이 등장했다. 이들에 대한 자세한 논의는 한 권의 책으로도 부족할 것이므로 여기서는 대표적인 몇 가지의 특징에 대해서 간단히 알아보기로 하자. 소프트웨어 공학의 정의로 본다면 모든 방법론들이 소프트웨어 공학으로 설명할 수 있겠으나 최근에 각광받는 몇 가지 방법론이 기존의 방법론과는 상당한 철학적인 차이를 보여주고 있기 때문에 여기서는 이들을 포괄적으로 ‘창조적 프로그래밍’으로 표현하고자 한다.

필자의 경우 소프트웨어 공학을 전문적으로 전공한 것은 아니므로 다소의 용어 선택과 학문적인 설명에 있어서는 오류가 있을 수 있으나 MDA를 설명하기 이전에 그 동안의 경험했던 것들을 바탕으로 접근하는 것이 도움이 될 것이라 생각하기 때문에 이 부분을 언급하도록 하겠다. 혹시라도 필자의 글에 대해 다른 의견이 있는 독자는 언제든지 필자에게 메일을 보낸다면 검토 후 다음 기회에 정정하도록 할 것이다.

전통적인 소프트웨어 공학 방법론은 소프트웨어의 개발 단계를 요구, 분석, 설계, 구현, 테스트와 유지보수로 분류하고 각 단계별로 소프트웨어의 생산성에 영향을 주는 여러 요소들을 관리하고 조정해 생산성을 높이는 방법과 절차, 그리고 이를 지원하는 여러 가지 도구 등으로 구성된다. 소프트웨어 개발에 관한 접근 방식에 따라 기능 중심의 개발방법론, 구조적 개발방법론, 객체지향 개발방법론 등이 있으며 최근에 가장 각광받는 것은 객체지향 개발방법론이라고 할 수 있다.

이러한 전통적인 소프트웨어 공학 방법론은 기본적으로 소프트웨어의 개발을 제조업에서의 생산 공정의 연장선상에서 소프트웨어라는 생산 대상의 특성을 고려하여 만들어지기 때문에 공정의 관리와 제어에 초점을 맞추게 되는 경우가 많다. 예를 들면 프로젝트의 관리를 위해 제안서 및 프로젝트 비용 산정, 프로젝트 계획과 스케줄링, 감독과 검토 과정을 거친 뒤 적합한 인력 배치 등을 하고 프로젝트 팀으로 구성된 조직은 조직 관리의 측면에서 최대한의 생산성을 갖출 수 있는 방향을 선택하도록 하는 것이다.

이런 소프트웨어 공학 방법론은 커다란 프로젝트의 위험성을 최대한 관리하면서 기존에 제조업과 사회과학 분야에서 개발된 여러 가지 형태의 도구를 응용할 수 있기 때문에 중간 규모 이상의 프로젝트에서 상당한 효과를 발휘한다. 이에 비해 창조적인 프로그래밍에 중점을 둔 방법론은 기본적으로 소프트웨어의 개발을 제조업에서의 생산 공정으로 보기보다는 개개인의 숨겨진 잠재력을 최대한 발휘해서 조합하는 것을 더욱 중요하게 생각한다. 그러므로 창조적인 프로그래밍에서는 소프트웨어 개발이란 일종의 공동 예술 작품을 완성하는 것과 비슷하다고 간주한다.

모든 방법론을 이러한 잣대로 분류하는 것은 무리이지만 최근에 각광을 받는 XP(eXtreme Programming), 애자일 소프트웨어 개발(Agile Software Development) 등의 방법론은 기존의 방법론에 비해 이러한 특성의 중요성을 많이 인정하고 있다. 필자는 개인적으로 이러한 두 가지 방법론을 모두 시험해 보았는데 어느 한 쪽의 방법론이 더 우월하다고 말하기는 어렵지만 전반적으로 프로젝트의 규모가 커지고 개발 인원의 수준의 차이가 많이 나는 구성원을 가질수록 전통적인 소프트웨어 개발 방법론이 효과적이며, 반대로 소규모 프로젝트이면서 개발 인원의 수준의 차가 적고 쌍방의 대화가 잘 이뤄질 수 있는 그룹이라면 창조적인 프로그래밍 방법론이 훨씬 효과적이었다.

소프트웨어 개발방법론에 관심이 있는 독자라면 시중에 좋은 책들이 많이 나와 있으므로 이들을 참고하면 큰 도움이 될 것이다. 전통적인 소프트웨어 공학적인 방법론은 교과서도 많이 나와 있고 방법론 자체에 대한 소개도 인터넷에서 쉽게 찾아볼 수 있다. 창조적인 프로그래밍과 관련해서는 켄트 벡(Kent Beck)이 쓴 ‘Extreme Programming Explained’와 애자일얼라이언스(AgileAlliance) 웹 사이트(http://www.aanpo.org/home)를 참고하기 바란다.

CASE vs. MDA
CASE(Computer Aided Software Engineering)는 소프트웨어 개발이나 각 과정별 작업을 컴퓨터의 도움을 받아 공학적 기술을 자동화하는 것으로 각각의 작업의 자동화에 이용되는 도구를 흔히 CASE 툴이라고 한다. 역사적으로 볼 때는 1980년대 초의 구조적 개발방법론이 80년대 후반의 CASE로 발전하게 되는데 초기의 CASE 제품들은 다양한 목적을 가지고 있었다. 대표적인 것으로는 모델링 작업을 도와주거나 데이터베이스를 생성하는 것에서부터 코드의 자동생성 및 역공학 툴, 그리고 프로그래머가 아닌 사람이 프로그래밍할 수 있도록 도와주는 GUI 툴 등이 있다.

초창기 CASE 제품들이 등장하던 시기에는 메인프레임에 대한 COBOL 프로그래밍과 관계형 데이터베이스에 대한 SQL CASE 툴이 주를 이뤘는데, 운영체제나 데이터베이스 아키텍처를 지원하기 위한 정보를 모아두는 공통적인 저장소(repository)가 존재하지 않아서 그 활용에 제한점이 있었다. 이를 극복하기 위해서 IBM에서는 AD 사이클/저장소(Cycle/Repository)를 만드는 프로젝트를 시도했지만 실패로 끝났다.

1990년대에 들어서면서 컴퓨팅 환경이 급속도로 메인프레임에서 유닉스, PC 기반의 운영체제로 그리고 컴퓨터 언어의 측면에서는 C와 베이직 같은 언어가 주류 언어로 자리를 잡게 되면서 기존의 CASE 툴의 활용도가 급속히 떨어지는 현상을 보여주게 된다. 1990년대 후반에 들어가면서 객체지향 프로그래밍 언어의 사용이 일반화되고 1997년에 OMG(Object Management Group)에서 UML(Unified Modeling Language)을 표준화한 것을 계기로 객체지향 모델링을 이용한 사례가 늘어나면서 기존의 CASE보다는 객체지향 개발방법론을 따르는 새로운 형태의 비주얼 툴들이 각광을 받게 되는데 대표적인 제품이 현재는 IBM에 합병된 래쇼날(Rational)의 로즈(Rose)나 볼랜드에 합병된 투게더(Together) 등이다.

CASE 툴의 경우 아직까지 성공과 실패를 논하기에는 이르지만 초기에 가졌던 거창한 목표 달성에는 이르지 못했다고 판단된다. 가장 큰 이유는 CASE가 지향하는 목표가 너무나 방대하고 이를 지원하는 툴들이 실제로 등장해서 소프트웨어 개발에 이용되기에는 실무 소프트웨어 개발 프로세스가 개발 방법론이라는 것을 모두 소화할 만큼 충분히 성숙하지 않았다는 것을 들 수 있을 것이다. 다시 말해, 이론은 좋았지만 현실세계에 받아들여지기에는 아직 먼 이상향이라고나 할까?

이러한 현상은 비단 CASE 툴에 국한된 것만은 아니다. OMG에서 최초로 표준화 작업을 했던 분산객체 표준인 CORBA(Common Object Request Broker Architecture) 역시 초기의 기대와는 다르게 그렇게 많은 영역에서 넓게 채택되지 못했다. 그 이유는 여러 가지가 있겠으나 가장 커다란 요인으로 생각되는 것 중의 하나가 지나치게 무겁고 복잡한 표준 규격으로 인해 실제 소프트웨어 개발에 있어서 쉽게 받아들여지기 어려웠다는 점이다. 그에 비해 최근의 웹 서비스가 그 자체가 가지고 있는 많은 약점에도 불구하고 비교적 쉽게 채택되고 있는 것은 가벼운 표준 규격으로 인해 자유로운 선택의 영역을 넓혔으며 여러 가지 지원 도구들을 통해 쉽게 구현이 가능했기 때문이라고 생각된다.

MDA는 이러한 CASE와 CORBA에서의 교훈을 바탕으로 CASE처럼 지나치게 야심차지 않으면서 보다 느슨하게 연관되어 있는 표준안들의 연결고리를 제공하는 방식으로 접근하고 있다. 또한 이러한 연결고리를 쉽게 구성할 수 있는 도구들을 적시에 제공함으로써 그 성공 가능성을 높이고 있다.

MDA 탄생 설화 - CORBA, UML, MOF 그리고 MDA
OMG는 1989년에 설립이 된 객체 기술에 대한 사용을 증진시키고 이에 대한 표준안을 만들기 위해 소프트웨어 산업계에서 구성한 컨소시엄이다. 이들이 모여서 최초로 만든 것이 바로 OMA(Object Management Architecture)인데, OMA에서 가장 중요한 역할을 하는 것이 바로 CORBA이다. CORBA는 다양한 프로그래밍 언어 간의 상호운용성을 위해 IDL(Interface Definition Language)이라는 것을 도입했다. 플랫폼에 독립적인 프로토콜을 위해 IIOP(Internet Inter-Operable Protocol)와 같은 프로토콜 표준안을 제시하였다. <그림 1>은 OMA를 도식화해서 나타낸 것이다.

<그림 1> Object Management Architecture


그리고 OMG에서는 객체지향 소프트웨어 디자인을 위해서 그동안 다양하게 제시되었던 모델링 시스템을 표준화해 UML을 1997년에 발표하게 된다. 그 밖에도 OMG에서 발표된 대표적인 표준안으로는 플랫폼에 독립적으로 메타 데이터와 데이터를 정의, 조작, 통합할 수 있는 모델 기반의 프레임워크인 MOF(Meta-Object Facility), MOF를 모델 기반의 XML 통합 프레임워크로 재구성하며 MOF 기반의 메타모델의 XML 맵핑을 지원하는 XMI(XML Metadata Interchange), 데이터 웨어하우스 도구, 웨어하우스 플랫폼 그리고 서로 다른 환경에 분산된 웨어하우스 메타 데이터 저장소들 사이의 비즈니스 메타 데이터의 상호교환을 가능하게 하는 표준 인터페이스인 CWM(Common Warehouse Metamodel) 등이 있다. 이들은 MDA를 구성한 하부 요소로서 그 역할을 하게 된다.

CORBA에서 UML로의 중심 이동
필자는 2000년부터 수년 간 OMG 총회를 참석하면서 매번 총회에 참석할 때마다 조금씩 변화해가는 분위기를 느낄 수 있었다. 2000년 초기의 OMG 총회만 하더라도 대부분의 업체들은 총회의 세부 모임에 있어서 CORBA를 중심으로 CORBA 표준 규격안을 발전시키는 데 주력하였다. 또한 CORBA를 분산객체 기술의 산업 표준안으로 발표했음에도 불구하고 이와 유사한 다양한 플랫폼들이 등장하였기 때문에 이들과의 상호운용성을 위한 작업들도 지속적으로 진행되었다. 대표적인 것이 CORBA/COM, CORBA/J2EE의 상호운용성을 확보하는 것이었다.

그리고 비교적 CORBA 표준안이 많이 채택되었던 통신업체와 국방 관련 업체들을 중심으로 해당 도메인 영역에서의 서비스(Service)와 퍼실리티(Facility)를 정의하는 작업들이 한창이었다. 그런데 해가 바뀌면서 OMG 총회에 참석하는 사람들의 움직임이 지속적으로 UML쪽으로 움직이기 시작했다. 그에 비해 CORBA 표준안과 관련한 것들은 특정 도메인 영역으로 축소가 되는 양상을 보이면서 OMG의 중심이 CORBA에서 UML로 넘어가기 시작했다.

이러한 변화는 기술 표준안의 우월성에서 기인한 것이 아니라 여타 다른 경쟁 기술들의 존재로 인해 더 이상 산업계 표준으로서의 추진동력을 점차 상실하고 있었던 CORBA에 비해 객체지향 프로그래밍의 폭발적인 증가와 경쟁이 되는 표준안이 없는 상태에서 사실상의 유일한 표준안으로서 그 위세를 떨쳐나가던 UML의 시장에서의 반응에 의한 것이다. UML은 기존의 플랫폼의 우월성이나 벤더들 사이의 제품 팔아먹기 경쟁에서 옆으로 비켜난 상태로 많은 개발자들에게 받아들여지면서 자연스럽게 OMG를 이끌어가는 핵심 표준안으로 각광받게 된 것이다.

MDA의 탄생과 그 이후
이때부터 OMG 내부에서는 MDA에 대한 구상을 의논하는 그룹들이 생겨나기 시작했다. UML의 시장주도적인 힘을 바탕으로 CORBA가 이뤄내지 못했던 이기종 플랫폼 및 언어독립성을 비교적 받아들이기 쉬운 방법으로 제시하려는 노력들이 모여서 MDA가 탄생하게 되었다. MDA는 2001년 3월 OMG의 CEO인 리차드 솔리(Richard M. Soley)가 ‘OMG Model Driven Architecture’라는 제목의 프리젠테이션을 통해 공식적으로는 처음 세상에 알려지게 되었다. 이 발표는 OMG 웹 사이트에서 현재도 10분 정도의 오디오 파일로 들을 수 있으므로 관심 있는 독자는 한 번쯤 들어보기 바란다. 이 발표를 시작으로 MDA에 대한 구체적인 규격을 논의하기 시작해서 2003년 6월에 공식적인 ‘MDA Guide v 1.0’이 공개되었다.

2001년 3월 MDA에 대한 발표가 있은 뒤부터 MDA 표준안을 실체화하고 MDA에서 내세운 개념들을 구체화하기 위한 움직임들은 점점 활성화되었는데, 그 이후의 OMG 총회에서는 이러한 논의들이 하나씩 소기의 성과를 거두게 된다. 필자는 처음 MDA를 발표하는 자리에 있었는데, 그 발표를 보는 순간 이것이 앞으로 갈 길은 멀어 보이지만 언젠가 세상에서 커다란 반향을 끌어낼 수 있을 것이라는 느낌을 받았다. 그러나 그 때만 하더라도 적어도 5년 안에는 저기서 이야기하는 것이 실체화하기는 어려울 것이라고 생각했었다.

이런 필자의 생각을 깨뜨린 것이 2001년 여름 OMG 총회에서 최초의 MDA를 지원하는 개발 툴로 소개된 아크스타일러(ArcStyler )였다. 당시의 제품은 UML 모델을 바탕으로 플랫폼에 독립적인 모델(PIM, Platform Independent Model)과 플랫폼 의존적인 모델(PSM, Platform Specific Model)을 따로 관리하는 기본적인 MDA의 형태를 갖추기는 했으나 실제로 지원하는 플랫폼은 J2EE 밖에 없는 것이었다. 그렇지만 MDA에 대한 기본적인 개념이나 구현 가능성을 실제로 보여줬기 때문에 당시에 커다란 반향을 불러일으켰다. 아크스타일러는 현재 볼랜드의 엔터프라이즈 스튜디오에 통합되어 있으며 이를 시발점으로 여러 MDA 지원 개발 툴들이 등장하기 시작했다.

MDA의 목표는 소프트웨어 분석가와 개발자가 소프트웨어와 비즈니스 자산을 기술하는데 이용할 수 있는 엔터프라이즈 아키텍처 모델링이 가능하도록 하는 것이다. 이러한 아키텍처를 소프트웨어 도구를 이용해서 만들어냄으로써 실제 소프트웨어 개발에 있어서 아키텍처를 구현하는 특정 애플리케이션을 쉽게 작성할 수 있게 되며 애플리케이션을 다양한 변경 요구에 맞게 쉽게 변경할 수 있을 것이다.

MDA 구성 요소
MDA는 여러 가지 모델과 표준들에 의해서 구성된다. 모든 MDA 모델 들은 MOF라는 추상적인 메타모델에 기초를 두고 있기 때문에 모두 연관되어 있다. 다시 말해 모든 MDA 모델은 MOF와 호환성이 있다. 이런 특징은 MDA에서 이용된 모든 모델들이 다른 MOF 호환 모델들과 호환성을 가진다는 것을 보장한다.

MDA에서 또 한 가지 중요한 부분이 UML 프로파일(profile)이다. UML 프로파일은 UML을 확장한 것이기 때문에 MOF와 호환성이 있으며, UML 2.0에서는 UML의 다양한 기능적인 사용 방법을 기술할 수 있다. UML 프로파일은 UML의 확장인 동시에 그 자체로 MOF 메타모델이다. 일부의 MDA 메타모델은 이미 OMG에서 과거에 정의된 것들이며 일부는 OMG 외부의 그룹 또는 벤더들에 의해 MOF 호환이 되는 형태로 만들어진 것이다. 이러한 OMG 외부의 메타모델들은 비록 공식적인 OMG 표준은 아니지만 MOF 호환성이 있기 때문에 MDA 개발 과정에는 사용되는 데 문제가 없다. 대표적인 MDA 모델들과 프로파일들로는 다음과 같은 것들이 있다.

CWM
CWM(Common Warehouse Metamodel)은 OMG에서 표준화한 데이터 웨어하우스를 관리하는 데 이용되는 메타 데이터 모델이다. CWM을 이용하면 개발자들이 관계형 데이터베이스 테이블, 레코드, 구조체, OLAP, XML 그리고 다차원 데이터베이스 디자인 등과 같은 수많은 데이터 모델 또는 포맷을 생성할 수 있다. 또한 CWM에는 데이터 웨어하우스 이외에도 사용될 수 있는 유용한 부분들도 있는데 예를 들면 데이터 모델이나 데이터 변환, 소프트웨어 배포 등에 관련된 내용도 포함되어 있다.

UML 메타모델
UML의 초기 버전은 MOF와의 완전한 호환성이 보장되지 않지만 UML 2.0은 MOF 호환성을 가진다. 이런 이유로 진정한 MDA 개발을 위해서는 UML 2.0이 사용돼야 한다. UML은 핵심 모델링 개념들과 이들을 위한 다이어그램들로 정의된다. 또한 개발자들이 UML 구성 요소에 다양한 제한(constraint)을 할 수 있도록 허용하고 있다. UML 2.0에서는 UML이 모델들의 행위를 지정할 수 있으며 이러한 행위의 표현들이 직접 코드로 변경된다.

XMI
XMI 표준은 MOF와 호환성이 있는 모델들을 XML 문서 형태로 표현하고 이를 MOF 호환 데이터베이스에 저장할 수 있도록 하는 표준이다. 그러므로 사실상 XMI 문서는 곧 MOF XML 문서라고 할 수 있다.

UML 프로파일
UML 프로파일은 특정 도메인에 대한 UML 모델을 작성하기 위한 일반 확장(generic extension) 메커니즘을 정의하는 것이다. 그러므로 UML 프로파일은 추가적인 스테레오타입(stereotype)과 태그 값(tagged value)이 적용된 엘리먼트(element), 애트리뷰트(attribute), 메쏘드(method), 링크(link) 등으로 이뤄진다. UML 프로파일은 이러한 확장 컬렉션을 이용해서 특정 도메인에 대한 모델링 문제를 기술하고 해당 도메인의 내용들을 모델링할 수 있도록 해준다.

현재 다양한 UML 프로파일이 나와 있으나 MDA와 관련해서는 플랫폼에 따라 각각의 프로파일이 정의될 수 있다. 현재 CORBA 프로파일, EAI 프로파일, EDOC 프로파일, 리얼타임 컴퓨팅을 위한 스케줄링 프로파일 등이 OMG에서 정의되었으며 EJB 프로파일은 JCP(Java Community Process)를 통해, 닷넷 프로파일은 마이크로소프트에 의해 정의되고 있다.

MDA 개발 프로세스와 특징
MDA는 이와 같이 그동안 발표된 여러 OMG의 표준 기술들이 하나로 접목되고 유기적으로 연결된 형태를 가지고 있다. 그 중에서도 MOF와 UML 프로파일의 역할이 가장 중요하다는 것을 아마도 독자도 느낄 수 있을 것이다. 이러한 구성상의 역할과는 별도로 MDA는 소프트웨어 리소스의 관리와 개발 자체에도 초점을 맞추고 있기 때문에 소프트웨어 개발 프로세스에서 모델들이 어떻게 이용돼야 하는지에 대해서도 기술하고 있다.

<그림 2>는 비즈니스 프로세스나 소프트웨어에 대한 기술서를 바탕으로 추상적인 모델을 추출하고, 추상적인 모델을 실행 가능한 구현 모델로 변환시키는 과정을 보여준다. 이런 프로세스에서 사용된 모델들은 UML과 같은 메타모델로 표현되는데 일부의 모델은 플랫폼과 무관하고 일부는 플랫폼과 밀접한 관계를 가진다.

<그림 2> MDA 개발 프로세스


<그림 2>에서 3가지 유형의 MDA 모델을 볼 수 있을 것이다. 일반적으로 비즈니스를 분석하고 요구사항을 추출하는 역할을 맡은 분석가가 CIM(Computation Independent Model)을 모델링하며, 비즈니스 아키텍트나 디자이너가 PIM(Platform Independent Model)을 만든다. PIM 모델은 특정한 구현 모델과는 독립적으로 어떤 형태의 아키텍처를 가질 것인지 기술한다. 그리고 나서 개발자와 테스터가 소프트웨어 툴을 이용해서 PSM을 PIM에서 뽑아낸 뒤에 이를 플랫폼 특성에 맞게 최적화를 하고 코드를 뽑아내게 된다.

MDA 개발 프로세스를 단계별 스텝으로 간단히 정리해 보면 다음과 같다.


[1] 애플리케이션에 대한 비즈니스 요구사항을 수집한다.
[2] 도메인 모델에 대한 UML 다이어그램을 작성한다. 먼저 J2EE, 닷넷, CORBA와 같은 특정 기술에 종속적이지 않은 모델을 먼저 만든다. 이렇게 작성된 UML 모델은 핵심 비즈니스 서비스와 컴포넌트를 나타내게 되는데, 예를 들자면 가격 결정 엔진이라든지 쇼핑 카트 모델이라든지 주문 모델 같은 것들이 될 것이다. 이러한 UML 모델을 PIM이라고 한다.
[3] 애플리케이션에 대한 특정 기술과 관련한 UML 모델을 작성한다. 이와 같이 특정 기술에 종속적인 UML 모델을 PSM이라고 한다. PSM은 직접 작성할 수도 있고 MDA 지원 툴을 이용해서 PIM에서 자동으로 만들어내거나 또는 만들어진 모델을 수정하는 방식으로 작업할 수 있을 것이다.
[4] 마지막으로 MDA 툴을 이용해서 애플리케이션 코드를 만들어낸다. 현재 이미 J2EE의 경우에는 MDA 툴들이 대부분의 서블릿, JSP, EJB와 관련한 코드를 자동을 생성해준다. 그 다음에 만들어진 코드를 바탕으로 수정과 최적화 과정을 거쳐서 프로젝트를 완성한다.


이와 같이 MDA에서는 결국 UML 모델에서 코드를 만들어낸다. 그런데 어찌보면 이것이 새삼스러운 것은 아니다. 아마도 많은 개발자들이 래쇼날 로즈를 이용해서 자바 클래스를 UML 모델에서 만들어낸 경험이 있을 것이다. 이러한 작업과 MDA와의 가장 큰 차이점이라면 MDA는 플랫폼 독립적인 모델에서 시작해서 다양한 플랫폼을 지원하는 프로세스와 툴을 거의 완벽하게 지원한다는 점일 것이다. 이런 측면에서 간단히 MDA에서 만들어지는 모델들과 코드에 대해서 정리를 하면 다음과 같다.


◆ MDA는 다른 디자인 프로세스보다 고수준의 추상화를 먼저 시작한다. 예를 들어 PIM은 매우 추상적이다. 여기에는 엔티티와 서비스만 정의되어 있을 뿐이다.
◆ PSM은 메타 데이터의 형태로 애플리케이션을 완전하게 기술한다. PSM 수준에서 개발자들은 해당 기술과 관련된 디자인을 직접 코드를 건드리지 않고 향상시킬 수 있다.
◆ PSM에서 만들어진 코드는 완성된 애플리케이션에 근접하게 된다. 기존에 많은 도구들이 자동으로 만들어낸 코드들이 애플리케이션의 일부에 해당하는 것에 비해서 그 범위와 정도에 큰 차이가 있다.
◆ PIM에서 PSM을 만들어내고 PSM에서 코드를 만들어내는 알고리즘은 기본적인 것은 제공되지만 아키텍트가 변경하거나 재정의할 수 있다.


MDA의 중요한 특징 중의 하나가 이와 같이 모델링 언어를 프로그래밍 언어처럼 이용하는 것이다. 모델링 언어는 일반적으로 자바나 C++와 같은 프로그래밍 언어에서 하는 것보다 더 높은 수준에서 ‘추상화(abstraction)’를 가능하게 하며, 이러한 추상화 프로세스를 통해 개발자들은 시스템 코드에 꼭 필요한 데이터 이외의 것들은 모두 숨길 수가 있다. 이런 과정은 개발 과정에서의 복잡도를 감소시키는 효과를 가져오기 때문에 개발의 생산성을 높일 수가 있다.

아마도 여기까지 읽은 독자는 일반적인 객체지향 프로그래밍 언어의 가장 큰 특징 중의 하나인 ‘캡슐화(encapsulation)’라는 단어를 떠올린 경우가 많을 것이다. 보통 객체지향 프로그래밍 언어의 3대 특징을 이야기하면 캡슐화, 상속성(inheritance) 그리고 다형성(polymorphism)을 언급한다. 그 중에서도 객체지향이라는 개념을 위해서 가장 중요한 것은 바로 캡슐화이다. 다른 두 가지 특징은 완전히 지원하지 않더라도 객체지향 언어라는 말을 할 수 있지만 캡슐화가 지원되지 않으면 객체지향이라고 말할 수가 없다. MDA는 이러한 객체지향 개념의 캡슐화를 극대화한 것으로 이해할 수 있다. 단순히 프로그래밍 언어 수준에서의 캡슐화가 아닌 모델링을 포함해서 구현에 이르는 과정을 단계별로 캡슐화한 것이 바로 MDA의 정체이다.

MDA 개발 프로세스에서의 이러한 추상화는 시스템의 규격을 하부의 컴퓨팅 환경에 덜 좌우되게 하기 때문에 시스템의 생산성을 높일 뿐 아니라 지속성도 높여준다. 이와 같은 추상화의 수준을 높임으로서 소프트웨어 개발 생산성을 높이려는 시도는 새삼스러운 것은 아니다. 가장 직접적인 예로 현재 우리가 사용하고 있는 대부분의 프로그래밍 언어는 3세대 언어(3GL, 3rd Generation Language)라고 할 수 있는데, 3GL 언어는 기본적으로 어셈블리 언어를 추상화한 것이라고 할 수 있다.

이를 통해 어셈블리 언어로 프로그래밍할 때 알아야 했던 많은 어려운 문제들을 간단하게 컴파일러에게 맡겨버린 뒤 소프트웨어의 생산성이 급속도로 높아졌다는 것은 모두가 인정할 것이다. 그러나 3GL이 처음 등장했을 때만 해도 여러 개발자들은 3GL이 개발자들이 코드를 작성하기 쉽게 해주고 높은 수준의 생산성을 보장하지만 만들어진 실행 프로그램의 수행 속도 문제 때문에 한동안 수많은 논쟁을 했었다. 이러한 문제는 하드웨어의 비약적인 성능 발전으로 인해 자연스럽게 작은 문제로 치부되기 시작했고 이를 통한 소프트웨어 생산성의 증가는 많은 것을 가능하게 하였다.

왜 MDA인가
지금까지 설명한 내용을 충분히 이해한 독자라면 아마도 MDA가 어떤 녀석인지 정도에 대해서는 감을 잡았을 것이다. 그렇다면 왜 MDA가 최근에 각광받게 되었는지에 대해서 조금 더 생각해 보도록 하자. MDA는 MDA 표준을 적용해서 모델의 자동화와 변환(transformation)을 통해 여러 플랫폼을 쉽게 지원하고 개발자의 입장에서는 시간을 많이 잡아먹는 코드 작성 부분을 줄일 수 있으며 개발 프로세스의 측면에서도 품질 관리(QA, Quality Assurance)를 수월하게 할 수 있다. 그리고 과거의 CASE나 4GL과는 다르게 구현시에 유연한 컨트롤이나 커스텀화가 가능하도록 하였다. 잘 알다시피 손으로 코딩하는 부분이 줄고 동시에 빨리 버그를 잡을 수 있다면 소프트웨어의 생산성은 월등히 증가하게 된다. MDA를 이용하는 것이 좋은 이유를 몇 가지 나열해서 설명하면 다음과 같다.

기술 플랫폼 및 기능 변화에 대한 신속한 대응
MDA에서는 PIM과 PSM을 분리했기 때문에 PIM을 변경하지 않고도 기술 플랫폼의 변화나 요구사항 변화에 발빠르게 대처할 수 있다. 예를 들어, 새로운 플랫폼에 대한 애플리케이션을 배포해야 하는 경우 PIM에는 수정이 필요 없으므로 단지 PIM을 이용해서 새로운 플랫폼에 대한 PSM을 자동으로 만들어내고, 이를 수정해서 코드를 다시 생성하는 과정을 통해 쉽게 새로운 기술 플랫폼에 대한 대응이 가능하다. 또한 기능의 변화를 요구하는 경우에도 PSM 수준에서의 변경을 고려하지 않고 PIM에 새로운 기능 변화와 관련된 수정을 가함으로써 쉽게 대응할 수 있다.

시스템의 지속성
플랫폼 의존적인 시스템은 기존의 아키텍처가 새로운 요구사항을 만족시키지 못하는 경우가 많이 발생한다. 이 경우에 완전히 새로운 아키텍처를 다시 잡기보다는 많은 사용자들은 작은 패치와 수정으로 이런 문제를 해결하기를 원하며 현실적으로도 그렇게 할 수밖에 없는 시간?경제적 여유만을 가지는 경우가 많다. MDA를 이용하면 기능과 아키텍처를 분리해서 정의하기 때문에 아키텍처의 변화가 있더라도 변환 과정을 거쳐 구현 과정으로 쉽게 진행할 수 있기 때문에 기능과 요구사항 변화에 의한 아키텍처 변경에 비교적 자유롭다. 이런 장점은 시스템의 생명주기를 연장시키며 더 안정된 시스템 유지를 가능하게 한다.

개발 생산성 증진
MDA는 모델의 자동화와 변환을 통해 여러 플랫폼을 쉽게 지원하고 시간을 많이 잡아먹는 코드 작성 부분을 줄일 수 있으며 쉽게 유지보수가 되도록 한다. 이와 같이 손으로 코딩하는 부분이 줄고 동시에 빨리 버그를 잡을 수 있다. 또한 한 번 작성된 PIM의 경우 비즈니스 핵심 부분에 대한 모델이기 때문에 향후 다른 시스템에서도 쉽게 이용될 수 있으며 재사용성이 높아지게 된다.

용이한 문서 작성
디자인 문서를 업데이트하고 관련 코드를 수작업으로 관리하는 것은 매우 시간이 많이 걸리고 귀찮은 작업이다. MDA에서는 모델과 코드, 문서가 항상 동기화되기 때문에 이러한 문서 작업에 필요한 일의 양을 줄여준다.

품질 관리 비용의 감소
개발 과정에서 소프트웨어의 문제가 늦게 발견될수록 이를 고치는 데 들어가는 비용은 증가한다. MDA 모델 자동화와 테스트 툴을 이용하면 개발자들이 애플리케이션을 모델 수준에서 테스트할 수 있기 때문에 디자인의 문제점이나 애플리케이션 로직의 에러를 빨리 잡을 수 있다. 이를 통해 품질 관리에 들어가는 비용도 감소한다.

양질의 시스템 구축
PIM의 단순함은 양질의 시스템 구축을 가능하게 한다. 모델링은 팀 멤버들 사이의 의사소통을 원활하게 만들고 동시에 결점이 있을 때 이를 빨리 제거할 수 있도록 도와준다. 또한 MDA의 자동화 도구들은 잘 정리된 코딩 패턴을 모델에 적용하기 때문에 손으로 직접 작성한 코드에 비해 결점이 적을 가능성이 많다.

기술 플랫폼 통합의 기로에 서서
OMG는 소프트웨어 개발에 있어서 운영체제와 프로그래밍 언어의 통합이라는 어려운 과제를 수행하기 위해 과거 CORBA를 통해 불철주야 노력했으나 현 시점에서 판단해 보면 그다지 성공적이라고 하기는 어렵다. 그렇지만 CORBA를 논의하면서 모였던 많은 지식과 경험들이 바탕이 되어 MDA라는 새로운 접근 방법을 합의하는 데 성공했으므로 그런 노력들이 헛되기만 한 것은 아니었다는 것을 보여준다. 마지막으로 MDA가 기술 플랫폼 통합의 기로에서 앞으로 어떤 길을 걷게 될 것인지 여러 가지 측면에서 살펴보자.

환경의 변화
최근의 소프트웨어 개발 환경은 과거 CORBA를 만들어낼 때보다도 더 많은 프로그래밍 언어와 기술 플랫폼들이 등장하고 있다. 그리고 소프트웨어의 복잡도도 나날이 증가하고 있으며 이런 복잡한 소프트웨어를 더 효과적으로 개발하는 것이 가장 커다란 도전이 되고 있다.

그렇지만 과거의 환경에 비해 좋은 점들도 있다. 그것은 실질적인 표준으로 받아들여지는 기술이 많이 등장하고 있다는 것이다. 네트워크 프로토콜에 있어서는 TCP/IP, 데이터베이스 질의와 관련한 SQL, 객체지향 모델링에 대한 UML 등은 이제 진정한 표준으로서의 입지를 굳혀가고 있다. 또한 리눅스에서 출발하고 아파치를 통해 더욱 널리 퍼져가고 있는 오픈소스 프로젝트와 이들의 개방 정신은 소프트웨어 개발 정신과 철학에도 지대한 영향을 미치고 있다. 모든 것을 처음부터 만들어내는 접근 방식을 채택하는 것보다 컴포넌트의 재사용과 ERP 애플리케이션을 개발하는 것과 같이 효율적인 소프트웨어 개발을 중요시하는 시각의 변화도 이러한 긍정적인 환경 요소이다.

그에 비해 날이 갈수로 레거시 애플리케이션과 데이터베이스가 늘어가고 ERP 애플리케이션과 같이 수정이 어려운 것들이 많아진다. 또한 다양한 미들웨어가 이용되는 최근의 환경은 다양한 요구사항을 수용하는 엔터프라이즈 아키텍처를 구성하는데 많은 장애가 되고 있다. 그렇지만 이런 장애는 바꾸어 보면 MDA의 성공을 촉진하는 계기가 될 수도 있다. 중요한 것은 MDA가 활약해야 할 환경의 조성은 확실히 되고 있다는 점이다. 그런 측면에서 현재의 환경 변화는 MDA의 성공에 긍정적이라고 할 수 있겠다.

시장의 반응
CORBA가 절반의 실패를 하게 된 가장 큰 이유는 무엇이었을까? 물론 기술적으로 여러 가지 문제를 지적하는 사람들도 많겠지만 필자는 시장의 반응에서 그 답을 찾고 싶다. OMG에서 CORBA를 처음 발표한 이후 실제 이 표준안 시장에 적용하여 내놓은 제품은 전무에 가까웠다. 기껏해야 아일랜드의 작은 벤처기업인 아이오나(IONA)가 오빅스(Orbix)를 통해 CORBA 시장에 뛰어들었고, 뒤를 이어 인도의 비지제닉(후에 볼랜드에 합병된다)이 비지브로커(Visibroker)로 승부를 해온 정도이다.

이들 회사는 마이크로소프트와 같은 거대 회사가 추진하는 COM이라는 강력한 적수를 상대로 나름대로 선전했지만 CORBA가 가지고 있었던 거대한 목표를 달성하는 데에는 실패했다. CORBA의 실패는 표준안이 만들어지는 것이 성공의 요소가 아니라 시장 주도 세력과의 타협을 이루는 것이 더 중요하다는 것을 보여주는 매우 중요한 사례이다.

그렇다면 MDA는 어떠한가? 기본적으로 MDA에 대한 시장은 객체지향 모델링과 개발 툴 벤더를 통해 활성화가 될 것이다. 이들은 MDA를 자신들이 제공하는 톨에 통합할 것이며 기존에 OMG에서 만든 표준안인 UML, MOF, CWM, XMI 등을 지원하는 툴들이 늘어나면서 자연스럽게 그 영역을 확장하고 있다.

 

이런 움직임은 이미 가장 커다란 벤더들을 통해 주도되고 있으며 여기에 반기를 들고 상대하는 벤더가 없는 상태이다. 쉽게 말하면 초기 장수들의 기세에 무혈 입성하는 분위기이다. 특집 3부에서 다뤄질 각 벤더들의 MDA 지원에 대한 기사를 읽어보면 이러한 움직임을 확실하게 느낄 수 있을 것이다. MDA의 성공 여부에 대해 의문을 가지고 있는 많은 독자들에게 필자는 과감하게 이야기하고 싶다. MDA는 기술의 우위성을 떠나 확실히 성공할 수밖에 없는 단계에 도달하고 있다는 것을… @

:
Posted by 뽀기

정지훈


객체지향 모델링 언어의 표준인 UML과 분산객체기술 표준인 CORBA, 그리고 그 밖에도 MOF, XMI, CWM 등의 표준안을 발표했던 OMG가 이번에는 MDA로 최근 소프트웨어 개발 방법 패러다임에 새로운 변화를 시도하고 있다. 여기서는 소프트웨어 개발 방법론과 OMG에서 MDA가 나오기까지의 역사적인 흐름과 숨겨진 뒷이야기를 알아보고 MDA의 기본적인 구성과 앞으로의 전망을 논의해 본다.

우리가 현재 살고 있는 산업사회에 있어서 중요한 것은 무엇인가? 혹자는 믿음이라고 답할 것이고, 혹자는 밥이라고 답할 것이며, 어떤 사람은 그저 애니메이션이나 실컷 보면서 살 수 있으면 그만이라고 말할 것이다. 무엇이 되었든 개인에게 있어서 틀린 답은 아니겠으나 산업사회의 발전을 위해서 가장 중요한 사회학적인 요인을 하나만 선택하라고 하면 그것은 곧 ‘생산성(productivity)’이 될 것이다. 소위 우리가 말하는 ‘산업혁명’이라는 것도 생산성의 폭발적인 증대로 귀결될 수 있다.
소프트웨어를 개발하는 입장에서도 이 생산성의 문제는 무척이나 중요하다. 같은 수준의 완성품을 만드는 데 있어서 가능하면 적은 시간과 적은 노동력과 원자재만을 이용하는 방법을 연구하는 것은 그런 측면에서 커다란 의미를 가지는 것이다. 이번 호 특집에서 언급하는 MDA(Model Driven Architecture) 역시 소프트웨어의 생산성과 관련해서 중요한 의미를 가진다.

소프트웨어의 생산성은 어떻게 높아지는가

그렇다면 과연 어떻게 소프트웨어의 생산성을 높일 수 있는가? 이 문제는 비단 최근에 제기되는 문제가 아니다. 물론 과거에 비해 복잡한 시스템 요구사항과 수많은 기술들이 등장하고 있는 근래에 그 중요성이 높아지고는 있으나 이미 소프트웨어 초창기부터 이 문제는 많은 사람들의 고민거리였다. 이 문제를 해결하기 위해서 등장한 대표적인 학문이 바로 소프트웨어 공학이다.
소프트웨 어 공학의 영역은 상당히 포괄적이다. 소프트웨어 개발에 투여되는 수많은 자원들의 효과적인 관리 방법과 평가, 그리고 조직과 의사결정 구조, 심지어는 개발자들의 심리상태에 대한 것도 대상이 될 수 있다. 소프트웨어 공학에 대해서 자세한 이야기를 하자면 소프트웨어의 위기에서부터 많은 방법론에 대한 것까지 커다란 연구 분야라고 할 수 있으며, 그 중요성은 복잡해진 환경에 의해 점점 더 높아가고 있다.

전통적인 방법론 vs. 창조적 프로그래밍

소프트웨어의 생산성을 높이기 위한 방법으로 그 동안 다양한 방법론과 이론이 등장했다. 이들에 대한 자세한 논의는 한 권의 책으로도 부족할 것이므로 여기서는 대표적인 몇 가지의 특징에 대해서 간단히 알아보기로 하자. 소프트웨어 공학의 정의로 보면 모든 방법론들이 소프트웨어 공학으로 설명할 수 있겠으나 최근에 각광받는 몇 가지 방법론이 기존의 방법론과는 상당한 철학적인 차이를 보여주고 있다. 여기서는 이들을 포괄적으로 ‘창조적 프로그래밍’으로 표현하고자 한다. 필자의 경우 소프트웨어 공학을 전문적으로 전공한 것은 아니므로 MDA를 설명하기 이전에 그 동안의 경험했던 것들을 바탕으로 접근하는 것이 도움이 될 것이라 생각하기 때문에 이 부분을 언급하도록 하겠다.
전통적인 소프트웨어 공학 방법론은 소프트웨어의 개발 단계를 요구, 분석, 설계, 구현, 테스트와 유지보수로 분류하고 각 단계별로 소프트웨어의 생산성에 영향을 주는 여러 요소들을 관리하고 조정해 생산성을 높이는 방법과 절차, 그리고 이를 지원하는 여러 가지 도구 등으로 구성된다. 소프트웨어 개발에 관한 접근 방식에 따라 기능 중심의 개발방법론, 구조적 개발방법론, 객체지향 개발방법론 등이 있으며 최근에 가장 각광받는 것은 객체지향 개발방법론이라고 할 수 있다.
이 러한 전통적인 소프트웨어 공학 방법론은 기본적으로 소프트웨어의 개발을 제조업에서의 생산 공정의 연장선상에서 소프트웨어라는 생산 대상의 특성을 고려하여 만들어지기 때문에 공정의 관리와 제어에 초점을 맞추게 되는 경우가 많다. 예를 들면 프로젝트의 관리를 위해 제안서 및 프로젝트 비용 산정, 프로젝트 계획과 스케줄링, 감독과 검토 과정을 거친 뒤 적합한 인력 배치 등을 하고 프로젝트 팀으로 구성된 조직은 조직 관리의 측면에서 최대한의 생산성을 갖출 수 있는 방향을 선택하도록 하는 것이다. 이런 소프트웨어 공학 방법론은 커다란 프로젝트의 위험성을 최대한 관리하면서 기존에 제조업과 사회과학 분야에서 개발된 여러 가지 형태의 도구를 응용할 수 있기 때문에 중간 규모 이상의 프로젝트에서 상당한 효과를 발휘한다. 이에 비해 창조적인 프로그래밍에 중점을 둔 방법론은 기본적으로 소프트웨어의 개발을 제조업에서의 생산 공정으로 보기보다는 개개인의 숨겨진 잠재력을 최대한 발휘해서 조합하는 것을 더욱 중요하게 생각한다. 그러므로 창조적인 프로그래밍에서는 소프트웨어 개발이란 일종의 공동 예술 작품을 완성하는 것과 비슷하다고 간주한다.
모든 방법론을 이러한 잣대로 분류하는 것은 무리이지만 최근에 각광을 받는 XP(eXtreme Programming), 애자일 소프트웨어 개발(Agile Software Development) 등의 방법론은 기존의 방법론에 비해 이러한 특성의 중요성을 많이 인정하고 있다. 필자는 개인적으로 이러한 두 가지 방법론을 모두 시험해 보았는데 어느 한 쪽의 방법론이 더 우월하다고 말하기는 어렵다. 전반적으로 프로젝트의 규모가 커지고 개발 인원의 수준의 차이가 많이 나는 구성원을 가질수록 전통적인 소프트웨어 개발 방법론이 효과적이며, 반대로 소규모 프로젝트이면서 개발 인원의 수준의 차가 적고 쌍방의 대화가 잘 이뤄질 수 있는 그룹이라면 창조적인 프로그래밍 방법론이 훨씬 효과적이었다.
전통적인 소프트웨어 공학적인 방법론은 교과서도 많이 나와 있고 방법론 자체에 대한 소개도 인터넷에서 쉽게 찾아볼 수 있다. 창조적인 프로그래밍과 관련해서는 켄트 벡(Kent Beck)이 쓴 ‘Extreme Programming Explained’와 애자일얼라이언스(AgileAlliance) 웹 사이트(http://www.aanpo.org/home)를 참고하기 바란다.

CASE vs. MDA

CASE(Computer Aided Software Engineering)는 소프트웨어 개발이나 각 과정별 작업을 컴퓨터의 도움을 받아 공학적 기술을 자동화하는 것으로 각각의 작업의 자동화에 이용되는 도구를 흔히 CASE 툴이라고 한다. 역사적으로 볼 때는 1980년대 초의 구조적 개발방법론이 80년대 후반의 CASE로 발전하게 되는데 초기의 CASE 제품들은 다양한 목적을 가지고 있었다. 대표적인 것으로는 모델링 작업을 도와주거나 데이터베이스를 생성하는 것에서부터 코드의 자동생성 및 역공학 툴, 그리고 프로그래머가 아닌 사람이 프로그래밍할 수 있도록 도와주는 GUI 툴 등이 있다. 초창기 CASE 제품들이 등장하던 시기에는 메인프레임에 대한 COBOL 프로그래밍과 관계형 데이터베이스에 대한 SQL CASE 툴이 주를 이뤘는데, 운영체제나 데이터베이스 아키텍처를 지원하기 위한 정보를 모아두는 공통적인 저장소(repository)가 존재하지 않아 그 활용이 제한적이다. 이를 극복하기 위해서 IBM에서는 AD 사이클/저장소(Cycle/Repository)를 만드는 프로젝트를 시도했지만 실패로 끝났다.
1990년대에 들어서면서 컴퓨팅 환경이 급속도로 메인프레임에서 유닉스, PC 기반의 운영체제로 그리고 컴퓨터 언어의 측면에서는 C와 베이직 같은 언어가 주류 언어로 자리를 잡게 되면서 기존의 CASE 툴의 활용도가 급속히 떨어지는 현상을 보여주게 된다. 1990년대 후반에 들어가면서 객체지향 프로그래밍 언어의 사용이 일반화되고 1997년에 OMG(Object Management Group)에서 UML (Unified Modeling Language)을 표준화한 것을 계기로 객체지향 모델링을 이용한 사례가 늘어나면서 기존의 CASE보다는 객체지향 개발방법론을 따르는 새로운 형태의 비주얼 툴들이 각광을 받게 되었다. 대표적인 제품이 현재는 IBM에 합병된 래쇼날(Rational)의 Rose나 볼랜드에 합병된 투게더(Together) 등이다.
CASE 툴의 경우 아직까지 성공과 실패를 논하기에는 이르지만 초기에 가졌던 거창한 목표 달성에는 이르지 못했다고 판단된다. 가장 큰 이유는 CASE가 지향하는 목표가 너무나 방대하고 이를 지원하는 툴들이 실제로 등장해서 소프트웨어 개발에 이용되기에는 실무 소프트웨어 개발 프로세스가 개발방법론이라는 것을 모두 소화할 만큼 충분히 성숙하지 않았다는 것을 들 수 있다. 다시 말해, 이론은 좋았지만 현실세계에 받아들여지기에는 아직 먼 이상향이라고나 할까?
이러한 현상은 비단 CASE 툴에 국한된 것만은 아니다. OMG에서 최초로 표준화 작업을 했던 분산객체 표준인 CORBA(Common Object Request Broker Architecture) 역시 초기의 기대와는 다르게 많은 영역에서 넓게 채택되지 못했다. 가장 커다란 요인으로 생각되는 것 중의 하나가 지나치게 무겁고 복잡한 표준 규격으로 인해 실제 소프트웨어 개발에 있어서 쉽게 받아들여지기 어려웠다는 점이다. 그에 비해 최근의 웹 서비스가 그 자체가 가지고 있는 많은 약점에도 불구하고 비교적 쉽게 채택되고 있는 것은 가벼운 표준 규격으로 인해 자유로운 선택의 영역을 넓혔으며 여러 가지 지원 도구들을 통해 쉽게 구현이 가능했기 때문이다.
MDA는 이러한 CASE와 CORBA에서의 교훈을 바탕으로 CASE처럼 지나치게 야심차지 않으면서 좀더 느슨하게 연관되어 있는 표준안들의 연결고리를 제공하는 방식으로 접근하고 있다. 또한 이러한 연결고리를 쉽게 구성할 수 있는 도구들을 적시에 제공함으로서 그 성공 가능성을 높이고 있다.

MDA 탄생 설화 - CORBA, UML, MOF 그리고 MDA

OMG는 1989년에 설립이 된 객체 기술에 대한 사용을 증진시키고 이에 대한 표준안을 만들기 위해 소프트웨어 산업계에서 구성한 컨소시엄이다. 이들이 모여서 최초로 만든 것이 바로 OMA(Object Manage ment Architecture)인데, OMA에서 가장 중요한 역할을 하는 것이 바로 CORBA이다. CORBA는 다양한 프로그래밍 언어 간의 상호운용성을 위해 IDL(Interface Definition Language)을 도입했다. 플랫폼에 독립적인 프로토콜을 위해 IIOP(Internet Inter-Operable Protocol)와 같은 프로토콜 표준안을 제시했다. <그림 1>은 OMA를 도식화해서 나타낸 것이다.



그 리고 OMG에서는 객체지향 소프트웨어 디자인을 위해서 그동안 다양하게 제시되었던 모델링 시스템을 표준화해 UML을 1997년에 발표하게 된다. 그 밖에도 OMG에서 발표된 대표적인 표준안으로는 플랫폼에 독립적으로 메타 데이터와 데이터를 정의, 조작, 통합할 수 있는 모델 기반의 프레임워크인 MOF(Meta-Object Facility), MOF를 모델 기반의 XML 통합 프레임워크로 재구성한다. MOF 기반의 메타모델의 XML 맵핑을 지원하는 XMI(XML Metadata Inter change), 데이터 웨어하우스 도구, 웨어하우스 플랫폼 그리고 서로 다른 환경에 분산된 웨어하우스 메타 데이터 저장소들 사이의 비즈니스 메타 데이터의 상호교환을 가능하게 하는 표준 인터페이스인 CWM(Common Warehouse Metamodel) 등이 있다. 이들은 MDA를 구성한 하부 요소로서 그 역할을 하게 된다.

CORBA에서 UML로의 중심 이동

필자는 2000년부터 수년 간 OMG 총회를 참석하면서 매번 총회에 참석할 때마다 조금씩 변화해가는 분위기를 느낄 수 있었다. 2000년 초기의 OMG 총회만 하더라도 대부분의 업체들은 총회의 세부 모임에 있어서 CORBA를 중심으로 CORBA 표준 규격안을 발전시키는 데 주력하였다. 또한 CORBA를 분산객체 기술의 산업 표준안으로 발표했음에도 불구하고 이와 유사한 다양한 플랫폼들이 등장하였기 때문에 이들과의 상호운용성을 위한 작업들도 지속적으로 진행되었다. 대표적인 것이 CORBA/COM, CORBA/J2EE의 상호운용성을 확보하는 것이었다. 그리고 비교적 CORBA 표준안이 많이 채택되었던 통신업체와 국방 관련 업체들을 중심으로 해당 도메인 영역에서의 서비스(service)와 퍼실리티(facility)를 정의하는 작업들이 한창이었다. 그런데 해가 바뀌면서 OMG 총회에 참석하는 사람들의 움직임이 지속적으로 UML쪽으로 움직이기 시작했다. 그에 비해 CORBA 표준안과 관련한 것들은 특정 도메인 영역으로 축소가 되는 양상을 보이면서 OMG의 중심이 CORBA에서 UML로 넘어가기 시작했다.
이러한 변화는 기술 표준안의 우월성에서 기인한 것이 아니라 여타 다른 경쟁 기술들의 존재로 인해 더 이상 산업계 표준으로서의 추진동력을 점차 상실하고 있었던 CORBA에 비해 객체지향 프로그래밍의 폭발적인 증가와 경쟁 표준안이 없는 상태에서 사실상의 유일한 표준안으로서 그 위세를 떨쳐나가던 UML 시장의 반응에 의한 것이다. UML은 기존의 플랫폼의 우월성이나 벤더들 사이의 제품 팔아먹기 경쟁에서 옆으로 비켜난 상태로 많은 개발자들에게 받아들여지면서 자연스럽게 OMG를 이끌어가는 핵심 표준안으로 각광받게 된 것이다.

MDA의 탄생과 그 이후

이때부터 OMG 내부에서는 MDA에 대한 구상을 의논하는 그룹들이 생겨나기 시작했다. UML의 시장주도적인 힘을 바탕으로 CORBA가 이뤄내지 못했던 이기종 플랫폼 및 언어독립성을 비교적 받아들이기 쉬운 방법으로 제시하려는 노력들이 모여서 MDA가 탄생하게 되었다. MDA는 2001년 3월 OMG의 CEO인 리차드 솔리(Richard M. Soley)가 ‘OMG Model Driven Architecture’라는 제목의 프리젠테이션을 통해 공식적으로는 처음 세상에 알려지게 되었다. 이 발표는 OMG 웹 사이트에서 현재도 10분 정도의 오디오 파일로 들을 수 있으므로 관심 있는 독자는 한 번쯤 들어보기 바란다. 이 발표를 시작으로 MDA에 대한 구체적인 규격을 논의하기 시작해서 2003년 6월에 공식적인 ‘MDA Guide v 1.0’이 공개되었다.
2001년 3월 MDA에 대한 발표가 있은 뒤부터 MDA 표준안을 실체화하고 MDA에서 내세운 개념들을 구체화하기 위한 움직임들은 점점 활성화되었는데, 그 이후의 OMG 총회에서는 이러한 논의들이 하나씩 소기의 성과를 거두게 된다. 필자는 처음 MDA를 발표하는 자리에 있었는데, 그 발표를 보는 순간 이것이 앞으로 갈 길은 멀어 보이지만 언젠가 세상에서 커다란 반향을 끌어낼 수 있을 것이라는 느낌을 받았다. 그러나 그 때만 하더라도 적어도 5년 안에는 저기서 이야기하는 것이 실체화하기는 어려울 것이라고 생각했었다.
이런 필자의 생각을 깨뜨린 것이 2001년 여름 OMG 총회에서 최초의 MDA를 지원하는 개발 툴로 소개된 아크스타일러(ArcStyler)였다. 당시의 제품은 UML 모델을 바탕으로 플랫폼에 독립적인 모델(PIM, Platform Independent Model)과 플랫폼 의존적인 모델(PSM, Platform Specific Model)을 따로 관리하는 기본적인 MDA의 형태를 갖추기는 했으나 실제로 지원하는 플랫폼은 J2EE 밖에 없는 것이었다. 그렇지만 MDA에 대한 기본적인 개념이나 구현 가능성을 실제로 보여줬기 때문에 당시에 커다란 반향을 불러일으켰다. 아크스타일러는 현재 볼랜드의 엔터프라이즈 스튜디오에 통합되어 있으며 이를 시발점으로 여러 MDA 지원 개발 툴들이 등장하기 시작했다.
MDA 의 목표는 소프트웨어 분석가와 개발자가 소프트웨어와 비즈니스 자산을 기술하는데 이용할 수 있는 엔터프라이즈 아키텍처 모델링이 가능하도록 하는 것이다. 이러한 아키텍처를 소프트웨어 도구를 이용해서 만들어냄으로서 실제 소프트웨어 개발에 있어서 아키텍처를 구현하는 특정 애플리케이션을 쉽게 작성할 수 있게 되며 애플리케이션을 다양한 변경 요구에 맞게 쉽게 변경할 수 있을 것이다.

MDA 구성 요소

MDA는 여러 가지 모델과 표준들에 의해서 구성된다. 모든 MDA 모델 들은 MOF라는 추상적인 메타모델에 기초를 두고 있기 때문에 모두 연관되어 있다. 다시 말해 모든 MDA 모델은 MOF와 호환성이 있다. 이런 특징은 MDA에서 이용된 모든 모델들이 다른 MOF 호환 모델들과 호환성을 가진다는 것을 보장한다.
MDA에서 또 한 가지 중요한 부분이 UML 프로파일(profile)이다. UML 프로파일은 UML을 확장한 것이기 때문에 MOF와 호환성이 있으며, UML 2.0에서는 UML의 다양한 기능적인 사용 방법을 기술할 수 있다. UML 프로파일은 UML의 확장인 동시에 그 자체로 MOF 메타모델이다. 일부의 MDA 메타모델은 이미 OMG에서 과거에 정의된 것들이며 일부는 OMG 외부의 그룹 또는 벤더들에 의해 MOF 호환이 되는 형태로 만들어진 것이다. 이러한 OMG 외부의 메타모델들은 비록 공식적인 OMG 표준은 아니지만 MOF 호환성이 있기 때문에 MDA 개발 과정에는 사용되는 데 문제가 없다. 대표적인 MDA 모델과 같은 것들이 있다.

CWM

CWM(Common Warehouse Metamodel)은 OMG에서 표준화한 데이터 웨어하우스를 관리하는 데 이용되는 메타 데이터 모델이다. CWM을 이용하면 개발자들이 관계형 데이터베이스 테이블, 레코드, 구조체, OLAP, XML 그리고 다차원 데이터베이스 디자인 등과 같은 수많은 데이터 모델 또는 포맷을 생성할 수 있다. 또한 CWM에는 데이터 웨어하우스 이외에도 사용될 수 있는 유용한 부분들도 있는데 예를 들면 데이터 모델이나 데이터 변환, 소프트웨어 배포 등에 관련된 내용도 포함되어 있다.

UML 메타모델

UML의 초기 버전은 MOF와의 완전한 호환성이 보장되지 않지만 UML 2.0은 MOF 호환성을 가진다. 이런 이유로 진정한 MDA 개발을 위해서는 UML 2.0이 사용돼야 한다. UML은 핵심 모델링 개념들과 이들을 위한 다이어그램들로 정의된다. 또한 개발자들이 UML 구성 요소에 다양한 제한(constraint)을 할 수 있도록 허용하고 있다. UML 2.0에서는 UML이 모델들의 행위를 지정할 수 있으며 이러한 행위의 표현들이 직접 코드로 변경된다.

XMI

XMI 표준은 MOF와 호환성이 있는 모델들을 XML 문서 형태로 표현하고 이를 MOF 호환 데이터베이스에 저장할 수 있도록 하는 표준이다. 그러므로 사실상 XMI 문서는 곧 MOF XML 문서라고 할 수 있다.

UML 프로파일

UML 프로파일은 특정 도메인에 대한 UML 모델을 작성하기 위한 일반 확장(generic extension) 메커니즘을 정의하는 것이다. 그러므로 UML 프로파일은 추가적인 스테레오타입(stereotype)과 태그 값(tagged value)이 적용된 엘리먼트(element), 애트리뷰트(attribute), 메쏘드(method), 링크(link) 등으로 이뤄진다. UML 프로파일은 이러한 확장 컬렉션을 이용해서 특정 도메인에 대한 모델링 문제를 기술하고 해당 도메인의 내용들을 모델링할 수 있도록 해준다. 현재 다양한 UML 프로파일이 나와 있으나 MDA와 관련해서는 플랫폼에 따라 각각의 프로파일이 정의될 수 있다. 현재 CORBA 프로파일, EAI 프로파일, EDOC 프로파일, 리얼타임 컴퓨팅을 위한 스케줄링 프로파일 등이 OMG에서 정의되었으며 EJB 프로파일은 JCP(Java Community Process)를 통해, 닷넷 프로파일은 마이크로소프트에 의해 정의되고 있다.

MDA 개발 프로세스와 특징

MDA는 이와 같이 그동안 발표된 여러 OMG의 표준 기술들이 하나로 접목되고 유기적으로 연결된 형태를 가지고 있다. 그 중에서도 MOF와 UML 프로파일의 역할이 가장 중요하다는 것을 아마도 독자도 느낄 수 있을 것이다. 이러한 구성상의 역할과는 별도로 MDA는 소프트웨어 리소스의 관리와 개발 자체에도 초점을 맞추고 있기 때문에 소프트웨어 개발 프로세스에서 모델들이 어떻게 이용돼야 하는지에 대해서도 기술하고 있다.
<그림 2>는 비즈니스 프로세스나 소프트웨어에 대한 기술서를 바탕으로 추상적인 모델을 추출하고, 추상적인 모델을 실행 가능한 구현 모델로 변환시키는 과정을 보여준다. 이런 프로세스에서 사용된 모델들은 UML과 같은 메타모델로 표현되는데 일부의 모델은 플랫폼과 무관하고 일부는 플랫폼과 밀접한 관계를 가진다. <그림 2>에서 3가지 유형의 MDA 모델을 볼 수 있을 것이다. 일반적으로 비즈니스를 분석하고 요구사항을 추출하는 역할을 맡은 분석가가 CIM(Compu tation Independent Model)을 모델링하며, 비즈니스 아키텍트나 디자이너가 PIM(Platform Independent Model)을 만든다. PIM 모델은 특정한 구현 모델과는 독립적으로 어떤 형태의 아키텍처를 가질 것인지 기술한다. 그리고 나서 개발자와 테스터가 소프트웨어 툴을 이용해서 PSM을 PIM에서 뽑아낸 뒤에 이를 플랫폼 특성에 맞게 최적화를 하고 코드를 뽑아내게 된다. MDA 개발 프로세스를 단계별 스텝으로 정리하면 다음과 같다.

1. 애플리케이션에 대한 비즈니스 요구사항을 수집한다.

2. 도메인 모델에 대한 UML 다이어그램을 작성한다. 먼저 J2EE, 닷넷, CORBA와 같은 특정 기술에 종속적이지 않은 모델을 먼저 만든다. 이렇게 작성된 UML 모델은 핵심 비즈니스 서비스와 컴포넌트를 나타내게 되는데, 예를 들면 가격 결정 엔진이라든지 쇼핑 카트 모델이라든지 주문 모델 같은 것이 될 것이다. 이러한 UML 모델을 PIM이라고 한다.

3. 애플리케이션에 대한 특정 기술과 관련한 UML 모델을 작성한다. 이와 같이 특정 기술에 종속적인 UML 모델을 PSM이라고 한다. PSM은 직접 작성할 수도 있고 MDA 지원 툴을 이용해서 PIM에서 자동으로 만들어내거나 또는 만들어진 모델을 수정하는 방식으로 작업할 수 있을 것이다.

4. 마지막으로 MDA 툴을 이용해서 애플리케이션 코드를 만들어낸다. 현재 이미 J2EE의 경우에는 MDA 툴들이 대부분의 서블릿, JSP, EJB와 관련한 코드를 자동을 생성해준다. 그 다음에 만들어진 코드를 바탕으로 수정과 최적화 과정을 거쳐서 프로젝트를 완성한다.



이 와 같이 MDA에서는 결국 UML 모델에서 코드를 만들어낸다. 그런데 어찌보면 이것이 새삼스러운 것은 아니다. 아마도 많은 개발자들이 래쇼날 로즈를 이용해서 자바 클래스를 UML 모델에서 만들어낸 경험이 있을 것이다. 이러한 작업과 MDA와의 가장 큰 차이점이라면 MDA는 플랫폼 독립적인 모델에서 시작해서 다양한 플랫폼을 지원하는 프로세스와 툴을 거의 완벽하게 지원한다는 점일 것이다. 이런 측면에서 간단히 MDA에서 만들어지는 모델과 코드에 대해 정리하면 다음과 같다.

◆ MDA는 다른 디자인 프로세스보다 고수준의 추상화를 먼저 시작한다. 예를 들어 PIM은 매우 추상적이다. 여기에는 엔티티와 서비스만 정의되어 있을 뿐이다.

◆ PSM은 메타 데이터의 형태로 애플리케이션을 완전하게 기술한다. PSM 수준에서 개발자들은 해당 기술과 관련된 디자인을 직접 코드를 건드리지 않고 향상시킬 수 있다.

◆ PSM에서 만들어진 코드는 완성된 애플리케이션에 근접하게 된다. 기존에 많은 도구들이 자동으로 만들어낸 코드들이 애플리케이션의 일부에 해당하는 것에 비해서 그 범위와 정도에 큰 차이가 있다.

◆ PIM에서 PSM을 만들어내고 PSM에서 코드를 만들어내는 알고리즘은 기본적인 것은 제공되지만 아키텍트가 변경하거나 재정의할 수 있다.

MDA의 중요한 특징 중의 하나가 이와 같이 모델링 언어를 프로그래밍 언어처럼 이용하는 것이다. 모델링 언어는 일반적으로 자바나 C++와 같은 프로그래밍 언어에서 하는 것보다 더 높은 수준에서 ‘추상화(abstraction)’를 가능하게 하며, 이러한 추상화 프로세스를 통해 개발자들은 시스템 코드에 꼭 필요한 데이터 이외의 것들은 모두 숨길 수가 있다. 이런 과정은 개발 과정에서의 복잡도를 감소시키는 효과를 가져오기 때문에 개발의 생산성을 높일 수가 있다.
아마도 여기까지 읽은 독자는 일반적인 객체지향 프로그래밍 언어의 가장 큰 특징 중의 하나인 ‘캡슐화(encapsulation)’를 떠올린 경우가 많다. 보통 객체지향 프로그래밍 언어의 3대 특징을 이야기하면 캡슐화, 상속성(inheritance) 그리고 다형성(polymorphism)을 언급한다. 그 중에서도 객체지향이라는 개념을 위해서 가장 중요한 것은 바로 캡슐화이다. 다른 두 가지 특징은 완전히 지원하지 않더라도 객체지향 언어라고 말할 수 있지만 캡슐화가 지원되지 않으면 객체지향이라고 말할 수가 없다. MDA는 이러한 객체지향 개념의 캡슐화를 극대화한 것으로 이해할 수 있다. 단순히 프로그래밍 언어 수준에서의 캡슐화가 아닌 모델링을 포함해서 구현에 이르는 과정을 단계별로 캡슐화한 것이 바로 MDA의 정체이다.
MDA 개발 프로세스에서의 이러한 추상화는 시스템의 규격을 하부의 컴퓨팅 환경에 덜 좌우되게 하기 때문에 시스템의 생산성을 높일 뿐 아니라 지속성도 높여준다. 이와 같은 추상화의 수준을 높임으로서 소프트웨어 개발 생산성을 높이려는 시도는 새삼스러운 것은 아니다. 가장 직접적인 예로 현재 우리가 사용하고 있는 대부분의 프로그래밍 언어는 3세대 언어(3GL, 3rd Generation Language)라고 할 수 있는데, 3GL 언어는 기본적으로 어셈블리 언어를 추상화한 것이라고 할 수 있다. 이를 통해 어셈블리 언어로 프로그래밍할 때 알아야 했던 많은 어려운 문제들을 간단하게 컴파일러에게 맡겨버린 뒤 소프트웨어의 생산성이 급속도로 높아졌다는 것은 모두가 인정할 것이다. 그러나 3GL이 처음 등장했을 때만 해도 여러 개발자들은 3GL이 개발자들이 코드를 작성하기 쉽게 해주고 높은 수준의 생산성을 보장하지만 만들어진 실행 프로그램의 수행 속도 문제 때문에 한동안 수많은 논쟁을 했었다. 이러한 문제는 하드웨어의 비약적인 성능 발전으로 인해 자연스럽게 작은 문제로 치부되기 시작했고 이를 통한 소프트웨어 생산성의 증가는 많은 것을 가능하게 하였다.

왜 MDA인가

지금까지 설명한 내용을 충분히 이해한 독자라면 아마도 MDA가 어떤 녀석인지 정도는 감을 잡았을 것이다. 그렇다면 왜 MDA가 최근에 각광받게 되었는지에 대해서 조금 더 생각해 보자. MDA는 MDA 표준을 적용해서 모델의 자동화와 변환(transfor mation)을 통해 여러 플랫폼을 쉽게 지원하고 개발자의 입장에서는 시간을 많이 잡아먹는 코드 작성 부분을 줄일 수 있으며 개발 프로세스의 측면에서도 품질 관리(QA, Quality Assurance)를 수월하게 할 수 있다. 그리고 과거의 CASE나 4GL과는 다르게 구현시에 유연한 컨트롤이나 커스텀화가 가능케 했다. 손으로 코딩하는 부분이 줄고 동시에 빨리 버그를 잡을 수 있다면 소프트웨어의 생산성은 월등히 증가하게 된다. MDA를 이용하는 것이 좋은 이유를 몇 가지 나열해서 설명하면 다음과 같다.

기술 플랫폼과 기능 변화에 신속한 대응

MDA에서는 PIM과 PSM을 분리했기 때문에 PIM을 변경하지 않고도 기술 플랫폼의 변화나 요구사항 변화에 발빠르게 대처할 수 있다. 예를 들어, 새로운 플랫폼에 대한 애플리케이션을 배포해야 하는 경우 PIM에는 수정이 필요 없으므로 단지 PIM을 이용해서 새로운 플랫폼에 대한 PSM을 자동으로 만들어내고, 이를 수정해서 코드를 다시 생성하는 과정을 통해 쉽게 새로운 기술 플랫폼에 대한 대응이 가능하다. 또한 기능의 변화를 요구하는 경우에도 PSM 수준에서의 변경을 고려하지 않고 PIM에 새로운 기능 변화와 관련된 수정을 가함으로써 쉽게 대응할 수 있다.

시스템의 지속성

플랫폼 의존적인 시스템은 기존의 아키텍처가 새로운 요구사항을 만족시키지 못하는 경우가 많이 발생한다. 이 경우에 완전히 새로운 아키텍처를 다시 잡기보다는 많은 사용자들은 작은 패치와 수정으로 이런 문제를 해결하기를 원하며 현실적으로도 그렇게 할 수밖에 없는 시간·경제적 여유만을 가지는 경우가 많다. MDA를 이용하면 기능과 아키텍처를 분리해서 정의하기 때문에 아키텍처의 변화가 있더라도 변환 과정을 거쳐 구현 과정으로 쉽게 진행할 수 있다. 따라서 기능과 요구사항 변화에 의한 아키텍처 변경에 비교적 자유롭다. 이런 장점은 시스템의 생명주기를 연장시키며 더 안정된 시스템 유지를 가능하게 한다.

개발 생산성 증진

MDA는 모델의 자동화와 변환을 통해 여러 플랫폼을 쉽게 지원하고 시간을 많이 잡아먹는 코드 작성 부분을 줄일 수 있으며 쉽게 유지보수가 되도록 한다. 이와 같이 손으로 코딩하는 부분이 줄고 동시에 빨리 버그를 잡을 수 있다. 또한 한 번 작성된 PIM의 경우 비즈니스 핵심 부분에 대한 모델이기 때문에 향후 다른 시스템에서도 쉽게 이용될 수 있으며 재사용성이 높아지게 된다.

용이한 문서 작성

디자인 문서를 업데이트하고 관련 코드를 수작업으로 관리하는 것은 매우 시간이 많이 걸리고 귀찮은 작업이다. MDA에서는 모델과 코드, 문서가 항상 동기화되기 때문에 이러한 문서 작업에 필요한 일의 양을 줄여준다.

품질 관리 비용의 감소

개발 과정에서 소프트웨어의 문제가 늦게 발견될수록 이를 고치는 데 들어가는 비용은 증가한다. MDA 모델 자동화와 테스트 툴을 이용하면 개발자들이 애플리케이션을 모델 수준에서 테스트할 수 있기 때문에 디자인의 문제점이나 애플리케이션 로직의 에러를 빨리 잡을 수 있다. 이를 통해 품질 관리에 들어가는 비용도 감소한다.

양질의 시스템 구축

PIM의 단순함은 양질의 시스템 구축을 가능하게 한다. 모델링은 팀 멤버들 사이의 의사소통을 원활하게 만들고 동시에 결점이 있을 때 이를 빨리 제거할 수 있도록 도와준다. 또한 MDA의 자동화 도구들은 잘 정리된 코딩 패턴을 모델에 적용하기 때문에 손으로 직접 작성한 코드에 비해 결점이 적을 가능성이 많다.

기술 플랫폼 통합의 기로에 서서

OMG는 소프트웨어 개발에 있어서 운영체제와 프로그래밍 언어의 통합이라는 어려운 과제를 수행하기 위해 과거 CORBA를 통해 불철주야 노력했으나 현 시점에서 판단해 보면 그다지 성공적이라고 하기는 어렵다. 그렇지만 CORBA를 논의하면서 모였던 많은 지식과 경험들이 바탕이 되어 MDA라는 새로운 접근 방법을 합의하는 데 성공했으므로 그런 노력들이 헛되기만 한 것은 아니었다. 마지막으로 MDA가 기술 플랫폼 통합의 기로에서 앞으로 어떤 길을 걷게 될 것인지 살펴보자.

환경의 변화

최근의 소프트웨어 개발 환경은 과거 CORBA를 만들어낼 때보다도 더 많은 프로그래밍 언어와 기술 플랫폼들이 등장하고 있다. 그리고 소프트웨어의 복잡도도 나날이 증가하고 있으며 이런 복잡한 소프트웨어를 더 효과적으로 개발하는 것이 가장 커다란 도전이 되고 있다. 그렇지만 과거의 환경에 비해 좋은 점들도 있다. 그것은 실질적인 표준으로 받아들여지는 기술이 많이 등장하고 있다는 것이다. 네트워크 프로토콜에 있어서는 TCP/IP, 데이터베이스 질의와 관련한 SQL, 객체지향 모델링에 대한 UML 등은 이제 진정한 표준으로서의 입지를 굳혀가고 있다. 또한 리눅스에서 출발하고 아파치를 통해 더욱 널리 퍼져가고 있는 오픈소스 프로젝트와 이들의 개방 정신은 소프트웨어 개발 정신과 철학에도 지대한 영향을 미치고 있다. 모든 것을 처음부터 만들어내는 접근 방식을 채택하는 것보다 컴포넌트의 재사용과 ERP 애플리케이션을 개발하는 것과 같이 효율적인 소프트웨어 개발을 중요시하는 시각의 변화도 이러한 긍정적인 환경 요소이다.
그에 비해 날이 갈수로 레거시 애플리케이션과 데이터베이스가 늘어가고 ERP 애플리케이션과 같이 수정이 어려운 것들이 많아진다. 또한 다양한 미들웨어가 이용되는 최근의 환경은 다양한 요구사항을 수용하는 엔터프라이즈 아키텍처를 구성하는데 많은 장애가 되고 있다. 그렇지만 이런 장애는 바꾸어 보면 MDA의 성공을 촉진하는 계기가 될 수도 있다. 중요한 것은 MDA가 활약해야 할 환경의 조성은 확실히 되고 있다는 점이다. 그런 측면에서 현재의 환경 변화는 MDA의 성공에 긍정적이라고 할 수 있겠다.

시장의 반응

CORBA가 절반의 실패를 하게 된 가장 큰 이유는 무엇이었을까? 물론 기술적으로 여러 가지 문제를 지적하는 사람들도 많겠지만 필자는 시장의 반응에서 그 답을 찾고 싶다. OMG에서 CORBA를 처음 발표한 이후 실제 이 표준안 시장에 적용하여 내놓은 제품은 전무에 가까웠다. 기껏해야 아일랜드의 작은 벤처기업인 아이오나(IONA)가 오빅스(Orbix)를 통해 CORBA 시장에 뛰어들었고, 뒤를 이어 인도의 비지제닉(후에 볼랜드에 합병된다)이 비지브로커(Visibroker)로 승부를 해온 정도이다. 이들 회사는 마이크로소프트와 같은 거대 회사가 추진하는 COM이라는 강력한 적수를 상대로 나름대로 선전했지만 CORBA가 가지고 있었던 거대한 목표를 달성하는 데에는 실패했다. CORBA의 실패는 표준안이 만들어지는 것이 성공의 요소가 아니라 시장 주도 세력과의 타협을 이루는 것이 더 중요하다는 것을 보여주는 중요한 사례이다.
그렇다면 MDA는 어떠한가? 기본적으로 MDA에 대한 시장은 객체지향 모델링과 개발 툴 벤더를 통해 활성화가 될 것이다. 이들은 MDA를 자신들이 제공하는 톨에 통합할 것이며 기존에 OMG에서 만든 표준안인 UML, MOF, CWM, XMI 등을 지원하는 툴들이 늘어나면서 자연스럽게 그 영역을 확장하고 있다. 이런 움직임은 이미 가장 커다란 벤더들을 통해 주도되고 있으며 여기에 반기를 들고 상대하는 벤더가 없는 상태이다. 쉽게 말하면 초기 장수들의 기세에 무혈 입성하는 분위기이다. 특집 3부에서 다뤄질 각 벤더들의 MDA 지원에 대한 기사를 읽어보면 이러한 움직임을 확실하게 느낄 수 있을 것이다. MDA의 성공 여부에 대해 의문을 가지고 있는 많은 독자들에게 필자는 과감하게 이야기하고 싶다. MDA는 기술의 우위성을 떠나 확실히 성공할 수밖에 없는 단계에 도달하고 있다는 것을…

[ 모델~ 모델 ? 모델 ! ]

모델이란 무엇인가? 최소한 3가지 의미를 생각해 볼 수 있다. 첫 번째 의미는 스케치(sketch)이다. 우리들은 흔히 새의 날개를 보고 간단하게 긴 타원형의 날개를 연필로 그리기도 하며 비행기가 이륙할 때 흐르는 공기의 흐름을 몇 줄의 선을 긋는 것으로 표현하기도 한다. 경우에 따라서는 몇 가지 스케치 사이의 관계를 한두 개의 공식으로 표현하기도 한다. 스케치는 완벽하지도 않고 그래서도 안 된다. 이러한 스케치의 목표는 아이디어를 구체화하는 것이다. 프로그래머는 그들의 생각을 클래스와 유스 케이스(Use Case)로 스케치를 한다. 이러한 스케치를 하기 좋도록 제공되는 것이 UML이다. 간혹 스토리(story)라는 표현도 있는데 본질적으로는 비슷한 거라고 볼 수 있겠다. 스케치를 좋아하는 개발자 중에서는 코드를 이용하는 사람들도 있다.
두 번째 의미는 청사진(blueprint)이다. 가장 흔한 예가 건물을 지을 때 단면도 등이다. 청사진의 특징은 실제 계획에 대한 실체화라는 점이다. 그런데, 보통 모델을 청사진으로 이야기할 때에는 건축이나 제조 공장에서 사용되는 의미가 강하기 때문에 소프트웨어 개발과 같이 창조적인 작업에는 잘 맞지 않는 측면도 있다. 그렇지만 이러한 제조 공정에서의 장점을 살려서 소프트웨어 개발의 생산성을 따지는 경우에는 이러한 청사진의 의미로 모델이 쓰이는 경우가 많다. 소프트웨어 공학에서는 모델을 이러한 의미로 이용하는 경우가 많다. 가장 대표적으로 SEI(Software Engineering Institute)의 CMM(Capability Maturity Model)을 들 수 있겠다. 그렇지만 소프트웨어를 제조 공정보다는 창조적인 작업이 우선하는 것으로 간주할 경우에는 모델을 청사진의 의미로 이용하는 것은 적절하지 않다. 왜냐하면 창조적인 작업의 경우에는 개발자들이 모델을 예측 가능한 것으로 보기보다는 변화에 대한 적응으로 봐야하는 경우가 많기 때문이다.
마지막 세 번째 의미는 실행 가능(executable)한 것이다. 예를 들면 실제 비행기에 대해 하늘을 나는 모형 비행기를 만드는 작업을 연상하면 된다. 이러한 모델링이 소프트웨어 개발에서 가능하려면 모델링 언어도 실행 가능해야 할 것이다. 그러므로 실행 가능한 UML 모델을 만들 수 있다면 마치 자바나 C++로 프로그래밍을 하듯이 시스템의 행위를 더 명확하게 기술할 수 있을 것이다. 이러한 실행 가능한 관점에서의 모델의 의미를 적용한다면 자바와 같은 고수준 언어로 만들어진 프로그램들도 모델이라고 할 수 있을 것이다. 자바 프로그램은 실행이 가능한 바이트 코드로 변환되며 이러한 과정을 모델의 변환을 통한 실행이라고 볼 수 있다. 이 경우 모델을 만드는 사람, 즉 프로그래머는 자바 컴파일러가 어떻게 동작하는지, 그리고 프로그램이 어떻게 실행될 수 있는지에 대해서는 알 필요가 없다.
앞서 언급한 XP나 애자일 소프트웨어 개발과 같은 방법론(방법론이라고 하면 반대하는 이들도 많겠으나, 이 글에서는 그냥 용어의 통일성과 비교를 가능하게 하기 위해서 그냥 방법론이라고 하겠다)에는 코드를 관리하는 방법이 나와 있는 것이 아니라 프로세스나 고객과의 관계 그리고 관리 방법에 대해 기술되어 있다. 그러나 이러한 프로세스의 원칙들은 실행 가능한 모델을 구축하거나 코드를 만들어나가는 데에도 그대로 적용할 수 있는 것이다.







소프트웨어 개발 패러다임이 MDA로 진화하고 있다. MDA는 기존의 OMA, CORBA에 대한 업그레이드 버전이 아니라 이식성, 상호운용성, 재사용성에 대한 문제를 근본적으로 해결해 보고자 하는 노력의 산물이다. 특집 2부에서는 개발자 관점에서 MDA 개발 환경을 이해하고 각 MDA 기술 요소를 살펴보며 어떻게 개발이 수행되는지 알아본다.

MDA또 새로운 이름이 나왔구나. 그런데 이것을 적용하면 정말 좋아지긴 좋아지는 것인가? 이거 장사 속 아니야? 그러면 재미있게 일할 수 있는 거야? 야근 안 해도 되나? 개발자들은 이러한 원초적인 질문을 던질 수 있다.
소 프트웨어 개발 패러다임이 점차 MDA(Model Driven Architecuture)로 진화하고 있다. 시스템 규모가 전사적으로 확대되면서 데이터 모델의 설계가 중요하다며 정보공학이라는 개발 방법론이 제시되더니만 시스템 규모가 점점 더 확대되고 통합과 유지보수에 대한 이슈가 큰 문제로 제기되자 객체지향 방법론이 그 대안으로 제시됐다. 그러나 정보공학이 안정되기 시작하는 때에 컴포넌트란 명찰을 달고 유행을 일으킨 CBD 방법론이 나왔고, 다시 새로운 개발 패러다임이라면서 MDA를 논하기 시작한다. 이것들을 자세히 들여다보면 모두 하나의 큰 흐름 속에 존재하며 진화해가고 있는 것을 발견할 수 있다.

MDA의 이해

아주 오래 전의 개발 환경을 생각해 보자. 그 시대에는 컴퓨터가 각 대학교 연구소에서만 활용됐다. 이때 연구원들은 무언가를 시뮬레이션하기 위해 프로그래밍했고, 프로그래밍 언어로서 어셈블리어 같은 저수준 언어를 사용했다. 이는 간단한 프로그램을 개발하는 데도 아주 많은 양의 코드가 필요하며, 가독성도 떨어지고 디버깅도 아주 어렵다. 어쩌면 이 시대에는 이것들이 특별한 문제점으로 인식되기보다도 ‘이 바닥이 원래 그렇지~’라고 스스로를 위안하며 어셈블리 코드를 작성하고 있었을지도 모르겠다.
이런 환경에서 코드 생산성을 증대시키는 하나의 혁명이라고 할 수 있는 것이 탄생했으니, 그것은 바로 ‘컴파일러’이다. 누구나 쉽게 프로그래밍할 수 있는 고수준의 언어가 정의되고 개발자는 이 언어를 사용해 프로그래밍하고 컴파일러를 사용해 일종의 변환(trans formation) 작업을 거치면 바로 실행할 수 있는 바이너리 모듈을 뽑아낸다. 컴파일러의 등장은 어느 정도의 수행 능력(performance)을 희생해야 했지만 개발자에게는 가히 상상도 할 수 없을 정도로 큰 생산성을 가져왔다. 여러분이 현재 개발하는 시스템을 모두 어셈블리어로 작성한다고 생각해 보라. 생각만으로도 아주 끔찍하지 않은가.
MDA도 그 이상을 그리자면 소프트웨어 개발 패러다임의 혁명이라고 할 수 있다. 이제는 소스코드 레벨의 개발 작업이 한 단계 위로 끌어올려져 개발자는 시스템의 비즈니스에만 집중해 PIM(Platform Indepen dent Model)이라는 플랫폼 독립적인 모델을 설계하고 이것을 특정 플랫폼에서 동작하는 하나의 애플리케이션으로서 뽑아내기 위한 변환 정보 정의를 기술함으로써 프로그래밍 작업을 수행한다. 이제 컴파일러 격에 대응되는 MDA 툴들은 이 정보들을 갖고 변환 작업을 수행한 후 바로 실행 가능한 하나의 거대한 시스템을 탄생시킨다.
컴파일러는 그 당시 가장 큰 문제였던 프로그래밍에 대한 문제를 고수준으로 끌어올려 코드의 생산성을 증대시켰다면 MDA는 현재 엔터프라이즈 급에서 큰 고민인 (미들웨어를 포함한) 구현 플랫폼으로의 포팅 문제, 상호운용성, 재사용성에 대한 문제를 고수준으로 끌어올려 그것을 구현하는 데에 생산성을 증대시키려 하고 있다.

과연 MDA가 뜰까?

OMG(Object Management Group)에서는 1990년대부터 분산 네트워크 상에서 서비스 가능한 객체 모델에 대해 고민하기 시작했다. 시스템 용량이 커지면서 이것들이 플랫폼에 독립적으로 수행되기 위한 객체 표준과 그 객체가 분산 시스템 환경에서 어떻게 서로 서비스를 수행할 것인지에 대한 표준의 필요성이 대두되었다. 이에 따라 세상을 개방형 객체 모델로 묶어 보자는 의미에서 개방된 표준 객체 기반의 인터페이스를 제공하는 참조 모델로서 상호운용성과 재사용 가능한 이식성 있는 소프트웨어 컴포넌트 아키텍처를 정의하는 OMA (Object Management Architecture)를 발표했고, 이 분산 객체 아키텍처를 실현하기 위한 객체 요청 브로커(ORB)에 대한 표준화 작업을 수행했으며 결국 OMG 프레임워크 모델에 대한 첫 번째 인스턴스인 CORBA가 탄생했다.
그러나 CORBA는 점차 시장에서 다른 미들웨어 제품에 밀리면서 OMG 개방형 표준으로서의 입지는 사라지게 되었고 그 와중에 MDA가 탄생했다. MDA는 기존의 OMA, CORBA에 대한 업그레이드 버전이 아니라 이식성, 상호운용성, 재사용성에 대한 문제를 근본적으로 해결해 보고자 하는 노력의 산물이다. 현재 OMG는 UML 표준, Executable UML, UML 프로파일, MOF(Meta Object Model), CWM(Common Warehouse Metamodel), XMI의 표준화 작업을 수행하고 있으며 이것들은 MDA라는 OMG의 새로운 프레임워크로 조직화되고 구현되고 있다(<그림 1>).

<그림 1> MDA 적용


OMG 표준화 단체뿐만 아니라 현재의 IT 기술 리더와 기존의 각 케이스 툴 벤더들도 MDA 툴로의 진화를 가속화하고 있다. IBM은 강력한 공개용 통합 개발 환경인 이클립스에 많은 돈을 투자하고 있고 래쇼날을 인수해 비즈니스 시스템을 구축하기 위한 핵심적인 방법론, 케이스 툴 등 많은 엔지니어링 기술 및 도구를 확보했다. 또한 엔터프라이즈 애플리케이션 플랫폼으로 웹스피어를 갖고 있다. IBM은 개발 방법론부터 구현, 배치까지 서비스 수행에 이르기까지의 소프트웨어 개발에 필요한 모든 도구들을 확보했으며, 이것들의 통합은 MDA라는 새로운 패러다임 속에서 강력한 힘을 발휘할 것이다. 볼랜드도 자사의 애플리케이션 서버에 개발 도구로 J빌더를 갖고 있고 투게더를 인수해 모델링 도구인 투게더를 통합, MDA 도구로서의 진화를 꾀하고 있다. 마이크로소프트도 비주얼 스튜디오 2005에서는 MDA 도구를 제공한다고 한다.
시장을 주도하는 기술은 그 기술의 질과 유용성도 중요하겠지만 더 중요한 것은 영향력 있는 회사가 얼마나 그 기술에 투자하느냐는 것이다. 두 가지는 모두 시장의 기술을 선도하기 위한 필수적인 것이다. MDA는 기술적으로 현재의 시장 요구를 실현해 생산성을 높일 수 있는 패러다임으로서 진화 단계에 있는 모델이며, IT 세상을 주도하는 주요 회사들 역시 MDA에 많은 투자를 하고 있다. MDA 이상이 실현되기 위해서는 좀 더 지켜봐야겠지만 이는 일단 앞에서 제시한 두 가지 조건을 모두 만족시킨다. 자본주의 관점에서 얘기하면 MDA는 기술을 주도하고 있는 여러 회사들에게 돈을 벌어다 줄 수 있는 도구이다.

시장 트렌드 속에서의 MDA

MDA에 대해 이야기하기 전에 먼저 현재의 개발 플랫폼의 트렌드를 살펴보자. 가장 큰 이슈가 되는 것은 다음의 세 가지이다.

◆ 빠른 시장 대응(time to market)

◆ 상호운용성(interoperability)

◆ 개발 생산성(productivity), 유지보수(maintenance)

오늘날의 비즈니스 환경은 아주 빠르게 변화하고 있다. 끊임없이 변화하는 비즈니스와 고객들의 요구에 빠르게 대응해야만 시장에서 살아남을 수 있다. 현재의 주요 미들웨어 벤더나 통합 개발 방법론(UP : Unified Process), 최근에 부각되고 있는 애자일 방법론(Agile Method)은 모두 변화하는 고객 요구와 시장에 효과적으로 대응하기 위해 우수한 시스템을 개발하는 것을 최종 목표로 삼고 있다. 소프트웨어 업계에서는 이 문제들을 해결하기 위해 시스템 개발시 비즈니스 로직을 담고 있는 기능적(functional) 요구사항과 비기능적인(non-functional) 부분을 분리하자는 노력이 계속되고, 시장에 빠르게 대응하기 위한 컴포넌트 모델이 제안됐다.
비즈니스의 규모가 커지면서 각 엔터프라이즈급 시스템간의 상호운용에 대한 문제가 발생하고 이 문제를 효과적으로 해결하기 위해 많은 EA, EAI 솔루션들이 나오고 있다. 앞으로는 엔터프라이즈 망과 이동통신 망도 통합될 것이고 상호운용성에 대한 문제는 더더욱 커질 것이다. 현재 이런 고민들은 ‘미들웨어’라는 고마운 도구들이 많이 해결해 주고 있다. 우수한 미들웨어들이 많이 개발되고 있음에도 불구하고 앞의 문제는 개발 프로젝트 수행과 시스템 확장 및 유지보수의 문제에 있어서 여전히 큰 위험으로 작용하고 있다. 이러한 문제들을 근본적으로 해결할 수 없을까 하는 고민에서 나오게 된 것이 MDA이다.

기존 개발 방식과 MDA 개발 방식

MDA는 아주 오래 전부터 소프트웨어 공학자들에 의해 만들어지고 정제되어 왔으며 대부분의 SI 프로젝트에서 적용되고 있는 ‘비즈니스 모델링쭻요구사항 분석쭻설계쭻구현’ 방식의 탑 다운(top down) 모델의 개념을 갖는다.
대 표적인 객체지향 방법론인 UP에서는 비즈니스 모델링을 통해 전체 비즈니스를 파악하고 유스케이스 다이어그램을 통해 요구사항을 모델링하며 이것으로 분석 모델을 만든다. 이때의 분석 모델은 비기능적 요구사항이 전혀 반영되지 않은 순수한 비즈니스를 모델링하고 있으며, 이 분석 모델을 기반으로 적절한 플랫폼과 설계 메커니즘을 적용해 플랫폼 종속적인 설계 모델을 만들고 이를 기반으로 구현한다. 요구사항 모델은 시스템을 개발자 관점으로 가져오기 전에 고객과 의사소통하기 위한 수단이고 분석 모델에서는 어떤 환경에서 어떤 제약 조건을 갖고 구현될지는 생각하지 않고 순전히 비즈니스만을 반영한 모델을 모델링한다. 그리고 설계 모델에서는 비기능적 요구사항을 반영해 세부 설계를 하고 이 모델을 이용해 구현 소스 코드를 만든다.
MDA는 기존 개발 방식과 확실히 다른 소프트웨어 개발 패러다임을 갖고 있다. 기존 방법은 분석, 설계 작업을 통해 나온 모델을 개발자가 직접 구현하는 데 반해 MDA는 개발 작업의 중심이 플랫폼 독립적인 모델인 PIM을 구축하는 것에 맞춰져 있고 이후에는 MDA 툴을 사용해 변환 작업을 하여 최종 소스코드, 애플리케이션을 만들어내는 것이다. MDA 중심에는 PIM이 있고 툴과 변환 작업이 애플리케이션을 만들어내는 데 중요한 역할을 한다.

<그림 2> 기존 개발 방식과 MDA 개발 방식


MDA 개발 프로세스와 기술

MDA는 플랫폼 독립적인 PIM(Platform Independent Model)을 구축하고 이를 플랫폼 종속적인 모델인 PSM(Platform Specific Model)으로 변환하고 최종 애플리케이션을 구축하는 프로세스를 갖는다. 이때 가장 중요한 작업 중 하나가 PIM에서 PSM 모델로, PSM 모델에서 소스코드로 변환하기 위한 변환 작업이다(<그림 4>). 그러면 이것들은 어떤 기술 요소를 갖고 있고 어떻게 개발 작업이 수행되는지 살펴보자.

<그림 3> MDA 기술


<그림 4> MDA 개발 방법


도메인 모델, CIM

CIM(Computation Independent Model)은 시스템 요구사항과 환경에 대한 것을 기술하는 모델이다. 지금까지 우리가 써오던 용어를 빌리자면 도메인 모델(domain model) 또는 비즈니스 모델(business model)이라고 할 수 있다.
CIM에서는 시스템을 어떻게 구축해야 하는가를 고민하기 이전에 시스템의 전체적인 운용 모습을 이해하고 그 속에서 시스템이 무엇을 해야 하는지를 기술하며 개발자에게 시스템의 도메인에 대한 전반적인 이해를 도와 도메인 숙련자와 개발자간의 틈을 줄이는 역할을 한다.
CIM은 시스템의 비즈니스를 설명하는 소프트웨어 독립적인 모델이기 때문에 CIM으로부터 PIM을 추출할 수는 없다. 다만 PIM을 구축할 때 CIM을 통해 요구된 비즈니스 정보는 중요한 자산이 되며 CIM으로부터 추적성을 제공한다.

플랫폼 독립적인 모델, PIM

MDA 개발 프로세스에서 개발자의 주요 관심사는 좋은 소스코드를 작성하는 것에서 좋은 PIM 모델을 구축하는 것으로 옮겨간다. 다시 말해 개발자의 가장 중요한 작업은 PIM을 구축하는 것이다.

PIM 개발
MDA 개발 프로젝트의 시작점은 PIM을 구축하면서부터이다. PIM은 이름 그대로 구현 플랫폼 환경은 고려되지 않은 순수한 플랫폼 독립적인 설계 모델이며 UML 표준 모델링 도구를 이용해 시스템을 모델링한다. PIM의 상위 모델인 컨텍스트와 요구사항은 CIM과 여러 요구사항 수집 도구 및 방법론을 사용해서 수집하고 정형화시킨다. 여기까지는 정확한 요구사항을 모델링하기 위해 고객 입장에서 요구사항을 기술하고 모델링한다.
이제 앞 모델을 개발자 관점으로 가져와 시스템 개발을 위한 플랫폼 독립적인 모델을 구축한다. 요구사항을 통해 비즈니스의 기능들을 추출해 구조화시키고 이 기능을 수행하기 위한 동적 모델을 구축한다. 이때 모델은 각 비즈니스 규칙에 따라 정확하고 명료하게 기술해야 한다. 기존의 개발 방식은 대략 비즈니스의 개념을 잡는 정도로만 모델링해도 이후에 개발자가 직접 소스코드를 작성할 것이기 때문에 큰 문제가 없었으나 MDA 개발 방식에서는 기술한 비즈니스 규칙이 그대로 마지막 소스코드로 구현되기 때문에 상당히 주의해야 한다. PIM 모델은 시간이 아무리 지나도 비즈니스 규칙이 바뀌지 않는 이상 변경되지 않는다. 이후에 시스템이 새로운 비즈니스 기회를 맞아 다른 외부의 시스템과 연동을 위한 아키텍처로 변경되더라도 이것은 플랫폼 선택에 대한 문제이다. 그리고 새롭게 변화되는 플랫폼과의 연동 문제는 PSM 모델에서 브릿지를 통해 해결할 수 있다. 개발자는 비즈니스 규칙을 자바, C++, C# 같은 프로그래밍 언어로 소스코드를 작성하는 것이 아니라 설계 모델과 비즈니스 규칙을 기술하기 위한 새로운 도구를 이용해 프로그래밍하게 된다.
그 다음으로는 시스템의 비기능적 요구사항을 기술한다. 이것은 PIM 모델을 플랫폼 종속적인 PSM 모델로 변환하기 위한 정보들로써 플랫폼에 대한 변환 정보, 객체 모델, 지속성 모델, 트랜잭션, 보안 등에 대한 정보를 기술한다. 기존의 통합 방법론의 4+1 아키텍처 뷰에서는 유스케이스 모델을 가장 중심에 두고 유스케이스가 개발 프로세스 전반을 주도하는 모델인 데 비해 MDA는 가장 핵심이 되는 PIM 모델을 중심에 놓고 관리하면서 PSM으로 변환을 하고 소스코드를 추출한 후 완전한 하나의 시스템으로 패키징되어 테스팅할 수 있는 모델을 갖고 있다. PIM 모델을 효과적으로 구축하는 것은 MDA 개발 활동의 핵심이라고 할 수 있을 만큼 아주 중요하다.

PIM 구축을 위해 필요한 것
PIM이 명확히 정의되어 있지 않으면 PSM 모델을 만들 수 없고 시스템 구축은 불가능하다. 그만틈 PIM은 아주 중요하다. 앞에서 PIM을 구축할 때 UML을 사용한다고 이야기했다. 그렇다면 평소에 쓰던 UML만을 사용해 시스템의 모든 부분을 구축할 수 있을까라는 의문을 가질 수 있고 UML을 사용해 본 대부분의 독자는 절대 그럴 수 없다고 주장할 것이다.
표준 UML은 시스템의 정적인 측면과 동적인 측면을 모두 모델링할 수 있다. UML은 시스템의 구조적 모델(structural model)을 모델링하는 데 강력한 기능을 갖고 있다. 클래스 다이어그램은 각 클래스간의 모든 관계를 표현할 수 있고, 그 이상으로 스테레오 타입과 태그 값 등을 사용해 개념적인 부분까지 표현할 수 있다. 또한 이 클래스들을 그룹화해 서브시스템 또는 컴포넌트로 표현 가능하고 인터페이스를 사용해 컴포넌트를 캡슐화할 수 있다.
그러 나 UML은 시스템의 동적인 부분을 세밀하게 나타내기에는 부족하다. 현재 UML에서 정의하고 있는 인터랙션 다이어그램, 상태 다이어그램, 활동 다이어그램만으로는 PSM을 추출하고 코드까지 완성하기 위한 동적인 부분을 상세하게 표현할 수 없다. 시스템의 정적 모델을 표현하는 것은 아주 중요하다. 이외에 어떤 상황에 따라 시스템이 어떻게 동작해야 하는지 기술하는 것도 시스템을 구현하기 위한 필수적인 요소다. 어쨌든 우리는 PIM 모델을 PSM을 거쳐 소스코드까지 변환시켜 완전한 시스템을 구축해야 한다. 그러면 이 동적인 부분을 어떻게 표현해야 할까. 그리고 PSM 변환을 위한 정보는 어떻게 구축해야 하는 것일까. 다음의 세 가지 확장 기술이 해결점을 제시하고 있다.

◆ Executable UML

◆ CWM(Common Warehouse Metamodel)

◆ OCL(Object Constraint Language)

▶ QVT(Query, View, and Transformations)

Executable UML은 Action Semantics를 정의하고 상태 머신을 이용해 동적인 부분을 보다 상세히 기술한다. 이것은 실제 소스코드로 변환시키기에는 여전히 추상화 레벨이 높고 명확히 표준화되지도 않았다. 결국 이것은 아직 PIM에서 사용하기에는 부족한 부분이 많다. CWM은 원래 데이터 웨어하우스를 관리하기 위한 메타 모델인데 스키마 모델 변환에 대한 정의도 포함되어 있다. 2003년 말을 기준으로 아직 CWM 변환에 대한 스펙 작업은 거의 이뤄지지 않은 상태여서 어떻게 이것이 정의될지는 좀 더 지켜봐야 한다. 오래 전부터 UML의 확장 메커니즘으로 사용된 OCL은 시스템의 동적 모델을 상세하고 명확히 표현하기 위한 좋은 도구이다. 우리는 UML과 OCL을 이용해 시스템의 비즈니스 규칙과 흐름을 기술할 수 있고, 이는 어느 정도 완성도 있는 PSM 모델로 변환할 수 있다. 그러나 사실 개발자가 상세한 부분까지 OCL을 이용해 모든 흐름을 기술한다는 것은 개발 속도를 저하시키는 피곤한 일이 아닐 수 없을 것이다.
QVT는 PIM에서 PSM으로, 그리고 PSM에서 소스코드로 변환시키는 작업을 위한 표준 변환 정보를 정의한다. 앞의 세 가지 모델은 모두 QVT 범주에 포함되며 이는 각 모델간의 변환 작업을 위한 표준으로서 현재 표준화 작업이 진행중이다. QVT에 대해서는 뒤에서 자세히 살펴보겠다.
기존의 개발 프로세스에서 이전의 모든 분석, 설계 작업은 결국 마지막 소스코드로 작성하는 것이다. 최종 목적은 좋은 코드를 만들어내는 것이다. 우리는 이 코드를 컴파일할 것이고 컴파일러는 시스템 레벨 또는 가상머신 레벨의 어셈블리 코드 또는 가상머신 코드를 만들어낸다. MDA 모델에서는 최종 목적이 PIM이라고 할 수 있다. 개발자는 좋은 PIM을 구축해야 한다. MDA 툴은 컴파일러처럼 PIM으로 플랫폼 종속적인 모델을 추출하고 소스코드를 만들어내고 최종적으로 완전한 시스템을 구축한다. 효과적인 PIM을 구축하는 데는 여러 설계 패턴들이 존재할 것이며 PSM으로 변환하기 위해서 변환 정보를 기술하기 위한 QVT가 그 핵심적인 역할을 할 것이다.

<그림 5> PIM 개발 모델


플랫폼 종속 모델, PSM

PIM 모델이 완성되면 이 모델 정보는 MOF 저장소에 저장되고 변환 정보에 기술된 내용에 따라 특정 플랫폼에 대한 PSM(Platform Specific Model) 인스턴스를 생성한다. PSM 모델은 특정 플랫폼에 종속적인 모델로서 모든 플랫폼에 대한 정보를 표현해서 이후 컴파일 가능한 코드를 산출한다.
PSM 모델을 구축하기 위해서는 먼저 대상 플랫폼을 선택하고 런타임 환경과 구성 정보를 기술해 특정 기술 및 미들웨어 플랫폼의 형태로 변환한다. 이때 효과적인 변환 작업을 수행하기 위해서는 해당 플랫폼에 대한 숙련자가 필요하고 이들이 PSM 변환 작업에 참여해 플랫폼에 적절한 패턴을 적용해 모델을 정제한다. 그러면 질 좋은 PSM 모델을 구축할 수 있다. 해당 플랫폼에 대한 모델은 특정 플랫폼에 대한 UML 확장판이 UML 프로파일을 이용해 표현하고, 특정 벤더 종속적인 정보는 각 벤더에서 제공되는 플러그인을 사용해 표현하거나 직접 개발자가 세부 내용들을 작성함으로써 표현할 수 있다.
하나의 PIM은 하나 이상의 PSM으로 진화되는데, 여러 시스템이 하나의 PIM 집합으로 관리되거나 한 도메인에서 정의된 서브시스템이 서로 다른 플랫폼을 가질 때(예를 들어 A 시스템의 비즈니스는 전체적으로 EJB 환경에서 구현되는데 특정 서브시스템의 인터페이스는 다른 시스템과의 상호운용성을 위해 웹 서비스화시킬 수 있다) 멀티 플랫폼 형태로 진화된다. 이 때 두 개 이상의 PSM 모델이 구축되며 각 PSM 모델간에는 브릿지를 통해 서로 통신한다.
플랫폼 독립적인 설계 모델은 특정한 기술, 미들웨어를 갖는 플랫폼 종속적인 모델로 변환하는 것은 결코 쉬운 작업이 아니다. 좋은 PSM을 구축하기 위해서는 좋은 PIM 모델 구축이 중요하겠지만 플랫폼 숙련자가 PSM 모델로 변환하는 데 얼마나 힘을 발휘하는지, PSM 변환 툴이 어느 정도나 자동화되는지, 또한 변환 툴이 각각의 플랫폼을 얼마나 확실히 지원하면서 어렵지 않는 UI 환경을 갖는지 등 현실적으로 구현되기 까다로운 부분이 많이 존재한다. 이것들을 어떻게 효과적으로 풀어가느냐가 OMG와 각 벤더들의 과제일 것이다.

<그림 6> PIM에서 PSM으로 변환 모델


애플리케이션 생성

PSM은 자동으로 해당 구현 환경에 대한 소스코드를 생성하고 런타임 구성 파일을 만든다. 만약 구현 환경이 EJB라면 홈 인터페이스, 리모트 인터페이스, EJB 클래스를 생성하고 배치를 위해 여러 배치 디스크립터(deployment descriptor)를 만든다. 구현 환경이 웹 서비스라면 툴은 WSDL 파일들을 생성하고 UDDI 레지스트리 정보를 구축한다.
기존의 케이스 툴도 모델로부터 소스코드를 생성하는 순공학(forward engineering)을 지원했다. 그러나 MDA 툴은 그것을 뛰어넘어 각 컴포넌트의 비즈니스 로직을 구현하고, 바로 실행시켜 서비스를 제공할 수 있는 ‘완전한’ 하나의 시스템을 구축한다는 것이다. 즉 컴파일러에서 전처리, 컴파일, 링크 과정을 거쳐서 실행할 수 있는 하나의 실행 파일을 뽑아내듯이 MDA 툴을 통해 바로 서비스 가능한 완전한 시스템을 만들어낸다. 자동화 툴에 의해 완전한 시스템 하나를 구축하는 것. 그것이 MDA의 지향점이다. 그러나 이것이 현실로 실현되려면 아직 넘어야 할 산들이 많다.

변환 정의(Transformation Definition)

소스코드를 작성하고 이것으로부터 실행 파일을 만들어낼 때 제일 중요한 것은 너무 당연하고 보편적으로 인식되어 어느새 우리의 기억속에서 잊혀져 버린 ‘컴파일러’이다. PIM에서 PSM, 그리고 PSM에서 소스코드로 변환하는 작업은 MDA의 꽃이라고 할 만큼 혁신적이고 중요한 개념이다. 이 작업을 수행하기 위해서는 먼저 컴파일러 격에 해당하는 변환 툴이 있어야 하고, 변환 작업이 어떻게 이뤄지는지에 대한 정보를 갖는 변환 정보가 정의되어 있어야 한다. 특정 프로그래밍 언어로 개발을 수행할 때 가장 기초가 되는 것이 바로 프로그래밍 언어의 문법이다. 이와 마찬가지로 MDA에서 변환을 수행하기 위해서는 반드시 표준화된 변환 정보가 정의되어 있어야 하고, 이 변환 정보 정의는 세부적인 표준화된 세부 지침과 명확한 가이드를 갖는 변환 규칙(transformation rules)들로 구성되어 있다.

<그림 8> MDA 개발 방식


<그림 9> 공통 QVT 컴포넌트 : 인프라스트력쳐 라이브러리의 특화된 형태로 나타난다


QVT 표준화 작업
OMG에서는 변환 작업에 대한 표준화 작업이 절실하다는 것을 느껴 변환 정보 정의의 표준화 작업을 시작했는데, 이 표준을 QVT(Query, View, and Transformation)라고 한다. QVT는 OMG에서 2002년 4월에 처음 RFP가 정의됐고 2003년 3월 첫 번째 서브미션을 거쳐 2004년 4월에는 두 번째 개선안이 발표됐다. QVT는 2003년도에 발표된 첫 번째 개선 작업에 의하면 전체 7개의 영역으로 나뉘어 정의하고 있다.

◆ 아키텍처 : QVT에 대한 전체 아키텍처를 설명한다.

◆ QVT Common Structure : QVT를 사용해 변환 정보를 정의하기 위한 기본 QVT의 개념과 용어들을 정리한다.

◆ QVT using a rule-based language : TRL(Transformation Rule-based Language)과 이를 표기하기 위한 BNF 문법을 정의한다.

◆ QVT in the context of UML Profiles : QVT에서 UML 프로파일에 기반한 변환 작업을 기술한다.

◆ QVT using a framework approach: QVT 작성시 프레임워크 적용에 대한 내용

◆ QVT Interoperability : 서로 다른 QVT 컴포넌트에 대한 상호운용 모델을 표현하기 위한 메타 모델을 제시한다.

◆ 추적성(Traceability) : 요구사항→PIM→PSM→소스코드까지의 추적성에 대한 개념을 제시한다.

모델 변환 방법
OMG에서 2003년 발표한 MDA 가이드 1.0에 따르면 다음 네 가지 형태의 변환 방법을 정의하고 있다.

◆ 손으로 직접 변환

◆ UML 프로파일을 사용한 변환

◆ 패턴을 적용해 변환

◆ 자동화 툴을 사용한 변환

일단 하나의 방법으로 개발자가 직접 변환하는 방법을 제시하고 있지만 이것은 지금까지의 설계 작업을 하는 것과 전혀 다르지 않다. 결국 MDA가 지향하는 바는 UML 프로파일과 패턴들을 적용해 툴에 의해 자동으로 변환 작업을 수행하는 것이다.

<그림 10> UML 프로파일을 적용한 QVT 변환 모델


<그림 11> QVT 프레임워크 접근법에 대해 인터프리터 패턴을 적용한 추상화 모델


변환 작업 수행에 대한 과제
개발자는 표준 QVT의 정의와 가이드에 따라 설계하고 시스템을 구현하기 위한 모든 정보를 기술한다. 이것은 툴을 통해 변환 작업을 거쳐 PSM을 추출하고 다시 이를 변환해 소스코드와 구성 정보를 만들어내고 이를 컴파일해 최종 애플리케이션을 구축한다. PIM 이후 작업은 툴에서 알아서 해주기 때문에 애플리케이션 구축 과정에 대해서는 신경 쓸 필요가 없다.
이 모델이 실현되기 위해서는 해결돼야 할 두 가지 문제가 있다. 그것은 시스템를 구축할 수 있을 정도의 정교함을 갖는 QVT의 표준화 작업과 QVT 정보를 분석해 모델을 변환하고 최종 시스템을 만들어내는 MDA 툴의 개발이다. 두 가지 작업은 이제 막 시작됐으며 아직은 안개 속에 있고 어떻게 그 최종 목표를 향해 진화될 것인지 지켜봐야 한다.

MDA 개발 모델과 개발자

앞서 MDA 개발 방식에 대해 살펴봤다. 독자들은 MDA를 적용한 개발 방식이 어떻게 수행되며 이것이 실현되기 위한 요소가 무엇인지 파악했을 것이다. 이제는 오늘날의 개발 프로세스를 생각해 보면서 MDA가 적용됐을 때 변화되는 개발자의 역할에 대해 살펴본다.

역할 변화

MDA 개발 프로세스는 기존의 개발 프로세스 방법과 차이점이 있다. 기존의 방법은 최종에 개발자가 효과적으로 하나의 거대한 시스템을 직접 손으로 구현해야 하는 것에 초점이 맞춰져 설계되어 있다. 그러나 MDA 개발 프로세스 모델은 요구사항을 정확히 분석하고 비즈니스를 세밀하게 모델링해 PIM을 구축하도록 하고 있다. 이 PIM에는 PSM으로 변화하기 위한 표준 지침인 QVT를 기술하고 이 정보들을 중심으로 PSM을 추출하고 소스코드를 추출하면서 플랫폼 아키텍처 모델, 데이터베이스 스키마, 테스트 코드, 구현 소스코드 등이 작성된다.
기존에는 대략 전체적인 개념만을 잡고 필요에 따라 세부 설계를 하여 개발했고 한번 작성한 모델이 구현 단계에서 변화하는 경우가 빈번했다. 그러나 MDA 개발 프로세스에서는 PIM 구축시 모든 비즈니스에 대한 정적 구조 및 동적인 부분을 완벽하게 기술해야 한다. 그래야만 원하는 애플리케이션을 구축할 수 있다. 결국 MDA 개발 프로세스로 인해 구축돼야 할 핵심이 변화하고, 이에 따라 개발자의 역할도 각 핵심을 실현하는 형태로 변화해야 한다.

각 역할에 대한 책임
MDA 개발 프로세스를 수행하는 각 역할을 정의하고 그 역할의 책임에 대해서 살펴보자. MDA는 새로운 개발 패러다임이다. 기존에 비해 각 개발 단계에서 추구하는 최종 목적이 다르며 이를 개발하는 개발자의 역할들도 그 목적에 따라 변화하게 된다.

◆ 분석가(Analyst, Requirement Analyst) : 고객으로부터 시스템에 대한 요구사항과 주요 문제점을 추출해야 할 것과 하지 말아야 할 것에 대한 접점을 명확히 분석해야 한다. 유스케이스 모델이나 활동 다이어그램을 이용해 PIM을 설계하는 데 중요한 정보인 기능적 요구사항 모델을 구축하고 보안, 성능, 가용성에 대한 내용을 갖는 비기능적인 요구사항을 정리해 CIM을 구축한다.

◆ 아키텍트(Architect) : 전 체적인 개발 프로세스를 진행하면서 기술 리더로서 시스템의 논리적, 물리적 모델을 제시하고 아키텍처 스타일, 설계 패턴, 플랫폼에 대한 선택과 적용에 대해 결정하며 효과적인 PIM을 구축하고 각 변환 작업을 수행할 때 핵심적인 역할을 수행한다.

◆ 설계자(Designer) : 요구사항 모델을 갖고 클래스 다이어그램, 상태 다이어그램, 시퀀스 다이어그램 등 각종 표준 도구를 사용해 요구사항을 모델링한다. 여기서 개발자가 설계하는 모델은 다른 모델과 소스코드로 변환되기 때문에 매우 엄격하게 분석 및 설계해야 한다.

◆ 구현자(Implementor) : 자 바, C++와 같은 소스 레벨의 프로그래밍 언어 대신에 QVT, OCL 및 Executable UML을 이용해 비즈니스를 구현한다. 이들은 MDA 개발 툴을 사용해 PSM 모델과 소스코드를 생성하고 그 자체로 하나의 시스템을 구축하기 위한 내용을 구현한다. 현재 개발자들 관점에서 보면 개발 작업이 더 지루해질 것 같은데, 앞의 사항들은 매우 세밀하게 기술돼야 하며 여러 테스트 도구와 유효화 도구들이 그 작업을 도울 것이다.

◆ 테스터(Tester) : MDA 툴에 통합되어 있는 MDA 테스트 도구는 현재 구축되어 있는 모델로부터 바로 테스팅 프레임워크를 구축하고 테스트 코드를 추출한다. 테스터는 MDA 테스트 도구의 특정 설정을 변경하거나 일정의 테스트 코드를 수정, 입력하면서 편리하게 테스트를 수행할 수 있다.

◆ 유지보수자(Maintenancers) : 유 지보수의 중요성은 아주 오래 전부터 이야기되고 있는 문제이다. 사실 소프트웨어 공학이 나온 가장 근본적인 시발점이 바로 유비보수 비용에서라고 말해도 무리는 아닐 것이다. MDA에서는 매우 정교한 PIM 모델이 존재하기 때문에 기존처럼 소스코드와 모델의 불일치성을 걱정하지 않아도 되고, 시스템의 비즈니스 흐름이나 구현 플랫폼이 변경되었을 때 소스코드가 아닌 PIM에 정의된 모델과 각종 규칙만을 변경하면 되므로 유지보수에 있어 생산성을 향상시킬 수 있다.

<그림 12> MDA 개발 모델에서의 역할


함께 하는 MDA 프로젝트

앞서 MDA의 개념, 개발 방식, 특징들에 대해 살펴봤다. 그러면 이것들을 적용해서 개발한다면 어떻게 진행될까. 정말 생산성 향상은 있을까?
현 재 여러 곳에서 MDA를 적용한 참조 구현 모델에 대한 자료가 발표되고는 있지만 아직 변환 정보 정의를 포함함 스펙 작업, MDA 개발 툴 및 WAS 벤더들의 MDA 적용을 위한 플러그인 등이 성숙되지 못해 그것의 실현 모델은 안개 속에 있다. 올해 말부터는 각 툴 벤더들이 MDA를 강력히 지원한다고 하니 필자도 MDA 도구들이 어떤 모습을 하고 있을까 기대된다. MDA의 실현에 대한 안개는 서서히 걷히고 있지만 그래도 변환 정보 정의가 정식 릴리즈되고 개발 방법론이 안정화되고 툴이 강력해지면서 MDA 시장이 성숙하는 데까지는 적잖은 시간이 소요될 듯싶다.

요구사항 정의

이제 MDA를 적용했을 때 어떻게 개발이 수행되는지 간단한 예제를 갖고 직접 MDA 개발 방식을 체험해 보는 시간을 갖자. 간단한 전자상거래 시스템을 예로 들자. 이 시스템은 전자 제품 쇼핑몰로서 사용자는 구입을 원하는 상품을 선택해 장바구니에 넣어놓고, 주문을 요청했을 때 주문되는 시스템이다. 우리는 이중에서 주문을 관리하는 부분을 구현할 것이다.
이 시스템은 3티어 구조를 갖고 있으며 누구나 쉽게 접속해서 주문 처리를 할 수 있도록 클라이언트는 웹 기반의 JSP 클라이언트가 필요하고 고객 데이터와 상품 정보, 주문 정보를 관리하기 위해 독립적인 데이터베이스를 사용한다. 그리고 미들티어에는 전자상거래 비즈니스를 수행하기 위한 비즈니스 로직을 구현한다.

PIM 모델 개발

개발자는 요구사항을 분석해 <그림 13>처럼 PIM 모델을 만든다. 사용자는 원하는 상품을 검색해 구매를 위해 ‘장바구니’에 지정한 상품을 넣는 작업을 할 것이고 이 상품은 ShoppingCart에 저장된다. 쇼핑을 마쳤으면 최종 주문을 위해 화면에서 ‘주문’ 버튼을 선택해 주문하고 주문 요청은 PurchaseOrder에서 받아 현재 장바구니에 있는 상품 리스트를 얻어와서 주문 총액을 계산한 후 주문을 처리한다.
현재 이 모델은 순수 비즈니스를 표현하고 있으며 어떤 플랫폼으로 어떻게 구현될 지 전혀 모른다. 다시 말해 이 모델은 웹 브라우저 클라이언트와 EJB로 구현될 수도 있고, 클라이언트 애플리케이션과 웹 서비스 모델로도 구현될 수 있다.
이제 PIM을 PSM으로 변환시켜야 한다. 모델 변환을 위한 변환 정보를 기술해야 한다. 이 시스템을 3티어 구조로 만든다고 정의했기 때문에 각 티어는 서로 다른 플랫폼을 갖게 되고 우리가 설계한 PIM은 최소 3개의 PSM으로 진화한다.
먼저 데이터 저장소를 위해 앞 모델로부터 객체 데이터 모델을 위한 변환 정보를 기술하고, 비즈니스 로직은 EJB로 구현하기 위해 EJB에 대한 변환 정보를 작성한다. 마지막으로 JSP, 서블릿 플랫폼에서 동작하기 위한 웹 클라이언트 모델로의 변환 정보를 작성한다. PIM에 변환 정보를 기술하고 MDA 툴을 통해 변환을 수행하면 자동으로 PIM에서 정의한 비즈니스를 갖는 각 티어별로 플랫폼 종속적인 PSM 모델이 만들어진다.

<그림 13> 쇼핑몰 예제의 PIM


<그림 14> 쇼핑몰 예제의 PIM의 진화 모델


PSM 모델 개발

PIM에서 변화된 PSM 모델을 살펴보자.

PSM #1 : EJB 플랫폼
PIM에서 설계한 클래스는 PSM으로 맵핑되면서 그 속성에 따라 세션 빈과 엔티티 빈으로 나뉜다. 세션 빈으로 맵핑되는 모델은 MDA 툴에 의해 홈 인터페이스와 리모트 인터페이스를 추출한다. 리모트 인터페이스의 리모트 메쏘드는 PIM 모델에서 특정 메쏘드에 리모트 메쏘드 성격을 기술해 만들어진다. 그리고 실제 비즈니스를 갖는 EJB 구현 클래스가 생성된다. 이 구현 클래스에는 PIM에서 정의한 비즈니스 메쏘드뿐 아니라 EJB 컨테이너에 의해 콜백되는 라이프사이클 관련 메쏘드도 만들어진다.
엔티티 빈으로 맵핑되는 모델도 홈 인터페이스와 리모트 인터페이스를 추출하고 해당 홈 인터페이스에는 엔티티 맵핑에 대한 메타 타입이 작성되면서 하나 이상의 find 메쏘드가 정의된다. 그리고 리모트 인터페이스에는 각 엔티티 애트리뷰트를 읽고 쓰기 위한 get, set 메쏘드가 정의되고 EJB 구현 클래스는 이들을 구현한다. 이처럼 만들어진 EJB 플램폼에 대한 PSM 모델은 UML 프로파일에 정의된 표준 EJB 표현 형식을 빌어 기술되고 이후 소스코드로 변환되기 위한 중간 단계인 PSM 모델을 완성하게 된다. 우리는 PIM에서 미들티어를 EJB로 선정해 플랫폼 종속적인 모델을 추출했다(<그림 15>). 만약 미들티어를 웹 서비스 환경으로 변환시키려면 웹 서비스 변환 규칙에 따른 변환 작업을 해주면 된다(<그림 16>).

<그림 15> PurchaseOrder의 PSM(EJB 맵핑)


<그림 16> PurchaseOrder의 PSM(웹 서비스 맵핑)


PSM #2 : 데이터베이스 플랫폼
PIM에서 정의한 비즈니스를 수행한 지속성 메커니즘으로 독립적인 데이터베이스 플랫폼을 사용한다. PIM에서 설계한 클래스는 관계형 데이터베이스 모델로 변환된다. PIM 모델에는 해당 비즈니스를 위한 엔티티 모델이 존재한다. 엔티티 객체 모델에서는 각 엔티티간의 관계를 모델링 또는 기술하게 되고 이 정보를 이용해 데이터 모델을 추출한다. 우리가 사용할 데이터베이스는 관계형 데이터베이스이기 때문에 객체-관계 맵핑(object-relational mapping) 규칙을 적용해 데이터 모델을 구축한다.
엔티티 객체 모델에서 하나의 클래스는 하나의 테이블로 맵핑되고, 클래스에 정의된 애트리뷰트는 하나의 컬럼으로 맵핑된다. 이때 애트리뷰트가 특정 데이터 타입을 갖는다면 그 애트리뷰트는 독립적인 테이블로 분리하며 분리된 테이블의 키 컬럼으로 foreign key 관계를 설정한다. 현재 설계 모델에서 바로 데이터 모델을 추출하는 문제는 많은 툴 벤더들이 오래 전부터 고민해 왔으며 현재의 케이스 툴에서도 상당 부분 지원하고 있다.

PSM #3 : 웹 클라이언트 플랫폼
웹 클라이언트의 모습을 살펴보면 사용자가 특정 요청을 보내면 시스템은 그 요청을 처리한 후 결과가 화면에 보여지는 형태를 띠고 있다. 웹 모델에는 화면에 결과가 디스플레이되기 위한 웹 엔티티와 화면 레이아웃, 사용자와 인터페이스하기 위한 HTML 페이지가 만들어진다.
이때 만들어지는 웹 데이터 스키마는 EJB 모델의 그것과 유사한 형태인데 다만 그것을 기준으로 사용자에게 나타나고 사용자를 통해 운용되기 위한 모델로 다듬어진다. 또한 클라이언트는 사용자와 인터페이스하며 사용자의 액션을 받아 처리하기 때문에 각 액션에 대한 정의도 구축되고 이 액션에 따른 웹 프리젠테이션 모델이 구축된다.
웹 인터페이스는 최초 요구사항을 수집할 때부터 정확히 정의하기 위해 사용되기 시작해서 프로젝트가 끝날 때까지 새로 생기고, 가장 빈번히 변경되는 부분이다. 결국 이 부분이 어떻게 구축되느냐는 프로젝트 생산성에 있어서 큰 영향을 미치게 되고 현재 이러한 문제를 해결하려는 관련 프레임워크들이 존재한다. 대표적으로 스트럿츠(Struts)의 예를 들 수 있다. 웹 모델로의 PSM 변환 작업은 QVT에 웹 모델 변환 표준들이 존재하면서 특정 또는 툴 벤더 자체의 웹 프레임워크를 적용한 형태로 변환 작업이 수행될 것이다.

소스코드 변환

이제 모든 비즈니스 정보를 갖는 PIM에서 각 플랫폼을 적용한 PSM을 통해 특정 환경 종속적인 소스코드를 만든다. 먼저 EJB 모델은 PSM 모델에서 구축한 모습에서 표준 소스코드 변환 규칙을 적용해 소스코드를 만들어내고, 데이터 모델은 DDL을 생성해 지정된 데이터베이스에 바로 데이터베이스 스키마를 구축한다. 그리고 웹 클라이언트는 지정된 프레임워크의 구현 모델로 변화하며 웹 저작툴을 이용해 사용자 친숙한 화면을 설계한다.
MDA 개발 방식에 대해 간단한 전자상거래 시스템 예제를 갖고 이야기를 했다. 사실 예제에서 제시한 방법은 MDA의 강력함을 느끼기에는 부족한 점이 많다. 비록 시스템의 일부분이지만 MDA 접근법으로 개발하는 모습을 그릴 수 있었으면 하는 바람이다.
예제에서 변환 작업을 수행할 때 아주 중요한 것을 빠뜨렸는데 그것은 바로 QVT를 정의하는 것이다. 앞의 내용들을 이해한 독자들은 변환 정보를 기술하기 위한 표준인 QVT가 어떻게 작성되는지 궁금할 것이고 이번 예제에서 모두 적용해 다뤄졌으면 할 지도 모르겠다. 사실 QVT에 대한 모든 내용과 변환 작업의 테크닉에 대한 내용은 이번 특집의 범위를 벗어난다. 또한 변환 정보 정의 작업은 그 형태가 MDA에서 그리는 이상향 정도로 완성되려면 앞으로 많은 시간이 필요하다. MDA의 변환 작업에 대한 것은 다음에 상세히 알아보고 일단 독자들의 궁금증을 해소시켜 주기 위해 쉽게 해석 가능한 변환 정보 정의 예제 코드를 제시하겠다.
<리스트 1>은 데이터 모델 변환화 작업을 할 때 사용되는 정의이다. 우리는 예제에서 기본적으로 하나의 클래스는 하나의 테이블로 맵핑시킨다고 이야기했다. 이것은 데이터 모델 맵핑 규칙에 대한 정의이다. UML을 SQL로 변환시킨다고 정의했고 UML에 정의된 각 속성을 SQL로 변환하는 모습을 확인할 수 있다.
<리스트 2>와 <리스트 3>은 UML 모델이 EJB로 맵핑할 때 사용되는 정의인데 UML 정의한 오퍼레이션은 EJB의 비즈니스 메쏘드로 변환시키고, UML 오퍼레이션에 사용한 인수들 또한 EJB 비즈니스 메쏘드의 인수로 변환시키겠다는 정의를 하고 있다.
미들웨어컴퍼니(http: //www.middlewarecompany.com/)의 자료를 찾아보면 흥미있는 리포트를 볼 수 있다. 그것은 바로 썬에서 J2EE 아키텍처 적용 예제 시스템의 PetStore 도메인을 갖고 두 개의 개발팀을 조직해, 한 팀은 전통적인 방법으로 개발을 수행하고 다른 팀은 MDA 방식을 적용해 개발을 수행한 후 개발 생산성에 대한 결과를 분석해 놓은 문서이다. 각 팀의 주 단위 개발 현황 리포트도 간략하게 볼 수 있다. 그 결과는 MDA의 승리였다. 전통적인 개발 방식을 선택한 팀은 전체 약 499시간이 소요될 것을 계획해 507.5시간에 개발을 완료했고, MDA 방식을 선택한 팀은 약 442시간이 소요될 것을 계획해 개발을 완료하는 데 330시간이 소요됐다. MDA를 적용한 개발 방식이 10~20% 빠른 개발 생산성을 가져왔다. 두 팀의 멤버들은 모두 EJB 숙련자들이었고 EJB 관련 툴 사용에 익숙한 상태였으나 MDA팀 멤버는 이번에 처음 MDA를 적용해 개발하고 툴을 사용하는 것이기 때문에 기술 습득의 소요 시간을 제외하면 더 나은 생산성을 추출할 수 있다고 한다. MDA를 적용한 모델은 추후 유지보수까지 생각한다면 더 많은 비용을 줄일 수 있을 것이다. 예제의 목적은 단지 빠른 개발에 목표가 맞춰진 개발 생산성의 분석이다. MDA 툴을 통해 만들어진 소스코드는 지저분하고 비효율적인 부분이 여럿 발견됐다고 하는데 이것은 앞으로 시간이 지나면서 개선될 것이다. 컴파일러가 점차 최적화되었던 것처럼 말이다.

MDA 호의 닻은 올라가고

MDA는 이제 막 마지막 결승점을 향한 스타트를 끊었다. 표준화 단체에서는 MDA 구축을 위한 여러 제반 기술들을 개발하고 있고, 모델 변환을 위해 QVT 표준화 작업도 계속 수행중이다. 주요 벤더들은 MDA 툴을 개발하기 시작했으며 올해 말에서 내년 초 정도면 그 모습이 서서히 드러날 것이다. 또한 MDA 방법론도 그 구색이 많이 갖춰지며 MDA 개발 프로세스를 가이드할 것이다. MDA 툴들이 출시되면 현재의 MDA 실현 모습을 물리적인 시야 안에 놓고 바라볼 수 있을 것이며 점차 진화해 소프트웨어 개발의 새로운 패러다임으로 자리잡을 것이다





새로운 IT 패러다임이 등장할 때마다 현기증을 느낀다. 특히 소프트웨어 개발 업계에 막 입문한 사람보다는 나름대로 경험과 노하우를 가지고 있고 실질적으로 새로운 작업을 추진할 만한 위치에 있는 엔지니어일수록 그런 현상이 더욱 심한 것 같다. 국내의 가장 전형적인 SI 업체의 R&D 팀에 소속된 필자의 경우도 꼭 그런 상황이다. 이 글에서는 최근 이슈가 되고 있는 MDA를 필자가 속한 회사가 전문 개발 영역인 ‘인사관리 업무’ 도메인에 도입해, 그것을 조직의 소프트웨어 개발 생산성 향상으로 접목시키는 과정에서 겪었던 그리고 앞으로 겪을 것으로 예상되는 문제점들과 그 해결 방안을 담담히 정리해 본 것이다.

현재 필자가 근무하는 회사에서는 지난 2003년을 전후해서 제대로 만들어진 컴포넌트 혹은 컴포넌트가 아니더라도 무엇인가 실질적인 소프트웨어 생산성을 끌어 올릴만한 묘안이 절실히 필요했다. 2~3년 전만 해도 그 묘안은 CBD 혹은 컴포넌트였고 현재도 그러한 사실에는 변함이 없다. 필자의 조직은 그러한 컴포넌트 패러다임이 이슈로 등장하던 시절의 초기에 나름대로 사활을 걸었다. CBD 사상에 전력투구했던 덕에 지금까지 업계에서 CBD에 관한 기술 선도 업체로서 인정받으면서 영업적으로나 마케팅적으로 그 효과를 보고 있는 것이 사실이었다. 하지만 컴포넌트가 가져다 줄 것이라고 믿었던 본질적인 소프트웨어 생산성 효과보다는 오히려 그 이면에 깔린 마케팅적인 대외 홍보 효과가 더 주효했던 것이 부정할 수 없는 사실이기도 하다. 이 글에서는 CBD 이후에 새롭게 주목받고 있는 MDA를 조직에 접목하게 된 배경을 비롯해 도입하는 과정에서 겪었던 문제들과 이를 해결하기 위해 고심했던 방안을 필자가 속한 R&D 팀을 중심으로 이야기하고자 한다.

MDA 도입 배경

2003년을 전후해서 필자의 회사는 꾸준히 프로젝트를 수주해 그 수가 증가했고 더불어 규모나 기간도 상대적으로 커지고 길어졌다. 어느 회사에서나 그러하듯이 전통적인(?) 대처 방식, 즉 새로운 개발자를 내부 직원으로 충원하거나 임시방편으로 외부 인력을 조달하면서 프로젝트를 진행했다. 하지만 시스템의 개발 납기가 자의반 타의반 연장되면서 그로 인한 기회비용 손실은 말할 것도 없고 회사가 ‘울며 겨자 먹기’ 식으로 어쩔 수 없이 떠안아야 하는 재정적 부담은 이루 말할 수 없었다. 회사의 재정을 관리하는 관리 부서를 중심으로 볼멘소리가 터져 나오기 시작했다. ‘대형 SI 업체들의 프로젝트 저가 수주 전략(?)이라는 육탄 세례 속에서, 이미 탁상공론을 넘어 저 먼 외국 사례로 인용되고 있는 정통부나 과기처의 소프트웨어 개발 단가표는 고사하고, 우리가 무슨 떼돈 벌겠다고 하는 것도 아닌데…’ 근본적인 해결책이 없는 가운데 프로젝트를 계속해서 수주하는 것이 관리부서 입장에서는 기쁜 일만은 아니었다. 사실 이에 대해서는 중소 SI 업체에서 할 말이 많을 것이다. 대형 SI 업체의 프로젝트 저가 수주는 곧 중소 하청 SI 업체가 고스란히 떠안아야 할 부담이 된다는 것은 자명한 사실이다.
실제 현장(site)에서 프로젝트를 수행해 보면 그런 소리 못한다고 현장에서 겪는 고초를 하소연하던 개발팀에서도 점차 자성의 목소리가 흘러 나왔다. 당장 촉박하게 프로젝트를 종료하기 위해 몸부림치던 당시에는 못 느꼈지만 막상 프로젝트를 마무리하고 본사로 철수하거나 다른 현장으로 투입된 초기에 가만히 앉아서 살펴보면, ‘저쪽 고객사에서 요구했던 것이나 이쪽 고객사에서 요구하는 것이나 크게 차이가 나지 않는데 왜 그렇게 프로젝트를 마무리하기 어려웠을까?’ 머리를 갸우뚱거리게 된다. 고객의 요구사항이 수시로 바뀌는 것이야 어제 오늘의 이야기가 아니고 인터넷 기반의 시스템 개발이 보편화되면서 특히 그런 요구사항 변경이 시스템 개발을 어렵게 한다는 것도 이해는 가지만, 어쨌든 SI 업계에서 수행하는 시스템 개발 프로젝트에서의 소프트웨어 생산성 문제는 대형 업체에 비해 중소 업체 입장에서는 사활이 걸린 중대 사안일 수밖에 없다.

MDA 도입 과정

‘Evaluating Software Architecture-Methods and Case Studies’라는 책을 보면 소속 조직에 새로운 기술이나 패러다임을 도입하기(ATAM 기법) 위한 아주 세부적인 전술을 제시하고 있다. ROI 분석에서부터 조직 내에서 동조자를 끌어들이고 의사결정권자를 설득하기 위한 세세한 조언까지 아끼지 않고 있다. 그러한 아이디어를 접목하여 필자의 조직에 MDA를 도입했던 과정을 살펴보기로 한다.

<그림 1> MDA 도입 메커니즘


경영진의 설득

앞선 MDA 도입 배경에서 살펴본 상황으로부터 쉽게 유추할 수 있겠지만 필자의 조직에서는 일단 어떠한 형태로든 돌파구가 필요한 상황이었다. 경영진들 입장에서의 압박감은 그 상황이 더욱 심각하게 고려되었다. MDA 이전에 가장 활발하게 이슈가 되었던 컴포넌트에 관해서는 조직 내부에서조차 그 효용성에 대해 다소 회의적인 시각들이 팽배해 있었다. 이는 곧 IT 업계에서 시기마다 발표되는 패러다임 혹은 기술이라는 것에 대한 부정적인 시각으로 확대 해석되고 있는 상황이었다. 그렇다고 컴포넌트 패러다임을 부정하면서 그것을 대체할 수 있는 패러다임이라고 MDA를 부각시키는 것은 타당성이 없어 보였다. 사실 MDA는 CBD를 대체하는 패러다임이라기보다 오히려 CBD 사상에 근거한 소프트웨어 개발의 단위인 컴포넌트를 보다 컴포넌트답게 시스템 개발 초기에 모델이라는 형태로 만들고 향후에도 그 수준에서 재사용해 보자는 것이 근본 취지였다. 물리적인 실행 파일 수준 컴포넌트 개념에서 모델 수준 컴포넌트 단위로의 형태론적(syntax) 변화일 뿐 컴포넌트의 전면적인 부정은 아니다.
필자는 그러한 사실을 지속적으로 조직 구성원, 특히 조직의 방향성에 대한 결정권을 가지고 있는 경영진들에게 전달하고 그러한 MDA 접근법이 보편화됐을 때 예상할 수 있는 시스템 개발 방식의 변화 등을 설명하려고 노력했다. 그리고 더 현실적인 측면에서 필자의 조직에서 활발하게 활용하고 있는 모 업체의 CASE 도구 속에서 그러한 MDA 사상을 접목시키고자 노력하고 있는 모습, 즉 추가하고 확장하는 기능이나 기술을 예로 들면서 앞으로 MDA가 향후 소프트웨어 산업계의 핵심 메커니즘으로 자리잡아 갈 것을 확신시키려 노력했다.
그 리고 MDA를 통해 획득할 수 있는 소프트웨어 생산성에 대한 참고자료를 소개하는 시간을 많이 갖도록 노력했다. 그 자료 중에 2003년 6월에 발표된 미들웨어 컴퍼니 연구팀에서 발표한 ‘MDA 접근법을 활용한 J2EE 플랫폼 기반의 모델중심 개발에 대한 생산성 분석(Model Driven Development for J2EE Utilizing a Model Driven Architecture (MDA) Approach-Productivity Analysis)’이나 2004년 1월에 추가로 발표한 ‘MDA 접근법을 활용한 J2EE 플랫폼 기반의 모델 중심 개발에 대한 유지보수성 분석(Model Driven Develo pment for J2EE Utilizing a Model Driven Architecture (MDA) Approach-Maintainability Analysis’ 보고서는 매우 설득력 있는 자료로 활용할 수 있었다.

수행 조직 결정

국내의 영세한 SI 업계 상황을 고려할 때 굳이 MDA가 아니더라도 고객의 명시적인 요구사항으로 제시되거나 모험을 감수해야 하는 신생업체가 아닌 이상, 일반 중소 SI 업체 입장에서는 특정 패러다임이나 기술에 대한 전문가가 부재한 상황에서 지금까지 쌓아온 개발 노하우나 방식을 전면 뒤엎으면서 새로운 패러다임이나 기술을 전면적으로 도입하는 일은 일종의 도박이다. 그것도 당장 법적으로 납기일이 명시된 현장에서 도입한다는 것은 더욱 불가능하다.
R&D 팀의 성격상 당장 고객들에게 무엇인가 보여줘야 하는 개발팀들과는 달리 2~3년 전에는 CBD, 그리고 이번에는 MDA와 같은 새로운 기술 혹은 패러다임을 접하고 그것을 조직의 새로운 패러다임으로 접목할 임무를 수행하기로 결정했다. 그것은 R&D 팀의 특정상 어쩌면 당연한 임무이자 역할이기도 하다. 하지만 관행적으로 SI 업체 R&D 조직의 보편적인 역할은 해당 업체에서 수행하는 여러 프로젝트에서 공통으로 사용하거나 상대적으로 난이도가 있는 모듈 혹은 컴포넌트 개발 및 유지보수를 전담하거나 프로젝트 초기 전반적인 시스템 아키텍처 설계 정도에 한정되는 것이 보통이다. 필자의 R&D 팀이 조직 내에서 수행하는 역할도 그런 R&D 팀의 보편적인 역할과 크게 다르지 않았다. 그러나 그것은 본격적인 작업(?)을 수행하기에는 적합하지 않기 때문에 그러한 문제를 해결하기 위해 내부적으로 R&D 팀 구성원을 추가로 확보하면서 조직 내의 개발팀들을 지원하는 팀과 MDA 기반의 내부 프로젝트를 진행할 팀으로 이원화하는 방식을 채택해 역할을 분담했다.

내/외부 공모자 확보

인사관리 도메인 지식을 갖춘 사람과 새로운 기술을 개척할 수 있는 엔지니어를 중심으로 MDA 관련 자료를 회람하면서 정보를 공유했다. 또한 내부적으로 기술적인 측면에서든 인사업무에서든 어느 한 편에 대한 상대적 경쟁력을 갖춘 역량 있는 인력을 R&D 팀으로 합류시키고, 외부적으로는 필자의 R&D 팀에서 MDA라는 새로운 패러다임에 기꺼이 도전하려는 새로운 인력 충원을 적극 고려했다. 설령 그것이 MDA가 아닌 다른 사상이나 기술이라 할지라도 개인적 신념만으로 일을 추진하기에는 너무나도 험난한 것이 사실이고, 최근의 시스템 개발 문제는 조직 내에서 단순히 능력이 뛰어난 몇 명의 컨설턴트나 엔지니어에 의해 주도적으로 수행되기에는 이미 그 규모나 비중이 커졌다는 사실을 반증하는 것이기도 하다.

개발팀과의 마찰 해소 방안

코드 중심의 개발 방식(code-centric development)에 익숙한 개발자에게 당장 눈앞에 직면한 프로젝트에 대해 반드시 MDA를 적용해서 시스템을 개발하라고 강제한다면 그것은 현장에서 고객들과 씨름해야 하는 개발팀원들은 ‘기름통을 들고 불 속으로 뛰어들어라’는 말과 다르지 않게 느낄 것이다. 그래서 일단 MDA 관련된 R&D의 작업 방향에 대해 관심을 내비치는 엔지니어에게는 그저 앞으로의 소프트웨어 개발 방식은 UML을 보다 적극적으로 활용하는 방식으로 진화할 것이며, 그에 대한 각자 나름대로의 대비가 있어야 한다는 정도로만 이슈로 제시하면서 R&D 팀을 중심으로 그러한 준비를 조직적으로 수행하고 있으며 점차 그러한 작업이 실체화되는 시점에 내부 인력을 대상으로 내부 교육이 진행될 것임을 주지시켰다. 그러한 내용을 굳이 개발팀원에게까지 전파하는 목적은 무엇보다도 향후 조직에서 필요로 하는 엔지니어가 되기 위한 실천방안을 사전에 비전으로 제시함으로써 최소한 UML을 활용해 구축된 모델을 접하게 될 시점에 거부감 없이 쉽게 수용하고 수월하게 전환 교육이 이루어질 수 있도록 채비시키기 위함이었다.

MDA 실현 이슈와 해결

지금까지 필자 조직에서 MDA라는 전략이 결정된 배경과 그 도입 과정을 간단히 살펴보았다. 이제부터는 더 구체적으로 MDA를 수용하고 필자의 조직에서 목적하는 시스템(여기서는 인사관리 시스템 중심으로)에 접목하는 과정에서 발생했고 앞으로 발생할 것으로 예상되는 난제(issue)와 그것을 해결하기 위해 채택한 전술에 대해 살펴보자.

전문 도메인 지식 습득

MDA 기반의 인사관리 시스템 개발을 추진하는 수행 조직으로 결정된 R&D 팀 이전 구성원들의 역량은 기술적인 측면에 비해 상대적으로 인사관리라는 비즈니스 측면에 대한 지식이 부족한 상황이었다. 어떠한 도메인이든 MDA 기반으로 제대로 된 시스템 개발을 위해서는 해당 도메인에 대한 전반적이고 체계적인 지식이 필수적인 상황에서 단지 CASE 도구를 활용한 UML 기반의 모델링이 가능하다는 사실은 그저 ‘무엇(business)’인가를 담을 수 있는 ‘틀(technology)’은 갖췄지만, 막상 담아야 할 ‘내용(business)’이 무엇인지는 모르는 형국에 지나지 않았다. 그렇다고 현재 추진 중인 R&D 팀의 MDA 관련 내부 프로젝트의 성격이 조직 내의 다른 개발팀처럼 특정 고객에게 필요한 인사관리 시스템만을 구축해 주고 나면 그 역할이 마무리되는 상황도 아니었다.
그러한 미비점을 보완하고자 일단 R&D 팀과 현장에서 시스템을 구축 중인 개발팀과의 유기적 정보 교류 네트워크를 구축하였다. 일단 R&D 팀에서는 지금까지 필자의 조직에서 수행한 인사관리 시스템 구축 시 활용되거나 작성한 자료를 수집하고 전사적인 차원에서 공통적으로 참고할만한 관련 자료를 정리 및 통합한 후 다시 그러한 내용을 각 개발팀원이 모두 공유할 수 있도록 시스템을 구축했다. 필요한 시점에 R&D 팀에서 각 사이트에서 진행 중인 프로젝트의 상황을 모니터링하여 필요시에 해당 사이트에 방문하여 추가적인 보완 자료를 수집할 수 있도록 사전에 협조를 요청하였다.
또한 R&D 팀에서 진행하는 인사관리 시스템은 특정 고객사를 대상으로 하는 시스템이 아니기 때문에 개발 현장에서 보편타당하게 수용할 수 있는 제품이어야 하는 특성을 고려해 인사관리에 관한 국내외 표준자료를 수집했다. 그것은 현재 OMG를 중심으로 특정 도메인에 대한 데이터 교환 표준 규약을 XML 기반으로 구축 중인 것과 일맥상통한다. 그러나 아직 OMG에서 명시적으로 인사관리 도메인에 대한 표준화 작업은 미비한 상태였다. 그 와중에 외국의 인사관리 관련 업체들을 주축으로 결성된 HR 컨소시엄에서 제시하는 공개된 자료를 발견했다. 그곳에서 제안하는 XML 기반의 인사 관련 표준 데이터 스키마와 프로세스를 중심으로 보편타당한 인사 시스템의 비즈니스 아키텍처 작성을 진행했다. 국제적으로 인정할만한 표준을 검토해 시스템 개발에 접목하게 된 것은 회사의 장기적인 비전인 국제무대로의 진출을 염두에 둔 것이었다. 명시적인 고객이 존재하지 않는 R&D 팀 관점에서 이것은 일종의 고객 요구사항으로 고려되었다.

제대로 된 UML 지식

간혹 필자는 주변에서 현재 OMG에서 준비 중인 UML 2.0 명세 최종 버전이 확정되고, 각 CASE 도구 벤더들이 본격적으로 UML 2.0을 지원하는 시점부터 MDA 기반의 시스템 개발이 가능하지 않겠느냐는 질문을 받을 때가 있다. 그러나 그것은 MDA 관련 참고할 만한 서적을 한번이라도 읽어 본 사람이라면 사실과 다르다는 것을 금방 알 수 있다. 기왕이면 ‘새로운 포도주(MDA)를 새로운 그릇(UML 2.0)에 담는 것’이 좋기는 하지만 현재 CASE 도구 벤더들이 표준으로 지원하고 있는 UML 1.4(필자가 글을 쓰고 있는 시점에 UML 2.0을 완벽하게 지원하고 있는 CASE 도구는 없다. 특정 벤더에서는 대외적으로 세계 최초의 UML 2.0 CASE 도구라고 선전하고 있지만, 실제 모든 UML 2.0에서 제시하는 모든 표기법과 다이어그램을 지원하지는 않는다. 무엇보다도 UML 2.0 명세는 현재 최종 초안이 공표되고 최종 심의를 거치고 있는 과정이다)에서도 충분히 MDA적인 시스템 개발 접근법이 가능하다. 무엇보다도 대표적인 많은 CASE 도구들이 이미 다양한 형태로 OMG에서 제안하는 MDA 구현 표준에 근거한 확장 메커니즘을 적용하여 CASE 도구에 새로운 기능을 추가하거나 기존의 기능들을 보완하고 있는 상황이다.
UML 2.0이 전제되어야만 MDA식 개발이 가능하다는 오해의 이면을 살펴보면 많은 업체들이 지금까지 표면적으로는 UML이라는 OMG의 모델링 표준에 근거해 분석과 설계 작업을 한다고 했지만, 실질적으로는 완벽한 UML 중심의 작업이었다기 보다는 UML의 일부 간단한 표기법(notation)을 중심으로 개념적이고 피상적인 수준에서 UML을 활용했다는 것을 반증하는 것이다. 그러한 결과 소프트웨어 산업계에 만연되어 있는 ‘시스템 개발 초반의 CBD 컨설팅 따로, 후반의 실질적인 개발 따로’라는 이분법적 고정관념을 엔지니어 머리속에 만들어 냈고, 이는 곧 프로젝트의 전반적인 생산성 저하라는 고질적인 소프트웨어 산업계의 악순환 구조를 양산하고 말았다.
무엇보다도 MDA 기반의 시스템 개발 방식과 지금까지의 코드 중심 개발 방식은 표면적으로는 큰 차이가 없어 보이지만, 그 내면을 들여다보면 혁신적인 패러다임의 전환이 필요하다는 사실을 직감할 수 있다. 소프트웨어 산업계에 MDA 개발 방식이 보편화되기 시작하면 이것은 예전처럼 소위 UML을 통해 어설프게 모델링 작업을 수행하던 사람들에게는 새로운 도전이며 그 동안 UML을 먼 나라 이야기처럼 한쪽 귀로 듣고 한쪽 귀로 흘려 왔던 엔지니어들에게는 가히 극복하기 어려운 도전으로 다가올 것이라 생각한다.
이러한 상황에서 R&D 팀은 내부적으로 이전에 대충 훑어보았던 기존의 UML 1.4 명세의 상급 수준 특징을 좀 더 면밀히 검토하고, 더 정밀한 모델링이 가능하도록 설계 문서간의 연관 관계에 대해 고민하기 시작했다. 특히 앞서 살펴본 인사관리 도메인을 체계적으로 모델링하는 작업에 전념했다. 한편으로는 현재 시중에 나와 있는 MDA 관련 서적을 집중적으로 살펴보고 UML 2.0 초안을 검토하여 기존에 활용하던 UML 1.4와의 차이점과 보강된 내용을 체계적으로 정리하는 작업을 수행했다.

CASE 도구 선정

특정 벤더의 모델링 CASE 도구를 필자 조직의 기본 도구로 선정하는 과정에서는 사실 거의 팀 내에서의 이견은 없었다. 무엇보다도 오랫동안 필자의 회사에서는 여러 프로젝트를 수행하면서 어설프게라도 해당 CASE 도구를 기본으로 활용해 작업했던 까닭에 상대적으로 다른 CASE 도구들에 비해 사용자 저변이 많이 확산되어 있는 상황이었다. 향후 MDA 기반의 시스템 개발이 확산되면 기존의 개발 도구나 각종 CASE 도구들에 상대적으로 모델링 CASE 도구에 대한 종속성이 커지겠지만 그렇다고 기존의 여타 코딩이나 형상관리 등의 작업에 필요한 도구 그리고 운영 플랫폼들과의 연동이나 통합을 고려하지 않을 수 없었다. 그런 관점에서 보더라도 필자의 조직에서 선정한 모델링 CASE 도구는 나름대로 적절한 선택이었고 최소한 필자 조직에서는 가장 자연스러운 최적의 선택이었다.
엔지니어의 도구 친밀도와 기술적인 측면에 덧붙여 해당 CASE 도구 제작 벤더에 대한 시장에서의 고객 선호도를 고려했다. 고객들의 CASE 도구 선정 경향은 일반적인 엔지니어적인 관점에서의 기술 구현 완성도나 기능 등에 국한되지는 않는다. 오히려 수치화가 어려운 고객간의 입 소문이라든지, 고객사의 과거 경험이나 주변인의 권유가 많이 작용하는 것이 현실이다. 고객사의 특별한 제약사항으로 제시되는 경우가 아니라면 프로젝트 초기에 개발업체에서 ‘이러이러한 도구들을 중심으로 개발하겠습니다’라는 식의 도구 제안이 들어가게 된다. 고객들은 앞서 언급한 바와 같이 특별한 상황이 아닌 이상 그러한 제안을 수용하는 것이 일반적이다. 무엇보다도 지금까지의 개발 방식에서는 CASE 도구 특히, 모델링 도구가 큰 비중을 차지하지 않았고 더욱이 국내에서 소프트웨어 개발 환경에서의 모델링 도구 위상이란 그저 요식적인 ‘그림 도구’에 지나지 않게 치부되었습니다. 그것은 곧 고객들이 모델링 도구 선정에 있어 각종 플랫폼 선정 작업에 비해 상대적으로 면밀히 검토하지 않는 원인이 되었다. 그런 현실적인 여건과 상황을 고려하여 필자가 속한 조직에서 선정한 모델링 도구는 도구 제안 시 전반적으로 고객들이 거부감을 느끼지 않을 정도의 도구로 평가될 도구였다.

모델 버전 관리

MDA 기반의 시스템 개발을 추진하게 될 고객 입장에서는 해당 결과물인 각종 모델들과 모델간 변환(transformation)을 위한 메커니즘 구현물(profiles)만을 대상으로 유지보수하는 것으로써 특정 상위 비즈니스 업무가 변경되더라도 상위 모델(PIM) 중심의 수정 작업을 통해 손쉽게 시스템을 재구성하거나 변형할 수 있다. 또한 기존의 형상관리 기법을 그대로 적용하면 손쉽게 조직 내에서 달성해야 할 소프트웨어 생산성 혹은 재사용성이라는 소기의 목적을 달성할 수 있다. 하지만 필자의 조직처럼 특정 업무영역을 특화해 비즈니스 모델 결과물(PIM)을 제작하고, 그것을 다시 재사용하면서 여러 고객 사이트에서 다양하게 변형하거나 커스터마이징 작업을 통해 수익을 창출해야 하는 경우라면 더 복잡한 메커니즘 구현이 필요하다. 이 글을 쓰고 있는 시점을 기준으로 이제 막 MDA를 도입해 시스템 개발을 추진하는 과정 중에 있는 관계로 그에 대한 명쾌한 방안을 제시할 수는 없지만 필자의 조직과 유사한 처지에 있는 상황에서 MDA를 전략적으로 도입하려고 준비하는 조직이라면 반드시 MDA 기반 시스템 개발 초기에서부터 염두에 둬야 할 주요 이슈일 것이다.
지금 현재 그러한 모델의 버전 관리 문제와 관련해서 시스템 설계 시에 고려하는 사안은 일단 인사 도메인에서 단위 비즈니스 프로세스 및 데이터 구조 자체의 비즈니스적인 유연성을 확보하기 위해 단위 업무별 공통성과 가변성을 시스템 설계 초기부터 모델에 반영하고 특정 플랫폼으로의 전환시 해당 변환 규칙에서 불필요한 요소를 아예 전환이 되지 않도록 제어하는 방식으로 고려하고 있다. 바로 SPL (Software Product Lines) 접근법이다. 또한 단위 업무(엄밀히 말하면 단위 업무별 비즈니스 모델)뿐만 아니라 변환 규칙 자체를 형상 아이템(CI: Configuration Item)으로 전환할 것을 염두에 두고 작업을 진행 중이다. 특정 CASE 도구에 종속적일 수도 있겠지만 보다 현실적인 접근법으로써 내부적으로 결정한 CASE 도구에서 제시하는 다양한 형태의 모델 단위 구조와 실제 구현 대상이 되는 인사관리 단위 업무를 매칭시켰다.
특히 필자가 주목하는 사항은 OMG의 MDA 접근법에서 제안하는 프로파일이라는 형태의 변환 규칙의 조직 자산화 문제이다. 즉 기존의 전통적인 개발 방식과는 다르게 메타 데이터를 통해 소스(PIM)를 목적 플랫폼(PSM)으로 변환하는 기술이 궁극적으로는 향후 MDA 기반 개발이 보편화되는 시점에는 필자의 조직과 같은 소프트웨어 개발 및 통합 업체 입장에서는 타사와 차별화될 수 있는 기술력에 대한 실질적인 자산이 될 것이라는 점에 주목하고 있다.

모델 보안 문제

마지막으로 앞선 단락에서 언급한 MDA 기반 기술 개발로 축적되는 모델이라는 새로운 형태의 조직 자산에 대한 보안 문제이다. 가장 중요한 문제이면서도 아직까지 명확한 방안을 생각해내지 못한 이슈이기도 하다. 가령 다른 업계에 비해 상대적으로 IT 업계에서 보편화되어 있는 엔지니어의 잦은 조직 이동 문제를 고려해 보자.
핵심 인력의 이동 문제는 과거에도 그랬고 향후에도 해당 인력의 이전 조직에 결정적인 타격을 줄 것임은 어쩔 수 없다. 하지만 MDA 기반의 개발 방식이 보편화 되는 시점에서의 핵심 인력 유출은 지금의 코드 중심의 개발 방식 시절에 비해 상대적으로 상당한 수준의 기술 자산 유출로 이어질 것임은 너무도 자명하다. 개발 소스나 자료의 보안 관리가 허술하게 이뤄지고 있는 국내 프로젝트 관리 실정을 고려하면 심히 염려되지 않을 수 없다. 가령 코드 중심의 개발 방식에서는 어떤 엔지니어가 이전 직장에서의 프로젝트에서 활용했던 아무리 잘 작성된 소스코드를 가지고 있다고 해도 막상 유사한 새로운 프로젝트에 적용하려면 소스 내부를 수정하지 않고서는 거의 재사용이 불가능한 것이 일반적이다. 반면 MDA 기반의 개발 방식에서는 소스 레벨보다 상위 추상화 수준에서 약간의 수정만으로도 지금까지 상상할 수 없었던 수준의 소프트웨어 재사용이 가능하고 이는 곧 해당 엔지니어의 역량으로 평가될 수 있을 것이기 때문이다.
모델이 아니더라도 기업의 보안 문제는 단순히 기술적으로 해결할 수 있는 문제가 아니라 사회의 제도적인 장치 마련이나 기업 문화의 성숙 등과 같은 복합적인 해결 방안이 강구돼야 할 사안으로 생각된다. 조직의 MDA 모델 자산의 보안 문제는 바로 그러한 맥락에서 그 해결 방안을 고려해야 하고 MDA 도입을 고려하고 있는 조직에서는 반드시 염두에 둬야 할 이슈가 아닐 수 없다.

컴포넌트를 더 컴포넌트답게

국내 IT 업계의 원로격이신 필자가 근무하는 연구소 소장님으로부터 예전의 구조적 프로그래밍이 팽배했던 시절 이야기나 그 원리 그리고 당시의 사회적 분위기에 대해 들으면서 놀라움을 금치 못하곤 한다. 필자의 미천(?)한 경험 탓이기도 하겠지만 이전의 패러다임에 대해 미처 제대로 경험하지 못했던 사실들을 간접적으로나마 이해하면서 점점 확신을 갖게 되는 것이 있다. 바로 구조적 분석 설계 기법(SADT: Structured Analysis & Design Technique)으로부터 시작된 정보 기술 패러다임의 변천사는 이전의 패러다임을 뒤엎는 ‘혁명(revolution)이 아니고 진화(evolution)’였다는 사실이다.
국 내 IT 업계의 문제는 이전의 패러다임에 익숙한 엔지니어들이 새로운 패러다임에 대해 피상적인 용어나 개념 정도 살펴본 것을 전부인 양 착각하게 된다. 또한 새로운 패러다임에 편승하거나 가장하다가 또 다른 새로운 패러다임이 등장하면 그 패러다임에 너무 쉽게 편승한다면 이는 뿌리 없는 기회주의(?)가 아닐까 싶다. 혹자는 MDA가 CBD를 대체하는 또 다른 유행 혹은 패러다임이라고 주장하기도 하지만 MDA는 분명히 CBD를 전제로 하면서 SPL이나 SOA (Service-Oriented Architecture) 등과 같은 동시대의 다른 패러다임들과 더불어 CBD를 엄밀히 말하면 ‘컴포넌트를 더 컴포넌트답게’ 혹은 제대로 된 컴포넌트를 만들려고 노력하는 이 시대의 선도 엔지니어들이 합의해 제시하는 미래 소프트웨어 산업계의 비전이다. MDA 개발 패러다임으로의 전환이 IT 업계 특히 소프트웨어 산업계에 가져올 가장 큰 후폭풍은 단순한 개발 방식의 변화가 아니라 전반적인 프로젝트 수행 조직의 역할 변화와 시스템 자산에 대한 인식 전환일 것이다.
마지막으로 OMG의 MDA 작업에 깊이 관여했던 전문가 중의 한 사람인 데이비드 프랑켈(David S. Frankel)의 의미심장한 비유를 소개하면서 이 글을 마무리하고자 한다. 그는 저서 ‘Model Driven Architecture-Applying MDA to Enterprise Computing’에서 데이비드는 MDA에 대해 현재 많은 엔지니어들이나 일반인들이 보이는 회의적인 시각을, 예전에 0과 1의 조합으로 작성하는 머신 코드로 프로그램을 작성하던(Machine-centric computing) 것이 일반적이었고 당시에는 혁신적인 어셈블러가 제안됐던 시절에 비유하고 있다. 당시에 많은 엔지니어들은 ‘어떻게 어셈블러와 같은 고급 언어로 프로그램을 작성할 수 있느냐’ 혹은 ‘엔지니어들은 절대 어셈블러를 개발 언어로 받아들일 수 없을 것이다. 프로그램을 제대로 작성하려면 머신 코드로 작성해야지…’라며 어셈블러에 크게 주목하지 않았다.
그러나 어셈블러는 엄연히 당대를 주름잡는 개발 언어로서 소프트웨어 공학사에 큰 족적을 남겼다. 세월이 흘러 인터넷 기반 시스템 개발이 보편화되고 있는 요즘 만약 여러분의 주변에 있는 어떤 엔지니어가 지금은 거의 특정 분야에서만 한정적으로 사용하고 있는 어셈블러를 가지고 고객관리(CRM) 시스템과 같은 인터넷 기반의 복잡한 업무 처리 시스템을 개발하겠다고 한다면 여러분은 과연 그 엔지니어에게 뭐라고 할 것인가? 데이비드는 주저 없이 그 답변에 대해 ‘제 정신 아니네’라고 단정하고 있다.












플라톤에 의하면 모든 존재는 이상과 본질의 영역인 이데아와 이데아를 모방한 현상, 현상을 모방한 가상(시뮬라크르)의 3단계 인식 전이를 말하고 있다. 우리가 상대하는 실체(여기선 MDA)의 인식 수준이 어느 단계에 있느냐에 따라 제대로 실체를 사용할 수 있을 것이다. 여기서는 실체에 대한 다각적 접근과 우리의 숙제에 대해 고민해 보겠다.

소프트웨어 발전 방향을 미리 예견하기는 무모하거나 용감한 작업이 될 수 있다. 선배의 말을 빌리자면 차세대 채택되는 패러다임을 예측하기란 현재 학계나 커뮤니티에서 논의되고 있는 패러다임들을 제비로 접어서 통에 넣어 흔든 후 하나를 뽑는 것과 같다고 한다. 그만큼 장담하기 힘들고 어려운 부분이다. 하지만 그것은 학계에서 발표되는 패러다임 101 버전의 얘기이지(이번 특집의 주제인 MDA만 하더라도 3년 전에 발표된 내용이다) 실제 산업에서 사용할 수 있는 수준의 차세대 패러다임을 점친다는 것은 메이저 벤더들의 제품 비전만 보더라도 쉽게 알 수 있다(대형 벤더들이 그런 고민은 다 해준다). 물론 ‘시장에서 채택될 것인가?’ 하는 아주 복잡하고 우연적인 변수를 제거했을 때 말이다. 또 다른 방법이 있다 약간의 추리력과 예지력이 필요한데 과거 소프트웨어가 어떻게 발전되었는지 흐름을 잡고 미래를 예측해보는 것이다. 이 방법은 ’메이저 벤더들의 제품 비전’ 보다 앞서 미래를 점쳐볼 수 있다.
MDA의 목적과 실체를 살펴보기 위해 OMG에서 꾸준히 추진해온 OMA(Object Manage ment Architecture)를 살펴보는 것은 도움이 된다. 결국 MDA는 OMG의 주력 표준인 UML과 OMA를 조합해 탄생한 총아이기 때문이다. 또한 MDA를 통해 얻고자 하는 것이 무엇인지, 그리고 다른 ‘제비’들과는 어떻게 다른지를 살펴보는 것은 흥미롭다. 자, 그럼 OMA를 통해 엔터프라이즈 소프트웨어가 어떤 것들을 어떻게 정복하며 발전했는지 살펴보자. 여기서 관전 포인트는 소프트웨어 재사용 단위, 형태를 기준으로 하는 것이 적당하겠다. 그리고 이 재사용성은 표준 혹은 규약으로 형상화된다.

OMA에서 MDA로

라이트 형제가 만든 플라이어호는 F-16 전투기와 너무도 다르다. 과거의 비행기에 비해 현재의 비행기는 너무 강력하고 많은 기능들이 추가되었다. F-16 전투기는 플라이어호부터 F-16 전투기 이전 버전까지의 결과물이다. 즉 F-16 전투기를 이해하기 위해 이전 버전들의 발전 과정을 추적해 보면 아주 잘 이해할 수 있다. OMA와 MDA의 관계가 그렇다.
OMA는 크게 분산 객체 명세와 객체간 원격 호출의 신뢰성, 상호운용성, 이식성 등을 보장하는 ORB가 그 핵심에 있다. ORB를 통해 객체(컴포넌트, 서비스 등으로 대치시켜도 무방하다)를 배포할 수 있었고 배포된 객체는 실행 환경에서 (재)사용된다(객체 표준/규약). 분산객체에 원격호출이 요청되면 그 객체가 가지는 비즈니스를 실행하게 된다. 엔터프라이즈 시스템에서의 비즈니스는 상당히 복잡한 처리과정을 갖는다. 네이밍 서비스를 통해(J2EE에서 JNDI) 타 서비스를 찾아 비즈니스를 연동하기도 하며 트랜잭션 서비스를 통해(J2EE에서 JTS or MTS, TP Monitor) 트랜잭션 처리를 위임하기도 한다. 또한 이벤트/Notification 서비스를 통해(J2EE에서 JMS or IBM MQ Series) 메시징 처리 및 EAI를 위임하는 것 이외에(데이터베이스) 쿼리, 보안, 라이선싱, 객체 저장 등의 서비스를 사용한다. 이런 각 코바 서비스는 비즈니스 처리를 위해 기술적 난점들을 재사용 가능하게 한다. 과거에 이 서비스들을 미들웨어라 불렀다(미들웨어 표준/규약).

<그림 1> OMA 아키텍처 1


객 체 표준과 객체간의 통신의 문제도 각종 기술적인 문제도 점령됐다. 이제 비즈니스 처리에 전념할 수 있다. OMA에서는 비즈니스 처리에 도움을 주는 편의 기능인 퍼실리티(CORBA Facility)를 준비하고 있다. 퍼실리티는 특정 도메인(금융권, 국방, 행정, 모바일)에서 자주 쓰이는 수직적(vertical) 퍼실리티와 소프트웨어 개발시에 일반적으로 사용할 수 있는(데이터 압축, 룰 처리, 워크플로우 처리, 컬렉션 등) 수평적(horizontal) 퍼실리티가 있다. 이로써 개발자들은 일반적으로 사용하는 수평적 편의 기능들과 현재 산업 도메인의 표준으로 정의된 모델인 수직적 편의 기능을 조합해 자신의 애플리케이션에 맞게 최적화, 특화해 개발하게 된다. 물론 도메인 퍼실리티는 CORBA 퍼실리티 프로바이더에 의해 제공된다. 그리고 이를 바탕으로 <그림 2>의 도메인(객체)이 형상화, 정규화된다(레이어가 연상되지 않는가?). CORBA 도메인은 금융권부터 국방, 통신에 이르기까지 상당히 광범위하게 정의되어 있으며 다루고 있는 영역도 상당히 정밀하다. 사실 이 도메인이나 퍼실리티는 해당 분야의 기술자들이 오랜 경험을 통해 축적된 지식과 노하우를 통해 연구하여 설계한 모델이다(산업 도메인 표준/규약).
OMA 아키텍처는 이렇게 가장 기본이 되는 것부터(밑에서부터) 하나씩 표준화 작업을 수행했으며 이 표준화된 규약에 맞춰 개발된 COTS를 이용해 좀 더 검증되고 안정된 플랫폼을 제공하려는 비전을 갖는다. 표준화 작업으로 점령하는 과정은 가장 근본적이고 재사용성이 강한 대상부터 시작한다(ORB -> Service -> Facility). 그리고 그 종국에는 도메인이 있다. 도메인은 해당 도메인에 가장 안정적인 아키텍처, 통신 프로토콜, 자료구조(경우에 따라서 데이터베이스 논리 테이블(Logical Data Model)까지) 등을 제시하고 있다. 따라서 개발자는 먼저 이미 벤더에 의해 제공되는 도메인을 도입하여 자신의 시스템에 맞게 애플리케이션을 최적화 시키고, 도메인이 제공하지 않는 자신의 시스템에 특화된 부분만 개발하는 작업으로 업무를 단순화할 수 있다.
그렇다면 왜 OMA 브레인들을 포함하여 다른 표준화 기관에서는 표준과 계약하고 싶어할까? 계약은 ‘계약에 의한 설계(Design by Contract)’를 통해 각 이해 당사자들이 상대방에 대한 서비스 제공과 같은 책임과 의무를 기술함으로써 대상에 대한 ‘How’가 아닌 ‘What’에 집중하게 한다. 표준은 반복해서 재사용할 수 있는 대상에 대한 ‘계약’으로서 지식과 경험에 대한 결과를 공유하려는 목적을 갖는다. 이를 통해 상호운영성, 통합, 연동의 문제에서 자유로울 수 있다. 그렇다면 OMA와 MDA가 어떤 연관관계가 있을까? <그림 2>에서 분류된 도메인 오브젝트와 <그림 3>에서 MDA의 화살표로 지시되는 수직적 도메인과 매우 일치하지 않는가? <그림 2>의 도메인 오브젝트는 OMA 도메인 오브젝트 중 그림 구성상 일부만 표현한 것이다. 마치 OMA의 도메인 오브젝트들을 MDA 마지막 레이어에 감싸놓은 것 같다. OMA의 경우 IDL(Interface Definition Language)로 도메인 오브젝트들을 정의한 반면 MDA는 모델로 대체하고 있다. 물론 이 연관 관계에는 OMG의 다른 표준인 MOF나 UML 등의 내용은 차치한다. MDA의 목표는 OMA의 그것과 다르지 않다. 개발을 위한 최대한의 플랫폼을 준비해 놓고 각 도메인 전문가들이 완성한 표준 모델들을 제공하여 개발에 필요한 최대한의 생산성을 극대화시킨다. MDA는 이제 CORBA만을 고집하지 않고 J2EE, 닷넷, 웹 서비스 등 다른 플랫폼을 아키텍처에 포함시켰다. 그러므로 OMA에서의 개발 프로세스와 매우 유사한 방법으로 개발 가능하다. 즉 MDA 산업 도메인 모델을 기반으로 현 시스템이 필요한 모델들을 추출한 후, 최적화시킬 부분을 커스터마이징한다. 끝으로 도메인 모델에 표현되지 않은 특화시켜 추가 개발해야 할 모델들을 모델링하므로 PIM을 완성한다. 다음 MDA 프로세스에 따라 PSM으로 모델 전이(Model Transform)를 하고 해당 플랫폼에 대한 소스코드를 생성한다. PIM → PSM → Code로의 전이는 매우 훌륭하지만(이 전이 과정은 일반적 방법론과 매우 잘 일치한다. 박스 기사 참조) 또 하나 상기해야 할 부분은 나와 똑같은 도메인 선배 전문가의 경험과 지식으로 표준화된 도메인 모델들을 잊지 않아야 할 일이다.

<그림 2> OMA 이키텍처 2


<그림 3> Model Driven Architecture


SOA 시대의 도래

엔터프라이즈 시장에서 소프트웨어 기술 추이는 분산객체(CORBA) → CBD 플랫폼(J2EE, 닷넷) → SOA로 흐르고 있다. 특징을 비교하면 재사용성이 높고 실세계적 체계를 갖는 객체지향 단위의 오브젝트에서 좀 더 굵은 입자(coarse grained)를 갖고 객체간의 상호작용을 더욱 느슨하게(loosely coupled) 하여 변경관리가 용이할 수 있는 컴포넌트가 대세가 된다. 그렇다면 컴포넌트와 서비스와는 어떤 차이가 있을까? 좀 더 구체적으로 J2EE에 배포되어 ‘서비스 되는’ 빈과 SOA(Service Oriented Architecture)의 서비스와는 어떤 차이가 있을까? SOA는 CORBA 오브젝트와 EJB를 서비스의 한 형태로 보고 있다. 단지 차이라면 그 자체가 아니라 동작되는 운용방식과 세계관에 있다. 결론만 말하면 비즈니스적이고 느슨한 관계를 가져 환경 통합이 용이하다.
웹 서비스와의 관계는 어떠한가. SOA와 웹 서비스와의 관계는 마치 애자일 방법론과 XP의 관계와 유사하다. SOA의 한 실체가 웹 서비스이고 웹 서비스는 SOA 아키텍처를 따른다. 웹 서비스와 SOA의 차별성과 개연성을 놓고 논란이 있지만 일반적인 관계는 이렇다. 즉 필자에게 객체와 컴포넌트, SOA의 차이를 규정하라고 한다면 그 기준을 크기(granularity)에 따른 책임에 두고 설명하고 싶다(단지 필자의 접근법이다). 객체는 기존에 알고리즘과 자료구조를 분리해서 다뤘던 프로그래밍 패러다임에서 실제 행위와 상태를 갖으므로 그 대상의 책임이보다 분명하게 정의할 수 있었다. 객체가 알고리즘과 자료구조를 추상화(facade) 한다. 하지만 객체간의 복잡한 관계들은 관리적 측면에선 단점이 된다. 컴포넌트는 좀 더 큰 범위에서 인터페이스를 통해 실제 사용하는 기능에 집중할 수 있게 한다. 컴포넌트가 객체들을 추상화한다. 서비스는 컴포넌트의 처리를 비즈니스 단위로 묶어 작업 단위를 상위 레벨에서 캡슐화해 접근하게 한다. 서비스가 컴포넌트을 추상화한다.
그러므로 사용자는 소비자로서 더욱 자신의 비즈니스 흐름에 집중할 수 있고 필요에 의해 비즈니스 흐름에 필요한 서비스를 선택하게 한다. <그림 4>는 SOA의 큰 범위에 등장하는 아키텍처적 레이어이다. 애플리케이션 아키텍처는 하나 이상의 서비스 공급자로부터 제공되는 서비스들을 찾아 비즈니스 프로세스 속으로 각 서비스들을 통합한다. 서비스 아키텍처는 실제 서비스를 처리하는 컴포넌트 아키텍처와 소비자 사이에 브릿지 역할을 함으로써 서비스를 위한 연동, 처리 관계, 상태 등을 관리해준다. 컴포넌트 아키텍처는 실제 서비스 구현을 담당하며 레거시 시스템이 좋은 후보가 될 수 있다. 이런 기반 구조에서 다음과 같은 효과를 얻을 수 있다.

◆ 프로세스 중심

◆ 플랫폼 독립적 애플리케이션 통합

◆ 느슨하게 결합된 메시지 지향

◆ 메시지 및 프로세스 상태 관리

<그림 4> SOQ의 3계층 아키텍처


SOA 의 기본 세계관은 이미 개발된 각 기업의 컴포넌트들을 통합하기 위해 규격화·일반화한 서비스 컴포넌트를 상호운용할 수 있도록 함으로서 개발자가 중복 개발하는 것을 지양하는 것이다. 즉 네이버에서 개발된 블로그를 엠파스에서 똑같은 작업을 반복할 필요없이 네이버 블로그 ‘서비스’를 사용한다면 개발자의 노동력을 1/n배 감소할 수 있을 거라는 가설이다. 하지만 SOA는 이처럼 단순하지만은 않다. SOA는 보다 비즈니스 프로세스 관점에서 접근한다. 비즈니스 프로세스적 접근이란 마치 C 언어에서 main 함수를 시초로 다른 함수들을 호출해 프로그램이 진행되듯이 소비자가 각 서비스를 이용해 하나의 비즈니스 프로세스를 수행하는 방식을 말한다. 물론 하나의 서비스는 독립성을 갖지만 서비스는 상호 이용할 수 있으며 하나의 비즈니스 서비스는 서비스들의 조합으로 완성된다. 이를테면 <그림 5>에 표현된대로 ‘항공권 구매’란 비즈니스 서비스는 컴포넌트의 복잡한 조합에서 작업 흐름(프로세스)에 따라 호출되는 서비스들의 집합으로 관계를 단순화시킬 수 있다(인터페이스 중심의 패러다임에서 프로세스 중심으로의 패러다임 전이). SOA가 행운의 ‘제비’가 된다면 개발자는 서비스 마켓 플레이스에 진열된 양질의 서비스를 선택해 비즈니스 프로세스의 구성요소들을 구매한다. 만약 원하는 서비스가 없다면 개발해 구성 서비스에 추가시킨 후 서비스로 제공하게 될 것이다.
여 기서 또 하나의 기대효과를 얻을 수 있다. 이렇게 서비스들이 빌트인 된 상황에서 각 서비스들의 상호작용을 마치 UML 시퀀스 모델링하듯이 비즈니스 프로세스 수행 시나리오를 작성하게 된다는 것이다. 웹 서비스의 WSFL(Web Service Flow Language)과 WSCL (Web Service Composition Language)이 이런 역할을 한다. 유사 기술은 BPM의 BPML(Business Process Modeling Language), 마이크로소프트의 XLANG가 있다.

<그림 5> SOA의 프로세스 중심적 서비스 조합의 예


디자인 완전정복, 패턴언어

필자는 본지에서 지난 2003년 3월부터 3회에 걸쳐 분산 프레임워크에서의 패턴 언어를 소개한 바 있다. 패턴의 대부분의 패러다임은 패턴의 아버지인 건축학자 알렉산더로부터 제안된다. 바꿔 말하면 알렉산더와 그의 연구소는 패턴이란 종교의 메카격이 된다. 알렉산더는 1987년 ‘A Pattern Language’란 책을 통해 ‘유기적 건축양식’이란 패턴이론을 소개했다. 소프트웨어에서도 거의 유사하게 ‘패턴언어’란 패러다임은 적용된다. 패턴언어에서의 패턴은 독립적으로 존재하지 않고(“No Pattern is an Island”) 패턴간의 조밀한 응집도를 구성할수록 그 마법과 위력은 막강해진다(“a dense composition of patterns”).
기존의 Pattern Vocabulary, Pattern System의 패러다임과는 달리 패턴언어는 도메인에 종속적인 성격을 갖는다. 즉 MDA에서 수직적/수평적 도메인이 모두 디자인 차원에서 언어화(Pattern Language)될 수 있는 대상이 된다. 하나의 도메인은 하나의 언어가 된다. 왜 도메인을 언어란 개념으로 다루고 있을까? 초기 GoF 패턴 개념은 패턴 용어집(Pattern Vocabulary)의 패러다임이었다. 사람은 정확하고 확실한 의사소통을 위해 잘 정의된 용어집이 필요하고 이 용어집의 어휘가 많을수록 의사전달자로서 유리한 자산을 확보한 셈이 된다. 디자인 패턴에서 하나의 어휘는 하나의 패턴에 해당한다. GoF의 패턴으로 설계 문제를 해결하는 방식은 문제영역(패턴의 목적과 동기)을 기준으로 패턴의 목록을 살핀 후 가장 적합한 패턴을 선택해 적용하는 방식이다. 이때까지 패턴간의 응집도는 상당히 일반적이며 그 밀도는 낮았다. 패턴언어의 개념에서는 좀 더 축적된 용어들을 체계화해 하나의 언어 영역을 구성한다고 본다. 따라서 하나의 도메인을 이해하기 위해서는 그 도메인을 구성하는 언어체계(그 도메인의 패턴과 패턴간의 관계)를 이해하는 것이 필요하며 이 언어체계를 잘 이해할수록 그 도메인의 문제영역 풀이가 쉬워진다.
그렇다면 도메인에서 패턴간의 응집도는 어떻게 구성될까? 일반적으로 그 도메인의 문제영역이나 아키텍처 구성을 기준으로 패턴간의 응집도가 구성된다. <그림 6>의 J2EE 패턴 언어를 보자. J2EE는 크게 presentation, business, integration tier로 구성된다. 처음 클라이언트로부터 HTTP 요청이 들어오면 Decorating Filter 패턴은 HTTP 요청정보를 분해, 수집(필터링) 한다. 그 후 Front Controller 패턴은 사용자 인증이나 로깅 같은 공통적인 처리를 하고 View Helper 패턴에게 다음 처리를 위임한다. View Helper는 서블릿 처리와 뷰에 관한 처리를 분리시키며 Business Tier의 Business Delegate 패턴으로 처리를 위임한다. Business Delegate 패턴은 서비스 컴포넌트의 복잡한 구조를 은닉하고 실제 비즈니스 처리를 담당하는 Session Fa ade 패턴으로 처리를 위임한다. Session Fa ade 패턴은 비즈니스 처리를 하면서 DB 접속을 시도할 경우 Integration Tier의 Data Access Object 패턴을 사용하므로 클라이언트 요청을 처리한다. 이렇게 패턴간의 응집도를 통해 처리의 전이가 발생하며 각 패턴은 해당되는 레이어의 문제들을 하나씩 해결한다.

<그림 6> J2EE 패턴언어


이 때 각 패턴들은 서로를 이용하기도 하고 같은 목적을 갖는 패턴들과 서로 경쟁하기도 한다. 또한 아키텍처 패턴과 디자인 패턴, 구현 패턴간의 계층적 포함 관계도 갖는다. 이렇게 패턴언어는 해당 도메인의 문제를 해결할 수 있는 여러 모델들을 제공하고 있다. 패턴언어에서 문제영역 해결방식은 개발자가 자신의 문제영역에 가장 적합한 패턴들의 집합을 조합하여 패턴의 구현 방식에 따라 시스템을 구성하면 된다. 즉 패턴언어의 프로세스는 이렇다. 첫째 개발자는 해당 도메인의 패턴언어를 살펴본 후 개발에 필요한 패턴 집합을 선택한다. 둘째, 선택된 패턴들을 구현한다. 셋째, 패턴으로 채워지지 않은 문제영역들을 구현한다. 패턴언어에서 재사용 자산은 도메인 개발자의 디자인 경험과 지식이고 그 결과물은 MDA에서의 모델과 다르게 패턴으로 형상화된다.

MDA, SOA, 패턴언어

눈치 빠른 독자라면 MDA, SOA, 패턴언어간의 유사성을 느꼈을 것이다. 이 세 기술은 각 공통적으로 축적 가능한 지적 자산을 최대한 모으고 축적된 자산들을 일정한 틀에 의해 배치시키고 저마다 하나의 문제영역을 해결하기 위한 최대한의 준비를 빌트인 하고 있다. 사용자의 몫은 이들을 잘 선택하고 조합해 이용할 수 있는 가치들을 최대한 획득하는 것이다.
MDA는 도메인 모델을 제공하고 모델을 재사용하게함으로서 모델을 통한 재사용성의 극대화를 추구한다. SOA는 서비스 형태를 갖는 이미 개발되어 컴포넌트들을 통합하여 하나의 비즈니스를 완성한다. 패턴언어에서는 하나의 도메인을 나타내는 패턴들의 조합들 중에 자신이 필요한 패턴의 집합을 선택해 시스템의 디자인을 구성한다. 하지만 이들의 사용 형태가 다른 만큼 이들이 채택한 기술도 차이가 있다. MDA는 Executable UML, CWM, OCL 등을 통해 모델을 정의하고, 정의된 모델들을 QVT(Query, View, and Transformations)함으로서 구현을 완성한다. SOA는 웹 서비스를 기준으로 XML을 십분 활용해 통신을 위한 프로토콜을 SOAP으로 채택하고 WSDL을 통해 서비스 명세를 하고 UDDI를 통해 서비스의 생성, 기술, 발견, 통합을 가능하게 한다. 또한 이렇게 갖춰진 환경들을 WSFL이나 WSCL을 통해 비즈니스 차원에서 워크플로우 개념으로 조합한다. 패턴언어의 경우는 좀 불행하다. 패턴의 구성 원리와 패러다임만 있을 뿐이지 기술적인 실체가 없다.
해당 도메인의 지적 자산을 모으기 위해서는 튼튼한 아키텍처와 표준이 필요하다. MDA는 도메인과 플랫폼, 이들을 유기적으로 엮을 수 있는 MOF, CWM, UML 같은 기술로 모델에서 실행 가능한 코드로의 전이를 보장하고 있다. 즉 모델링을 기반으로 한 프로세스적인 측면이 강하다. 또한 각 도메인별로 수직적인 도메인 모델들을 포진시킨다. SOA의 경우는 서비스를 독립적이고 서로간의 느슨한 결합도를 유도하는 시스템 배치, 구성, 효과적인 사용 방법에 주안점을 두고 아키텍처를 만들었다. 패턴언어의 경우 패턴언어 자체가 아키텍처를 유도하고 있다. 그러므로 MDA, SOA는 엔터프라이즈 시스템 구축에 보다 적합하며 패턴언어는 프레임워크나 자체 솔루션을 구축하기에 적합하다. 이것이 패턴이 크게 뜨지 못하는 장애요인 중에 하나이다.
끝으로 각 기술들이 꿈꾸는 유토피아는 어떤 것일까? MDA는 소프트웨어의 모든 모델들이 MDA를 통해 만들어져서 모든 마켓 플레이스에 MDA 모델들이 진열되길 바랄 것이다. SOA의 비전은 개발을 원하는 모든 컴포넌트들이 서비스화되어 컴포넌트 조합으로 프로젝트가 끝나버리는 모든 마켓플레이스에 서비스가 진열되는 세상을 목적할 것이다. 패턴언어의 경우는 어떠한가? 패턴으로 표현할 수 있는 모든 디자인의 문제들이 패턴언어로 점령되기를 바랄 것이다. 그리고 이를 통해 좀 더 실체적인 프레임워크나 플랫폼으로 구체화되길 바랄 것이다.

우리의 과제

바스티유는 무너져도 앙시앙 레짐은 무너지지 않았다.

필자가 C 프로그램을 익히고 자바 프로그래밍을 막 배울 때의 일이다. 필자의 코드를 본 선배가 ‘C-tic한 자바프로그래밍’이라고 놀리던 일이 기억난다. 구조적 습관과 접근법을 아직 벗어나지 못한 상태에서 객체지향 프로그래밍을 하니 당연히 자바로 구조적 프로그래밍을 할 수 밖에 없다. ‘C-tic한 프로그래밍?’ 그때 필자의 코드들은 문법은 자바를 사용하지만 구성과 형식은 C 언어였다. 무엇이 문제였을까? 객체지향 언어를 사용하지만 구조적 프로그래밍이란 세계관을 버리지 못했던 탓이 컸다. ‘바스티유는 무너져도 앙시앙 레짐은 무너지지 않았다.’ 프랑스 혁명의 성공으로 봉건제도의 진원지인 바스티유를 함락시켰지만 앙시앙 레짐(old regime:구체제)의 습속은 여전히 남아있는 현상을 보고 한 지식인이 한 말이다. 우리는 새로운 패러다임을 접할 때 그 패러다임이 원하는 세계관은 무시한 채 그 패러다임의 용법에만 관심을 갖는 경우가 많다. 마치 필자의 ‘C-tic한 자바프로그래밍’의 경우처럼… ‘새 술을 새 부대에 담으라’는 말이 있다. 진정으로 체화되어 그 패러다임을 십분 활용하기 위해서는 그 패러다임이 원하는 방식과 접근법을 따라야 한다. 요즘 많이 거론되는 ‘내재적 접근법’으로 MDA란 새 기술을 대하는 태도가 필요하다.
세계관과 더불어 문화를 바꿔야 한다. XP를 도입하는 조직이 힘든 이유는 XP의 기민하게 상호작용해야 하는 참여자간의 문화를 바꾸기 힘들기 때문이다. 특히 MDA처럼 개발 단계가 극도로 축소되는 프로세스를 따르는 경우는 개발 참여자의 역할이 크게 구조조정된다. 즉 MDA는 개발 조직 체계를 바꾼다. MDA를 도입한 프로젝트에서 구성원의 시스템을 문화적인 부분까지 고려하여 바꾸지 않는다면 성공을 담보하기 힘들다.

Learning Curve를 고려해야…

하나의 기술을 프로젝트에 적용하기 까지 우리는 충분히 준비하지 못한 상태에서 프로젝트에 투입되곤 한다. 마치 소총 격발법 한번 읽어보고 전쟁터에 나간다고 비유하면 억지일까? 조금 여유있는 프로젝트의 경우 그 제품에 대한 교육도 받고, 파일럿도 하고 충분히 숙련과정을 거친 상태에서 제품 컨설턴트까지 대동하여 프로젝트에 투입하는 경우도 있다. 하지만 대부분의 경우는 이렇게 행복하지 않다. 짧은 시간 안에 스파이크 솔루션을 해보고 예제 실습 및 약간의 테스트를 거치고 곧바로 프로젝트에 적용한다. 필자의 동료는 더 고약한 경우를 당했다. PHP 프로젝트에 투입됐는데 두꺼운 PHP 책을 한권 주고 하루의 여유를 주더니 다음날부터 바로 코딩을 시켰다고 한다. 당연히 제대로 프로젝트가 끝날 수 없다. 이정도 되면 ‘내재적 접근법’은 아예 시도조차 못한다.
MDA는 모델링을 통해 프로세스 전반을 아우르는 상당히 거대한 범위를 다루고 있다. 따라서 학습해야 할 부분도 상당히 많다. 이를테면 PIM을 시스템 독립적으로 설계해야 하는 기술과 PIM에서 PSM으로 전이시키기 위해 고려·설정해줘야 하는 것들 타겟 플랫폼에 대한 이해·실행 가능한 코드를 생성할 때 코드 최적화를 위해 고려해야 하는 사항, 이후 테스트, 산출물 작업에 이르기 까지 알아야 할 것도 많고 배워야 할 것도 많다.
그러므로 현재 프로젝트 성공을 위해, 그리고 성공적일 다음의 MDA 프로젝트를 위해 체계적이고 전략적인 학습과정이 필요하다. 바쁘고 촉박한 프로젝트 환경에서 프로젝트 성공을 위한 충분한 학습을 하자는 것이 아니다. (물론 가장 바람직한 경우겠지만) 프로젝트를 통해 학습할 수 있는 기회를 가능한 만들어 다음 프로젝트에서는 좀 더 숙련된 기술을 보유하려는 노력, 함부로 사용하려 하지 않는 노력이 필요하다. ‘운전은 한다. 차는 모른다’라는 광고 카피가 있었다. ‘MDA를 프로젝트에 적용시켰다. MDA는 잘 모르겠다’ 엔지니어인 우리에게 적용되기엔 너무 곤란한 경우이다.

적용을 위한 타당성, 적합성 검증 우선

좀 규모가 큰 프로젝트에서는 ‘단지 이 기술이 대세’이기 때문에 사용하려는 경우가 많다. 물론 대부분의 개발자들은 이런 기술, 플랫폼, 운용환경 선택의 의사결정 기회가 없지만 필자가 보아온 몇 가지 J2EE 프로젝트는 ‘Apache + Tomcat’만으로도 충분히 괜찮은 플랫폼인데 J2EE란 트랜드를 따라 무겁고 비싼 J2EE를 이용한다. MDA가 담당할 수 있는 영역은 상당히 광범위하다. 임베디드 시스템에서 국방, 금융권에 이르기까지 MDA로 못하는 것이 없다고 봐도 무방하다. 하지만 잊지 말아야 할 질문이 있는데, MDA가 현재 프로젝트를 위해 필요조건을 만족하는가, 충분조건을 만족하는가이다. 타당성이 확인되지 않은 기술을 사용하는 프로젝트는 개발자의 목적의식을 절감시키는 요인이 된다.
필자가 소개한 기술들은 소프트웨어 진화에 일익을 할 것이다. 그러므로 대세에 발맞추는 것도 개발자로서 좋은 태도이다. 하지만 조금 약아 보이더라도 실용적 입장을 취하는 것은 나쁘지 않다. 끝으로 MDA가 꼭 뽑히는 ‘제비’가 되기를 개인적으로 바란다.

[ 전형적인 방법론 접근법이 이입된 MDA ]

전형적인 방법론에서의 모델링 순서는 ‘비즈니스 모델 → 유즈케이스 모델 → 분석 모델 → 설계 모델 → 구현 모델’ 단계로 구체화된다. 유즈케이스 모델은 시스템 요구사항을 도출하기 위한 모델이므로 일반적인 시스템을 추상화, 개념화 작업과 거리가 있기 때문에 현재 논의에서 제외시키자. 비즈니스 모델은 플랫폼에 독립적이며 동일 도메인의 다른 환경에서도 재사용가능하다. 이를테면 A회사의 윈도우 환경, C/S 기반의 인사관리 시스템의 비즈니스 모델은 B회사의 유닉스 환경, 자바 기반의 인사관리 시스템에도 적용할 수 있다. 분석모델도 플랫폼 독립적이다. 하지만 동일 도메인간의 재사용은 불가능하며 닷넷으로 구축된 시스템의 분석 모델을 J2EE 플랫폼으로 적용 가능하다. 디자인 모델은 플랫폼 종속적이다. 하지만 운영체제 같은 시스템에 독립적이다.
그렇다면 MDA의 모델들과 비교해 보자. <표1>은 RUP의 모델들과 MDA의 모델들을 비교한 대차대조표이다.

<그림 1> RUP 모델링과 MDA 모델링



출처 : 마이크로소프트웨어
:
Posted by 뽀기
2007. 2. 27. 13:22

구조적 VS 객체지향적 그거/Tech2007. 2. 27. 13:22

클래스(Class)란, 객체지향 프로그래밍에서, 객체 내부의 데이터 구조(필드, 멤버변수)와 그 조작(메소드, 멤버함수)을 기술한 것을 말한다.

무슨 뜻인지 이해가 가는가? 이해가 간다면, 이미 클래스에 대해 알고 있는 독자임에 틀림없다.

그렇다면, 클래스의 정의부터 짚어보도록 하자.

① 객체지향 프로그래밍을 기본으로 한다.
② 객체 내부의 데이터 구조(필드, 멤버변수)와 그 조작(메소드, 멤버함수)를 기술한 것이다.

객체지향 프로그래밍은 뭐고, 객체는 뭐고, 필드, 메소드 등등은 무엇인지... 처음 듣는 독자에게는 벽으로 느껴질 것이다. 자, 그렇다면 이 벽을 뛰어넘을 약간은 지루할지 모르는 시간을 가져보도록 하자.

구조적 프로그래밍 VS 객체지향 프로그래밍 그리고 객체와 클래스

구조적, 객체지향, 객체 클래스 등의 바로 감각적으로 느껴지지 않는 단어들의 나열이 혼란스럽게 느껴질 수도 있겠다. 하지만, 이것들을 하나하나 풀어 보도록 하자.

프로그래밍에 대한 방식을 나누는 가장 일반적인 방법이 구조적 프로그래밍(Structured Programming)과 객체지향 프로그래밍(Object Oriented Programming)이다.

먼저, 구조적 프로그래밍은 프로그램을 작성할 때, 알고리즘을 세분화해 계층적인 구조가 되도록 설계하는 프로그래밍 방법으로, 프로그램은 세 개의 기본구조(순차, 선택, 반복)만으로 구성된다. 커다란 문제를 해결하기 쉬운 작은 크기로 나누고 각각을 해결해 커다란 문제에 대한 해결책으로 만드는 분할 정보라는 기본적인 원칙을 갖는다.

다시 말하면, 구조적 프로그래밍은 C 또는 파스칼 같은 언어에서 주로 사용된 프로그램 방법으로, 기본적으로 순차적으로 접근된다는 것, 그리고 데이터와 그 데이터를 처리하는 방식이 분리되어 있다는 것에 이해의 중점을 두기 바란다. 이것은 모듈별로 프로그래밍한다는 의미로는 장점일 수 있으나, 새로운 데이터 형태의 도입이나, 기능에 대해 변경사항이 발생할 경우 유지보수와 재사용성 측면에 상당한 어려움을 갖게 된다.

객체지향 프로그래밍은 프로그램을 객체라는 모듈 단위로 작성하는 것을 말한다. 객체란, 하나의 데이터 구조로 처리할 데이터 값과, 그것에 관한 절차를 합한 것이다. 객체지향 프로그래밍은 객체지향 언어가 제공하는 캡슐화, 상속성, 다형성 등의 기능을 활용해 프로그래밍 하는 것을 말한다.

여러 개의 생소한 단어가 한꺼번에 나타나 여러분을 혼란케하더라도, 좀더 인내심을 갖자.

간단히 말하면, 객체지향은 객체를 기본으로 하는 프로그래밍 방식이다. 구조적 프로그래밍이 데이터와, 그 데이터를 처리하는 방식을 분리했다면, 객체지향 프로그래밍에서는 객체라는 개념을 통해, 데이터와, 그 데이터를 처리하는 방식을 통합하였다. 여기서, 일반적으로 객체에서의 데이터를 멤버 변수 또는 데이터 필드, 그리고 그 데이터를 처리하는 방식을 멤버함수 혹은 메소드라고 한다.

클래스는 객체지향 프로그래밍에서 객체 내부의 데이터 구조(멤버변수)와 그 데이터에 대한 기능(멤버함수)을 기술한 것을 말한다.

자, 여기서 독자들은 클래스와 객체에 대한 관계가 의심스러울 것이다. 그러면, 클래스, 객체, 객체지향 프로그래밍에 대한 관계를 예를 들어 살펴보자.

먼저, 우리가 로봇이 나오는 게임을 만든다고 가정해보자. 여기서 우리가 객체로 구성하게될 대상이 로봇이 될 것이다. 앞에서 말했듯이, 객체에는 멤버 변수와, 메소드로 구성되어 있다. 우리의 게임에서 로봇이라는 객체에는 어떤 멤버변수가 있어야 할까? 먼저, 이름이 필요할 것 같다. 그리고 색깔과 에너지, 일반무기, 특수무기 그리고 현재 위치를 나타내는 좌표 같은 멤버변수가 필요할 것 같다. 멤버변수는 이 정도로 하고, 멤버함수는 무엇이 필요할까? 먼저 움직임을 나타내는 함수가 필요할 것 같다. 그리고, 일반무기 발사, 특수무기 발사, 에너지 충전 등의 멤버함수가 필요할 것 같다. 자, 지금 우리가 나열에 대한 것을 정의하고, 세부사항을 구성한 틀을 바로 클래스라고 한다.

그렇다면 객체는 무엇인가? 객체는 실체다. 즉, 클래스가 실체화 된 상태를 나타내는 속성과 행위를 묶은 것을 말한다. 실제로 존재하는 대상으로 클래스는 개체가 어떤 특성과 행위를 하는지에 대한 서술을 담고 있다. 이러한 클래스를 실체화시킨 것이 객체다. 따라서 객체는 클래스와 달리 실제 기억장소에 만들고, 사용되며, 소멸된다. 그리고 객체지향 프로그래밍에서 프로그램은 객체간의 메시지를 교환하는 방식으로 수행된다.


로봇 클래스

로봇 객체

멤버변수

이름

로봇 1호

로봇 2호

로봇 3호

로봇 4호

로봇 5호

로봇 6호

로봇 7호

로봇 8호

색깔

에너지

일반무기

특수무기

현재 위치

멤버함수

앞으로 가기

뒤로 가기

위로 가기

아래로 가기

일반무기 발사

특수무기 발사

에너지 충전

다시 말해서, 로봇 설계도는 클래스가 되는 것이며, 그 설계도에 따라 우리가 로봇 클래스를 메모리에 생성하여 8개의 로봇을 만든 다면, 그것은 객체가 되는 것이다. 즉, 클래스는 객체가 가져야 하는 속성과 행위를 기술한 것을 의미하며, 객체는 메모리에 생성되어 실제 구현된 것을 의미한다.

출처 : naver blog

:
Posted by 뽀기
2007. 2. 23. 10:24

자바서비스넷의 기술자료 링크 그거/Tech2007. 2. 23. 10:24

안녕하십니까? 자바서비스넷 이원영입니다.
자바서비스넷(
http://www.javaservice.net) 커뮤너티에 올라온 글들 중 의미있는 기술자료
링크를 모아 보내드립니다.

또한 자바서비스넷이 (주)자바서비스컨설팅(
http://www.javaservice.com)으로 거듭나면서
저희들이 직접 개발한 국산 WAS성능관리툴 제니퍼 2.5를 소개드리고자 합니다. 주위에 알려주시고,
직접 사용해 보신 후 신랄한 피드백을 부탁드립니다.

[WAS 장애진단/통합성능관리툴 제니퍼 2.5 의 핵심기능]
http://www.javaservice.net/~java/bbs/read.cgi?m=resource&b=jscnews&c=r_p&n=1127241452

[자바서비스 성능진단 컨설팅 자료모음]
20  모사이트_메모리장애분석.doc
19  성능테스트단계에서의 제니퍼 활용.doc
18 [인터넷, 빈번한 Full GC로 인한 성능저하 사례]
17 [인터넷, 전반적으로 서비스 성능이 우수한 사례]
16 [인트라넷업무, 서비스 큐잉장애에 대한 병목구간 분석 사례]
15 XX사 OO시스템 개발단계에서의 장애요소분석
14 HH사 OOO시스템 성능진단 보고서
13 모모 사이트 성능진단보고서
12 모사이트 모시스템 성능 장애진단 결과보고서
11 OOO사 XXX 시스템 성능진단 결과보고서
10 BB사 XXX 시스템 성능장애진단 컨설팅 보고서
6 다운사이징 벤치마크테스트 및 플렛폼 이관 검증 컨설팅
4 성능테스트를 통한 한계 동시단말사용자 수 측정

[자바서비스넷 최근 IT 소식]
915 2010년의 기술과 산업
1071 선 "모든 SW 무료 제공" 선언
1022 CodeHaus releases Mule 1.0, an open-source ESB
953 차세대 데이터 베이스 프로그래밍 - SDO
987 티맥스, 국내 WAS 시장서 1위 차지
776 참여정부의 전자정부 로드맵

[프로젝트 실무 컨설팅]
1478 S/W아키텍트에 대한 권고
1363 긴급! 리소스 누수 현상 ㅜ.ㅜ
1341 바람직한 작명규칙법을 알고 싶습니다.
1316 [EJB]Stateless Session Bean의 갯수와 WAS 성능에 관하여
1309 EIP솔루션 선택에 대하여
1286 대용량 파일의 Sort관련
1263 웹 애플리케이션의 성능 분석
1230 웹사이트 개발시 필요한 문서 목록
1209 "MVC개발자 분리", "JUnit이용한 테스트"
1206 웹서비스(Web Service)가 뭔가요???
1174 오라클 cursor가 계속 쌓여만 갑니다. --;
1150 Unique number생성 문제점(WAS 3대 사용시)
1123 동시접속자 문제
1106 [질문] 코드성데이터의 클래스생성..
1062 OR Mapping 사례
1039 [질문] struts 프레임워크에 대해서...
997 LoadRunner 성능 테스트 방법론 자료
960 웹에서의 프레임웍 추천부탁드려요~ ^_^
933 자바의 이해하지못하는 프레임웤에 대하여
920 prepareStatement. 사용시 문제점...?
873 Hibernate?
845 리포팅툴에 관련된 질문입니다. 추천좀 해주세요.
786 인트라넷 환경 시스템 구축관련
688 여러 쇼핑몰들과의 Data교환
677 비동기 메시징과 동기 메시징에서의 queue처리방식?
644 J2EE 개발 툴, 방법론 최상의 조합은??

[성능관련 기술자 커뮤니티]
54 김균홍 대리님의 Mixed Throughput EQ.

[기술적 이슈에 대한 토론공간]
440 AJAX의 미래..
404 SOA 의 실증적인 모습이 무엇인지 궁금합니다.
397 ibatis를 대규모 프로젝트에 사용할때 문제점이 있을까요?
392 "차세대시스템"에 걸맞는 Application Architecture
342 이미 구성되어 있는 Server간(종류에 상관없이) file을 교환하게 만들기
332 Framework과 Architecture의 차이점을 설명해주세요
290 코드 인스펙션을 하시나요?
277 지능형 프로젝트 관리 시스템을 도입하려고 하는데요
218 Don't use StringBuffer!! Use String!!
199 비즈니스 컴포넌트에 대한 개인적의견
187 Re: 대량 데이타 온라인 배치성 업무
142 .NET J2EE 환경에 비하여 최대 331% 성능 입증
123 채팅서버구현 -쓰레드 동기화문제

[General Java Programming Tips]
683 실시간 접속 회원 현황 구현에 대해서...
673 이유를 모르겠습니다. (묵시적 캐스팅)
574 자바로 구현된 QuickSort 소스구합니다.
531 이런 알고리즘 아시는 분 ...
506 Generics in C#, Java and C++
503 자바 프로그램의 메모리 사용량 확인방법
496 [Book] J2EE™ Development without EJB™
495 [Book] Thinking in Java, 3rd ed. Revision 4.0
490 Functional programming in the Java language
467 자바 소스 내에서 다른 자바 소스를 컴파일 하는 방법이 궁금합니다.
453 아래 JAVA Non-blocking I/O에 관련된 자료입니다.
447 혹시 NIO에 대한 입문자료(?) 있을까요?
422 Thread Pool에 도전기
420 쉬프트 연산자 사용시의 암시적 형변환
413 길이가 2인 byte[] 에 {0x8E,0xF0} 를 담으려면?
349 시간안에 응답하지 않는함수는 에러로 뺄려면
340 new String(byte[]) 생성자의 위험성
328 TCP/IP Socket ObjectOutputStream.reset()
297 JUnit 테스트의 작성요령과 간단한 작성례
296 Design Patterns for Java
292 자바 어플리케이션 최적화
289 Effective Java Programming Language Guide -Joshua Bloch
280 OutOfMemory Error 해결방법...
276 DateFormat for Performance
269 이메일주소의 정확성 여부를 확인하는 자바 코드
260 Java Performance tip입니다.
259 Bitter Java - 안티패턴과리팩토링
248 IT Expert 자바 디자인 패턴& 리팩토링 샘플
245 DB 입출력 자동화 툴 - 피넛 0.9 공개
236 Double-checked locking과 Singleton 패턴
233 양력 <-> 음력 변환.
232 자바 소켓 및 쓰레드 정리
210 int 형 정수의 자리수 체크는???
209 *쓰레드에 안전한 싱글톤 클래스를 만드는 세가지 방법

[Java Servlet & JSP API 개발/코딩]
656 통합 게시판 Unicorn v0.9를 배포합니다
655 struts action mappings
570 Java beans에 대해...
502 JSF BOOK 1.0
494 Java Server Faces
407 이미지 섬네일(Thumbnail) 만들기
332 커넥션풀의 커넥션 갯수.
321 커넥션 객체는 사용이 완전히 끝난 후에 해제시켜라.
300 JSP/JSF와 XML/XSLT의 통합
297 OpenSymphony WebWork를 사용한 게시판
296 자카르타 Velocity를 이용한 게시판
293 JSP코드 작성요령 Code Conventions 한글판
270 EmotionalBrain.Protocol.Group1.V1 (Source v0.8)
250 [Tip] 브라우저/확장한글 인코딩(MS949, 샾햏잌)
240 HTTP POST Reqeust Java Socket Client

[EJB(Enterprise Java Beans)]
257 DDL2iBatis(Code Generator For iBatis) 프로그램 및 사용법
195 Core J2EE Patterns 컬러버젼 약간의 설명[이미지]
194 Bitter EJB
109 EJBean의 인스턴스 변수의 정보는 공유되는가?
102 CMP냐 BMP냐?

[WebService/SOAP/UDDI/WSDL/AXIS/XML]
321 SOAP 개발자를 위한 TCP Monitor
213 Tmocat Server와 XML-RPC연동 방법
111 WebService 아키텍쳐에 대해서
87 ebXML MSH 아키텍처 입니다.
77 2003년도 XML 예측 시장
66 웹서비스 적용 국책과제를 수행하려합니다.

[JDBC(Java Database Connectivity)]
400 PreparedStatement 로 실행하면 너무 느린데/분포도 (히스토그램) 문제입니다.
381 DB2/ JDBC/ PreparedStatement/ SQLCODE=-301
345 Oracle에서 executeBatch/return -2/ Re: AS400 JDBC 드라이버의 경우
339 varchar(4000) * 4 사용시 이상한점..
329 Sybase JDBC Driver PreparedStatement 메모리 증가
322 jdk1.4.2와 Microsoft SQL Server 2000 Driver for JDBC와의 문제
204 PreparedStatement 와 Statement 에 대한 고견을...
137 Oracle CLOB 긴글 입력
121 Bitmechanic 커넥션 풀 소개 문서

[Applet, SWING, AWT, SWT]
153 AIX 5.2, X-Window 상에서 Swing 컴포넌트 한글 깨짐 문제
145 Fw: 윈도우XP SP2, Applet Sign 이슈
131 자바 기술을 사용하여 사인드 애플릿 구현하기
128 Eclipse SWT GUI, 브라우져에서 구동하기
120 파일 업로드 애플릿(with Swing)
116 일본판 Windows에서 한글 문제
96 8 비트 BMP 이미지 만들기
82 import시 모든것(*) 는 어떤 의미를 가지는가?

[HTML/HTC/CSS/JavaScript]
192 Grid Control,Tree Control(JSP구현)소스공개
180 테이블 고정 어떻게 해야하나요?
155 브라우저에서 타이틀바 없애기 ..
141 자바스크립으로 디렉토리구조 트리구조 만들수있나요
130 input FileUpload시 "찾아보기.."버튼 대체 방법
125 모달 대화 상자(modalDialog) 예제
113 JavaScript로 구현한 HTML에디터...
102 Binary Behavior에 관하여..
71 HTTP QueryString의 SpecialCharacters Auto Fix
63 한글 또는 영문만이 존재하는지 체크하는 자바스크립트
56 버튼 죽여서 재전송 방지

[World Wide Web Information]
39 URLConnection에서 타임아웃을 세팅할수 있나요?
28 www.abc_def.com & Cookie/httpsession

[서블렛엔진(Tomcat, Resin, JServ, 환경설정, etc..)]
885 apache+resin jsp소스가 보여지는 현상
882 [Tomcat] 최대한 편하게 DBCP 사용해 보기
791 resin, AIX 5.2 64bit 컴파일/설치하기
787 Tomcat에서 MSSQL 2000 에 연동할때...
777 톰켓 4.1.xx 의 JSP include 한글문제 fix
784 NT Service로 기동할때 톰캣메모리설정이 안됩니다.
734 Apache2+JK2+Tomcat5 2개 로드밸런싱시키기
730 AIX 4.3.3 에서 Tomcat 자주 다운되는 현상 발생
721 Re: 톰캣 5.0 이상에서 get의 한글 문제는..
705 tomcat5 monitoring page
661 Heap Memory 늘리려면 catalina.bat의 어디를 수정해야 하나여?
652 Re: apache2.0.48 + tomcat5.0.18 + mod_jk2 완성!!!
612 tomcat4130버전 server.xml하고 web.xml파일 설정법 정리
519 JBoss 3.2.3 및 오라클 9.2연동법/다국어 코딩법

[elipse Project/Plugin]
105 [?] eclipse: 리모트 디버깅, 디플로잉...
78 이클립스에서 롬보즈(Lomboz)사용에 질문드립니다.
50 Eclipse 기능과 단축키들 정리해 보았습니다.
42 "토탈 이클립스" 미리 보기 원고(이클립스,SWT,PDE)

[Apache Struts Framework]
2 struts 프레임웍 사용의 효율성

[BEA/WebLogic Application Server]
1156 웹로직 8.1 모니터링 툴 WLMonitor 2.5 beta 입니다.

[Borland Enterprise Server]
28 J2EE/CORBA 상호 운용 Simplified IDL를 이용한 클라이언트 확장
24 [BES] BES 5.1 에서의 Clustering 관련 문서입니다

[Fujitsu Interstage Application Server]
105 [Interstage6]여러개의 머신에 EJB환경구축개요
104 Redhat 3.0 AS/ES 와 Java Thread에 관해 생각해야 할 문제
97 [ Interstage 6 ] 운영 가이드 및 TroubleShooting 가이드 자료
92 Re: Interstage V6 운영법 간단 정리 매뉴얼

[IBM WebSphere Application Server]
1537 Session Affinity,Session Persistence 와의 차이
1535 Re: Memory-to-memory 관련 문서입니다.
1516 WAS ND상에서 OS환경변수를 설정하기
1500 WSAD 5.1.2 문서 편집기 한글 문제 질문입니다.
1428 블루 스크린 뜹니다. was5.1 문의 드립니다.
1420 AS400에서 File in Use Error
1414 AIX에서 1G 이상의 힙사이즈 할당하기
1412 J2EE 어플리케이션에서 bottleneck을 검출하는 기법에 관한 문서입니다.
1406 DB Fail시 Connection Pool Failover 방안 문의
1567 Re: 웹스피어 : DB Fail-Over & Fail Back
1371 WebSphere3.5 -> 5.1 Upgrade 및 Migration 관련
1347 WAS5.0x: Function sequence error 해결
1313 WebSphere WebServices Gateway 설치 및 Deploy 동영상
1251 L4와 다수의 웹서버, 이미지 캐시 이슈
1247 "kca" for IBM JVM "core" analizing on AIX
1171 WebSphere App Server V5.1 초급 강의 자료 - 실습자료 포함
1149 Debugging WebSphere Application Server Version 5
1146 DB2 64bit Instance에대한 JDBC Driver 사용
1120 ReloadingEnabled(..ext.xmi) & Reload Enabled(console)
1092 WAS 5.x "분산가능" 옵션의 역할
1020 Re: WAS5.0 Base Edition에서 하나이상 서버만들기
969 웹스피어5.0 JNDI 캐싱 해제 옵션
934 WAS 5.0 Customized Error Reporter : error.jsp
932 JSP Re compile 설정은 어떻게 해야하는지 질문..
921 AppClient모듈에서 DataSource 사용기
890 WSAD/WAS 5.0 JSP 한글 깨짐 문제 정리
888 웹스피어 5.0 데이타베이스 연결 설정법
881 WAS5.0 관리콘솔 사용자암호 지정하기
874 WebSphere5.0 서버셋팅과 CMP Deploy방법(wsda5.0 이용하여)
122 WSAD에서의 개발 및 WAS로 deploy

[Marc Fleury's JBoss Application Server]
68 JBoss Admnin Guide 번역본
30 JBoss Guide 번역
25 Apache2.0과 Jboss3.2.x의 JK2 연동방법 및 DataSource 설정방법

[Unix/Linux/Network]
237 AIX, DNS우선순위 local->dns 설정
215 사용자별/프로세스별 시스템 자원 사용율 모니터링 쉘
202 AIX 시간 동기화
178 솔라리스 전원 자동 꺼짐 방지하기 (Solaris)
176 screen utility
174 솔라리스에서 자바 프로세스 띄우기 문의

[DataBase Management Systems]
267 oracle 10에서 정규식 사용하기
242 ORACLE RAC구성과 JDBC설정법에관한 질문

[JVM/JDK Issue]
142 Re: 요청하신 javacore분석결과 올립니다.
104 java.lang.OutOfMemoryError 의 case별 원인 파악및 해결방안 정리
102 IBM JDK HEAPDUMP가 solaris에도?
101 IBM JDK JIT 옵션
82 Re: Zip한글 해결방법
75 AIX IBM JDK 1.4.X reflection 반영시 나타나는 현상
67 IBM JAVA JDK 와 SUN JDK와의 비교되어있는거 없나요?
65 AIX JDK JNI 컴파일 옵션

[기타 자료실]
120 [AutoQuery 2005]다양한 기능을 제공하는 개발자 지원 Tool
116 SWT를 이용한 인터넷 영한사전 프로그램 JDic 0.7
109 [이클립스] 원격 또는 로컬의 로그파일 모니터링 플러그인
107 간단한 디카 사진 정리 프로그램...
98 Alzip보다 최대 2배 가까이 빠른 zip program.
95 [이클립스 플러그인] 원격 자동 업로드 FTP
64 조회,입력, 수정, 삭제 가 매우 쉬운 Grid & Sample

[자바서비스컨설팅/제니퍼 FAQ]
23 리퀘스트몬(requestmon) for Jennifer

[기타]
998 한국의 SI 수준은 이제 세계에서 경쟁력을 잃고 있다.
1050 어느 자바 개발자의 꿈...이원영 자바서비스컨설팅 대표
958 IT노동자 실태조사 자료 및 인터뷰 동영상
860 30대 은행원, 업무 스트레스 못 이겨 자살

[비기술적 이슈에 대한 토론공간]
611 돈 되는 개발이란?
350 [희승사화]여러분이 저라면 어떻게 하시겠습니까?
337 SI에서 개발자라는 것은.
328 아키텍터가 되기 위해서는 어떻게 해야 할까요?..
244 프리랜서 개발 단가..
188 용역단가에 대해서
114 IT(정보기술) 개발자들이 공무원들의 노예인가?(펌)
30 [KT목동사화] 경쟁업체 성능시험 방해 삼성SDS직원 둘 구속기소
1 프리랜서의 한계

[의견 나눔터/잡담]
1792 Object 관련 질문드립니다...(수정본)
1709 okjsp 에서 퍼온글...
1701 잡담
1690 Re: 컨설턴트가 되기 위해서는
1638 인생의, 선택의 문제앞에서. (조언을 부탁드립니다.)
1451 잠적하기 직전 개발자의 변(辨)
1446 [의중사화] Expert One-on-one J2EE 설계와 개발을 읽고 차의중씨에게...
1356 나이를 먹어가며...
1314 티맥스 소프트
1286 미국 환경은 어떤가요?.
1138 개발자의 정년
1109 [그냥 잡담]파견나간 일본회사
1093 50만원이 생긴다면 무엇을 하시겠습니까?
1002 노동자 대단결
944 고인의 명복을 빕니다.
796 추억의 놀이...
716 감동플러스..~~ㅡ_ㅜ
:
Posted by 뽀기