달력

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. 4. 16. 14:37

SunOS5.9에 Mysql 설치하기 그거/Tech2007. 4. 16. 14:37

System Configuration:  Sun Microsystems  sun4u

SunOS 5.9
Generic_118558-38
sun4u sparc SUNW,
Ultra-Enterprise
E3500

2 CPU 64bit
Memory size: 4096 Megabytes

1. http://www.sunfreeware.com/programlistsparc9.html#mysql 사이트에서 mysql을 다운로드 받는다.
   ftp://ftp.sunfreeware.com/pub/freeware/sparc/9/mysql-5.0.27-sol9-sparc-local.gz

2. 위 페이지에서 잘 읽어 보면 libgcc-3.4.6 or gcc-3.4.6, ncurses, zlib, shutil 등도 설치해야 됨을 알 수 있다.
   해당 파일들도 같이 받는다.
   ncurses : ftp://ftp.sunfreeware.com/pub/freeware/sparc/9/ncurses-5.6-sol9-sparc-local.gz
   zlib    : ftp://ftp.sunfreeware.com/pub/freeware/sparc/9/zlib-1.2.3-sol9-sparc-local.gz
   libgcc  : ftp://ftp.sunfreeware.com/pub/freeware/sparc/9/libgcc-3.4.6-sol9-sparc-local.gz
   shutils : ftp://ftp.sunfreeware.com/pub/freeware/sparc/9/shutils-2.0.15-sol9-sparc-local.gz

   여기서 libgcc는 gcc 3.4.6이 이미 설치되어 있을 경우에 따로 설치하지 않아도 된다.

3. 위에서 다운로드 받은 파일들의 압축을 푼다.
   $ gzip -d *.gz
   $ ls
     zlib-1.2.3-sol9-sparc-local
     mysql-5.0.27-sol9-sparc-local
     shutils-2.0.15-sol9-sparc-local

    
4. 압축을 풀어서 나온 pkg 파일들을 pkgadd 명령어를 이용해서 설치한다.

   $ pkgadd -d zlib-1.2.3-sol9-sparc-local
   $ pkgadd -d mysql-5.0.27-sol9-sparc-local
   $ pkgadd -d shutils-2.0.15-sol9-sparc-local

   각 pkg를 설치할 때 설치할거냐고 물어보면 all 또는 enter를 치면 되고, 중간에 사용중인데 설치할거냐고 또 물어보면 y를 입력하면 된다.
   이제 기다리면 pkg의 설치가 끝난다.

5. mysql이 설치되었으면 /usr/local/mysql 디렉토리를 확인한다.
   $ls /usr/local/mysql
    bin/         info/        libexec/     mysql-test/  sql-bench/
    include/     lib/         man/         share/       var/

6. /usr/local/mysql/bin 디렉토리에 가서 "mysql_install_db" 명령어를 실행한다.
   $/usr/local/mysql/bin/mysql_install_db

   위 명령어를 실행하면 DB가 설치된다.

7. 위 명령어를 실행하다가 library 관련 오류가 발생하면 아래 명령어를 이용해 필요한 library를 확인하고 설치한다.
   $ ldd /usr/local/mysql/bin/mysql
        libncurses.so.5 =>       /usr/local/lib/libncurses.so.5
        libmysqlclient.so.15 =>  /usr/local/mysql/lib/mysql/libmysqlclient.so.15
        librt.so.1 =>    /usr/lib/librt.so.1
        libgen.so.1 =>   /usr/lib/libgen.so.1
        libsocket.so.1 =>        /usr/lib/libsocket.so.1
        libnsl.so.1 =>   /usr/lib/libnsl.so.1
        libssl.so.0.9.8 =>       /usr/lib/libssl.so.0.9.8  - 잘못된 ELF 클래스: ELFCLASS64
        libcrypto.so.0.9.8 =>    (파일이 없음)
        libz.so =>       /usr/local/lib/libz.so
        libstdc++.so.6 =>        /opt/gcc/lib/libstdc++.so.6
        libm.so.1 =>     /usr/lib/libm.so.1
        libgcc_s.so.1 =>         /opt/gcc/lib/libgcc_s.so.1
        libc.so.1 =>     /usr/lib/libc.so.1
        libssl.so.0.9.8 =>       /usr/lib/libssl.so.0.9.8  - 잘못된 ELF 클래스: ELFCLASS64
        libcrypto.so.0.9.8 =>    (파일이 없음)
        libaio.so.1 =>   /usr/lib/libaio.so.1
        libmd5.so.1 =>   /usr/lib/libmd5.so.1
        libdl.so.1 =>    /usr/lib/libdl.so.1
        libmp.so.2 =>    /usr/lib/libmp.so.2
        /usr/platform/SUNW,Ultra-Enterprise/lib/libc_psr.so.1
        /usr/platform/SUNW,Ultra-Enterprise/lib/libmd5_psr.so.1

     위와 같이 나왔다면
     "libgcrypt-1.2.2-sol9-sparc-local", "openssl-0.9.8e-sol9-sparc-local" 두가지 library가 없어서 발생한 오류이다.
  www.sunfreeware.com 에서 해당 library를 다운로드 받아서 설치한다.

8. 필요한 library를 다 설치하고 DB까지 설치한 후에 .profile을 열어서 PATH 정보를 수정한다.
   $ vi .profile
   export PATH=$PATH:/usr/local/mysql/bin

7. 환경설정 정보를 reload 한다.
   $ . .profile

8. mysql을 실행한다.
   $ nohup /usr/local/mysql/bin/mysqld_safe --user=mysql &

:
Posted by 뽀기
2007. 4. 16. 14:21

SunOS 5.9에 vim7.0 설치하기 그거/Tech2007. 4. 16. 14:21

System Configuration:  Sun Microsystems  sun4u

SunOS 5.9
Generic_118558-38
sun4u sparc SUNW,
Ultra-Enterprise
E3500

2 CPU 64bit
Memory size: 4096 Megabytes

1. www.sunfreeware.com 사이트에서 vim7.0 을 다운로드 받는다.
   ftp://ftp.sunfreeware.com/pub/freeware/sparc/9/vim-7.0-sol9-sparc-local.gz
2. 다운로드 받은 파일의 압축을 푼다.
   $ gzip -d vim-7.0-sol9-sparc-local.gz
3. 압축을 풀면 pkg 파일이 나온다.
   $ ls
     vim-7.0-sol9-sparc-local

4. 위의 pkg 파일을 pkgadd 명령어를 이용해서 설치한다.
   $ pkgadd -d vim-7.0-sol9-sparc-local
   위와 같이 실행하면 설치할거냐고 물어본다. 그러면, 그냥 enter 친다.
   그리고, 조금 있다가 /usr/bin, /usr/local/bin 등 사용하고 있는데 설치할거냐고 물어보면
   그냥 y 눌러준다.
   조금 기다리면 파일들 복사하고 혼자 설치 잘 하고 마지막에 "클래스 검증" 어쩌고 하면서 끝난다.
   설치가 완료됐다.
   설치는 "/usr/local/share/vim/vim70" 디렉토리에 된다.
5. vim(실행파일)은 /usr/local/bin 디렉토리 밑에 복사된다.
   vim 명령어를 실행해본다.
   $ /usr/local/bin/vim
6. 아래와 같이 빈 화면에 글씨가 나온다면 성공적으로 설치가 됐다.

                     VIM - Vi IMproved
   ~
   ~                     version 7.0
   ~                by Bram Moolenaar et al.
   ~      Vim is open source and freely distributable
   ~
   ~            Help poor children in Uganda!
   ~     type  :help iccf<Enter>       for information
   ~
   ~     type  :q<Enter>               to exit
   ~     type  :help<Enter>  or  <F1>  for on-line help
   ~     type  :help version7<Enter>   for version info
   ~
   ~             Running in Vi compatible mode
   ~     type  :set nocp<Enter>        for Vim defaults
   ~     type  :help cp-default<Enter> for info on this

   ~

7. 실행이 제대로 안될경우 아래 명령어를 통해서 필요한 library들이 뭐가 있는지 확인한 후에
   없는 library 들을 추가로 설치해준다.
   (library들은 www.sunfreeware.com 에서 다운로드 받을 수 있다.)
  
   $ ldd /usr/local/bin/vim
        libgtk-1.2.so.0 =>       /usr/local/lib/libgtk-1.2.so.0
        libgdk-1.2.so.0 =>       /usr/local/lib/libgdk-1.2.so.0
        libgmodule-1.2.so.0 =>   /usr/local/lib/libgmodule-1.2.so.0
        libglib-1.2.so.0 =>      /usr/local/lib/libglib-1.2.so.0
        libXext.so.0 =>  /usr/lib/libXext.so.0
        libm.so.1 =>     /usr/lib/libm.so.1
        libXt.so.4 =>    /usr/lib/libXt.so.4
        libX11.so.4 =>   /usr/lib/libX11.so.4
        libSM.so.6 =>    /usr/lib/libSM.so.6
        libICE.so.6 =>   /usr/lib/libICE.so.6
        libncurses.so.5 =>       /usr/local/lib/libncurses.so.5
        libnsl.so.1 =>   /usr/lib/libnsl.so.1
        libsocket.so.1 =>        /usr/lib/libsocket.so.1
        libdl.so.1 =>    /usr/lib/libdl.so.1
        libc.so.1 =>     /usr/lib/libc.so.1
        libmp.so.2 =>    /usr/lib/libmp.so.2
        /usr/platform/SUNW,Ultra-Enterprise/lib/libc_psr.so.1

 
 위 결과는 제대로 설치됐을 경우의 화면이다.
 제대로 설치가 안되면  "=>" 이거 오른쪽에 있는 so 파일들의 정보가 아래 처럼 제대로 보이지 않는다.
    libgtk-1.2.so.0 =>       잘못된 ELF 클래스: ELFCLASS64

:
Posted by 뽀기
2007. 4. 16. 14:07

SunOS 5.9에 gcc 설치하기 그거/Tech2007. 4. 16. 14:07

System Configuration:  Sun Microsystems  sun4u

SunOS 5.9
Generic_118558-38
sun4u sparc SUNW,
Ultra-Enterprise
E3500

2 CPU 64bit
Memory size: 4096 Megabytes

  1. internet에서 "Solaris 9-GCC-3.4.6.tar.zip" 파일을 찾아서 다운로드 받는다.
     (나도 어디서 받았는지 잊어버렸다.)
  2. 압축을 푼다.
     $ gzip -d Solaris 9-GCC-3.4.6.tar.zip
     $ tar -xvf Solaris 9-GCC-3.4.6.tar
     $ ls
       gcc

  3. 압축을 풀면 나온 "gcc" 디렉토리를 "/opt" 밑으로 복사한다.
     $ cp -rp gcc /opt

  4. .profile 파일을 열어서 아래 내용을 설정해준다.
     $vi .profile
     export PATH=/opt/gcc/bin:$PATH
     export LD_LIBRARY_PATH=/opt/gcc/lib:/opt/gcc/lib/sparcv9:$LD_LIBRARY_PATH

Solaris 9-GCC-3.4.6.tar.zip
이 파일만 찾아라 그러면 쉽다!!

:
Posted by 뽀기
2007. 4. 16. 13:57

SunOS 5.9에 64bit JDK 1.5.0.11 설치 그거/Tech2007. 4. 16. 13:57

# 시스템 환경#
System Configuration:  Sun Microsystems  sun4u
SunOS 5.9
Generic_118558-38
sun4u sparc SUNW,
Ultra-Enterprise
E3500
2 CPU 64bit
Memory size: 4096 Megabytes

  1. http://java.sun.com/javase/downloads/index_jdk5.jsp 이 페이지에서
     JDK 5.0 Update 11 이 항목에 대해서 "Download" 페이지로 이동한다.
  (https://sdlc2b.sun.com/ECom/EComActionServlet;jsessionid=BFB2B91425F47264276E3456F4BB5B03 여기로 이동하면 된다.)

  2. 위에서 이동한 페이지에서 license 동의할 거냐고 물어보는 곳에서 "Accept"를 선택해준다.
     그러면 화면이 또 이동한다.
  (https://sdlc2b.sun.com/ECom/EComActionServlet/DownloadPage:~:com.sun.sunit.sdlc.content.DownloadPageInfo;jsessionid=BFB2B91425F47264276E3456F4BB5B03;jsessionid=BFB2B91425F47264276E3456F4BB5B03 여기로 이동)

  3. 위에서 이동한 페이지에서 세번째에 있는 "Solaris SPARC Platform - J2SE(TM) Development Kit 5.0 Update 11)" 에서
     "Solaris SPARC 32-bit self-extracting file" 이 파일과 " Solaris SPARC 64-bit self-extracting file (use 32-bit version for applet and Java Web Start support)" 이 파일을 다운로드 받는다.

  4. 위에서 다운 받은 "jdk-1_5_0_11-solaris-sparc.sh", "jdk-1_5_0_11-solaris-sparcv9.sh" 파일 중에서
     "jdk-1_5_0_11-solaris-sparc.sh" 파일을 먼저 설치하고, "jdk-1_5_0_11-solaris-sparcv9.sh" 파일을 나중에 설치해준다.
  (다운받는 페이지에서 나왔듯이 self-extracting 파일이므로 아래와 같이 실행하면 알아서 설치해준다.)
  $ sh jdk-1_5_0_11-solaris-sparc.sh
  $ sh jdk-1_5_0_11-solaris-sparcv9.sh

  5. 위 두 파일을 설치하고 나면 64bit 용 jdk 1.5가 설치완료된다.

  6. 설치가 끝났으면 PATH에 위에서 설치한 JDK 1.5의 실행파일이 있는 bin 디렉토리를 추가해준다.
     $ vi ~/.profile
      export JAVA_HOME=/usr/jdk1.5.0_11
      export PATH=$JAVA_HOME/bin:$PATH

  7. .profile 파일을 저장하고 환경을 reload 해준다.
     $ . .profile

  8. 위에서 $JAVA_HOME/bin 을 해준 후에 설치된 jdk 의 버전을 확인한다.
     $ java -version
     java version "1.5.0_11"
     Java(TM) 2 Runtime Environment, Standard Edition (build 1.5.0_11-b03)
     Java HotSpot(TM) Server VM (build 1.5.0_11-b03, mixed mode)
     HIT-TEST@root[/usr/jdk1.5.0_11/bin]# echo $PATH

  9. 64bit용 JDK를 사용하고 싶다면 PATH에 아래와 같이 추가해주면 된다.
     $ vi ~/.profile
     export JAVA_HOME=/usr/jdk1.5.0_11
     export PATH=$JAVA_HOME/bin/sparcv9:$PATH

  10. 위와 같이 추가한 후에 JDK의 버전을 확인해본다.
     $ java -version
     java version "1.5.0_11"
     Java(TM) 2 Runtime Environment, Standard Edition (build 1.5.0_11-b03)
     Java HotSpot(TM) 64-Bit Server VM (build 1.5.0_11-b03, mixed mode)

:
Posted by 뽀기


이일민 l http://dbguide.net/comb/common/edit/tobyilee@gmall.com l Epril 대표 컨설던트


AOP의 시대가 도래하고 있다. AOP가 등장한지 10년이 되면서 AOP는 많은 개발자들의 주목을 받고 있다. 하지만 AOP를 본격적으로 적용하는 길은 쉽지만은 않은 탓에 많은 개발자들이 AOP의 문턱을 넘지 못하고 포기하는 경우가 많다. AOP는 잘 쓰면 좋지만 없어도 크게 아쉬울 것 없는 선택 기술일까? 아니면 이제는 반드시 익혀야 하는 필수 기술일까?


객체지향프로그래밍(OOP)은 지금까지 출현한 개발 패러다임 중 가장 성공한 것으로 평가되고 있다. 그 때문에, 현재 시스템 개발에 가장 많이 쓰이는 언어의 대부분은 OOP기반으로 설계된 것들이다. 또, 가장 많이 사용되는 언어인 자바와 C#을 비롯해 오랫동안 사용되어져 온 C++, 스몰토크(Smalltalk) 그리고 최근에 각광받고 있는 다이나믹 언어인 파이썬(Python)과 루비(Ruby)에 이르기까지 모두 객체지향언어라는 것을 장점으로 내세우고 있을 정도다.



AOP의 도입의 필요성



객체지향 프로그래밍의 가장 큰 특징은 시스템이 다루고자 하는 도메인의 모델을 설계와 구현에 손쉽게 반영할 수 있다는 데 있다. 오브젝트 모델자체가 실제 세계의 개체를 투영하고 있기 때문이다. 고객의 요구사항과 도메인을 가지고 객체지향적인 설계를 하고 나면 그 다음 작업은 매우 수월하다. OOP의 경우 설계(design)와 구현(implementation)은 거의 1:1 관계에 있기 때문이다. 그래서 OO툴을 사용하면 설계 자료에서 자동으로 코드를 생성해 낼 수 있기도 하다. 따라서 많은 경우 설계상의 기능은 각각 하나의 클래스 또는 패키지로 구현될 수 있다.


설계와 구현이 정확히 대응되는 이상적인 상황이라면 OOP의 장점이 극대화될 수 있다. 각 패키지별로 독립적인 개발이 가능해 최적화된 형태의 개발 작업이 가능하다. 또 유지보수가 쉬워지고 독립적인 테스트도 수월해진다. 더 나아가 각각의 컴포넌트를 재활용하거나 확장해나가기도 쉽다. OOP나 객체지향분석설계(OOAD)를 설명하고 있는 책에 나오는 모델, 설계다이어그램과 그것을 구현한 코드를 보면 이런 OOP의 매력에 감탄하게 된다. 하지만 현실세계의 OOP 개발자들이 이런 OOP의 장점을 충분히 누리고 있는지 묻는다면 쉽게 그렇다고 대답하기 어렵다.


엔터프라이즈 시스템의 복잡도는 점점 증가하고 있다. 이런 복잡한 시스템은 요구사항과 기능을 모두 충족시키면서 객체지향적인 설계를 하는 것이 매우 어려워진다. 결국에는 객체지향적인 원칙을 지키지 못하고 타협할 수밖에 없는 상황에 이르는 것이 현실이다. OOP의 장점을 살리지 못하게 방해하는 이런 타협의 결과는 크게 두 가지로 나타난다. 하나는 기능의 분산이고 다른 하나는 코드의 혼란이다.



분산(Scattering)



하나의 기능을 하나의 모듈로 캡슐화하기 불가능하거나 힘든 경우가 있다. 도메인의 핵심기능을 객체지향적인 원칙에 따라 설계했다하더라도 많은 엔터프라이즈 서비스 기능들은 단일 모듈로 독립시킬 수 없고 여러 모듈에 분산해서 존재하게 된다. <그림 1>은 이러한 실제 예를 보여주는 그림이다. 객체지향적으로 설계한 모듈들 안에 보안이나 모니터링 기능이 분산되어 존재하고 있다. 이런 기능들은 단일 모듈로 만드는 것이 불가능하다. 따라서 이런 기능의 코드들은 여러 개의 모듈 안에 분산되어 반복적으로 나타나게 된다. 필연적으로 중복된 코드를 양산하게 되는 것이다.


결과적으로 이런 분산된 기능의 코드는 유지보수를 힘들게 한다. 더 나아가서는 객체지향적으로 설계된 모듈임에도 재사용이 어려워 질 수 있다. 분산된 코드에 침범 당한 모듈은 의존성이 매우 강해지기 때문이다.



혼란(Tangling)



여러 개의 모듈에 분산되어 중복되어있어 모듈화가 불가능한 코드들은 각 모듈의 코드자체를 혼란에 빠뜨리게 된다. 아무리 모듈의 오브젝트 설계자체는 단일책임원칙에 따라서 잘 설계했다 할지라도 이렇게 침범한(invasive) 코드들로 인해서 원래의 깔끔한 설계와 코드는 직접적인 연관성의 기능이 뒤엉켜 지저분한 코드가 되어 버린다.


<리스트 1>은 객체지향설계에 따라 설계된 이상적인 OOP코드이다. 하지만 많은 엔터프라이즈 서비스의 기능들이 독립적으로 존재하지 못하고 코드를 침범하면 <리스트 2>와 같이 다양한 기능이 뒤엉킨 지저분한 코드로 바뀌게 된다.


사용자 삽입 이미지

<그림 1> OOP모듈에 분산된 기능


<리스트-1> 침범당하지 않은 OOP코드

class MemberService {
MemberRepository repository;

public void upgradeLevel(List<Member> members, int newLevel) {
for(Member member : members) {
member.changeLevel(newLevel);
repository.updateMember(member);
}
}

}


<리스트-2> 침범당한 OOP코드

class MemberService {
MemberRepository repository;
Log logger;
TranasctionManager txManager;
SecurityService securityService
NotificationPolicy notificationPolicy;
EMailService emailService;

public void upgradeLevel(List<Member> members, int newLevel) {

// Logging
if (logger.isDebug()) { logger.debug("upgradeLevel() started", members, newLevel)
}

// Security
if (!securityService.checkPerssionForModify(Member.class)) {
if (logger.isWarn()) {
logger.warn("Security violation: Member modification", new Date());
}
throw new SecurityException(Member.class, SecurityType.Modification);
}

// Transaction
Transaction tx;
try {
tx = transactionService.beginTransaction();

for(Member member : members) {
member.changeLevel(newLevel);
repository.updateMember(member);
}
}
catch(DataException e) {
try { tx.rollback(); } catch(Exception e2) {throw e2;}
throw new ExceptionTranslater().translateToSystemException(e);
}

if (notificationPolicy.isEMailNotificationEvent("Member.upgradeLevel")) {
emailService.sendNotificationToMemberAdmin(members, newLevel)
}

}

}

OOP를 학습하기 위해서 작성하는 코드가 아니라면 현실 세계 대부분의 코드는 <리스트 2>와 유사하게 만들어진다. 엔터프라이즈 시스템의 복잡도와 요구사항이 날로 높아지는 요즘 같은 경우는 더더욱 복잡한 기능이 결합되어 있는 코드가 만들어지게 된다. 결과적으로 OOP의 장점인 모듈의 독립성이나 재사용, 이해하기 쉬운 코드, 리팩토링, 단위테스트 편이성 등이 극도로 저하된다. <리스트 2>는 OOP언어를 사용해서 작성한 코드이지만 더 이상 OOP의 장점을 찾아보기 힘든 코드가 되어버렸다.


이렇게 독립된 모듈로 분리시킬 수 없는 기능들은 주로 개발하는 시스템의 핵심 로직을 담은 핵심 도메인 로직과는 다른 차원에서 존재하면서 도메인 오브젝트에 분산되어 존재할 수밖에 없는 것들이다. 대표적으로 Tracing, Logging, Error and Exception Handling, Monitoring, Statistics gathering, Transaction, Session Management, Threading, Synchronization, Caching, Remote access 등이다. 이러한 기능은 대부분 엔터프라이즈 서비스나 인프라스트럭처 서비스라고 불리는 것들이고 날이 갈수록 점점 더 많이 요구되어지고 있다.


타협할 것인가 대안을 찾을 것인가?


이러한 문제에 맞닥뜨린 개발자들은 결국 두 가지 중에 하나를 선택해야 한다. 하나는 OOP의 장점을 포기하고 타협하는 것이다. 코드의 중복을 피할 수 없고 지속적인 유지보수, 재활용의 어려움이 있지만 복사&붙이기와 같은 방법을 통해서 일단은 쉽게 접근할 수 있는 <리스트 2>와 같은 방법을 선택하는 것이다.


두 번째 방법은 AOP를 이용해서 이러한 문제들을 풀어나가는 것이다. AOP는 바로 이런 분산, 혼재되어있는 코드의 문제점을 극복하고 <리스트 1>과 같은 OOP의 장점과 특성을 그대로 간직한 코드를 만들 수 있도록 돕기 위해서 존재하는 기술이다.


많은 개발자들은 AOP가 OOP를 대신해서 독립적인 프로그래밍 패러다임으로 존재하는 대체기술로 오해하고 있다. 하지만 AOP는 OOP가 가진 한계를 극복하고 OOP의 원칙에 충실한 설계와 개발이 가능하도록 지원해주는 역할을 하고 있다. 사실 AOP로 만들어낼 수 있는 모든 기능은 OOP로 할 수 있는 것들이다. 하지만 OOP만으로는 객체지향원칙에 타협을 해야 할 수밖에 없기 때문에 AOP를 사용하는 것이 훨씬 유리한 선택일 것이다.


AOP가 어떻게 OOP의 코드를 유지한 채로 이러한 기능들을 적용해낼 수 있는지는 지난 2005년 11월호 <마소>에 필자가 쓴 ‘관점지향프로그래밍-AOP’이라는 기사를 참조하길 바란다.



AOP의 도입의 장애물



그렇다면 이렇게 장점이 많은 AOP가 나온 지 10년이 되도록 폭 넓게 적용되지 못한 이유는 것일까? 그 이유는 AOP가 생각보다 도입하기가 쉽지 않기 때문이다. 많은 개발자들이 AOP에 대해서 관심을 가지고 있음에도 AOP를 실제 업무에 적용하지 못하게 하는 AOP 도입의 장애물은 세 가지 정도로 생각해 볼 수 있다.


AOP에 대한 미신과 오해


첫째는 AOP에 대한 미신과 오해들로 인해서 AOP도입을 꺼리게 되기 때문이다.


AOP를 처음 접하는 개발자들은 아주 간단한 예제로 만들어진 튜토리얼을 접하면서 시작한다. 현실 세계에서 가장 많이 필요로 하고 구현하기 쉽기 때문에 자주 등장하는 예제는 Logging과 Tracing이다. 간단한 크로스컷(crosscut)으로 표현이 가능하고 어드바이스(advice)코드가 단순하기 때문에 AOP 예제에 자주 등장한다. 문제는 이로 인해서 많은 개발자들이 AOP의 활용정도를 단순히 메소드 실행을 추적하거나 로그를 남기는 정도로 한정지어 생각하는 것이다. 그러나 AOP의 활용용도는 매우 다양하다.


다음의 요구사항을 생각해보자.


● 서비스 레이어의 메소드는 트랜잭션 안에서 동작해야하고 서비스 레이어의 경계를 넘어올 때 commit 되거나 예외가 발생했을 때는 rollback되어야 한다.
● 만약 메소드의 이름이 get으로 시작하면 읽기 전용의 트랜잭션이 생성되어야 한다.
● JDBC 코드로 만든 데이터 레이어에서 발생하는 SQLException은 자동으로 시스템의 표준 Exception hierachy의 하나로 변환되어 다시 던져져야 한다.
● 데이터 레이어의 메소드는 반드시 서비스 레이어의 메소드에서만 호출되어야 한다. 프레젠테이션 레이어나 JSP 등에서는 DAO 메소드를 호출할 수 없고 개발자가 강제로 그러한 코드를 만든다면 에러가 발생해야 한다.
● Optimistic-locking에 의해서 실패한 서비스 메소드는 정해진 횟수만큼 반복해서 재시도 되어야 한다.


AOP를 이용하면 이러한 요구사항을 충족하는 기능을 OOP의 핵심코드를 전혀 수정하지 않고 시스템 전체에 걸쳐서 적용하는 것이 가능하다. AOP의 활용용도는 이외에도 무궁무진하다.


AOP는 매우 복잡한 기술이고 결과적으로 전체 프로그램도 복잡해지며 이해하기 힘들어진다는 편견도 문제다. AOP의 구현기술이 복잡한 것은 사실이다. AOP를 사용하기 위해서는 OOP 언어를 확장하거나 별도의 컴파일러가 필요하고 바이트 코드조작 등의 고급기법이 사용된다. 하지만 그것은 AOP 툴을 만드는 개발자의 부담일 뿐이고 그것을 사용하는 개발자는 오히려 더 단순하고 심플한 개발을 할 수 있다. 그 이유는 AOP를 이용하면 훨씬 높은 수준의 추상화가 가능하기 때문이다. 물론 추상화가 잘 될수록 프로그램의 흐름을 이해하는 것은 더 어려워진다. 하지만 그것은 OOP도 마찬가지다. 추상화의 장점이 크기 때문에 절차적 프로그래밍보다 OOP를 선호하는 것처럼 AOP의 추상화도 마찬가지이다. 또 AOP로 추상화를 하지 않은 타협한 OOP코드는 사실 이해하기가 훨씬 더 어렵다.


그 외에도 AOP의 기능은 잘 설계된 인터페이스나, 디자인패턴 등을 잘 활용하면 대체할 수 있다는 것도 잘못된 이해이다. 인터페이스나 디자인패턴은 OOP 자체의 설계를 건전하고 견고하게 만들어줄 수 있다. 하지만 AOP는 OOP로는 해결할 수 없는 부분을 담당하는 것이기 때문에 단지 인터페이스나 디자인패턴이 AOP를 대체할 수 없다.


AOP의 학습에 대한 부담


두 번째 문제는 AOP의 학습에 대한 부담이다.


새로운 기술을 습득하고 더 나아가 새로운 패러다임을 수용하는 것은 언제나 부담스러운 일이다. AOP를 학습하고 사용하려면 절차적 프로그래밍에서 객체지향 프로그래밍으로 전환할 때만큼의 수고가 필요하다.


AOP 학습이 어려운 이유는 OOP와 패러다임 자체의 차이도 있지만, 거기에 다른 새로운 언어적인 요소도 추가됐기 때문이다. 가장 대표적인 AOP 툴인 AspectJ는 자바 언어 자체를 확장해서 만들어진 새로운 언어이다. 따라서 컴파일러도 전용컴파일러를 사용한다.


AspectJ에서 가장 중요하면서 제일 어려운 부분은 포인트컷(Pointcut)이다. <리스트 3>의 포인트컷 선언 같은 경우는 사실 쉽게 이해하기 어렵다. AOP에서 포인트컷은 RDB의 SQL과 유사하다고 생각하면 된다. SQL을 충분히 익히지 않으면 RDB 개발에 어려움이 있는 것처럼 포인트컷을 마스터하지 않고는 AOP를 제대로 활용하기 힘들다. 잘못된 포인트컷의 사용은 AOP를 적절히 활용하지 못하고 애플리케이션을 복잡하게 만드는 등의 부작용을 일으킬 수 있다.


<리스트-3> 복잡한 포인트컷 선언

(execution(!private * com.other..*.set*(*))
|| execution(!private * com.other..*.get*(*)))
&& cflow(execution(public * com.mycompany..*.*(*, *, ..)))


따라서 AOP를 학습할 때에는 포인트컷에 대한 투자를 충분히 해야 한다. 포인트컷의 문법을 숙지하고 다양하게 작성된 포인트컷을 분석해보는 작업이 필요하다. 포인트컷 안에 사용되는 조인포인트(Join Point)의 정의와 의미도 정확히 이해해야한다.


AspectJ 5에서는 Java5의 어노테이션을 이용한 포인트컷이 도입되었다. 와일드카드를 이용한 기존의 포인트컷보다 좀 더 편리한 타깃 설정이 가능하다는 장점이 있지만 반면에 포인트컷 자체를 더 이해하기 어렵게 만들기도 한다. <리스트 4>는 언뜻 보기엔 유사하지만 의미가 다른 어노테이션을 이용한 세 가지 포인트컷의 예이다. 첫 번째 포인트컷은 @Transactional 어노테이션을 가진 모든 종류의 메소드의 실행(execution)을 가리키는 포인트컷이다. 두 번째는@Transactional 어노테이션을 가진 오브젝트를 리턴하는 모든 메소드를 실행하는 것을 정의한 포인트컷이다. 세 번째는@Transactional 어노테이션을 가진 타입(type)에 정의된 모든 메소드의 실행에 대한 포인트컷이다. 매우 짧고 간단한 포인트컷들이지만 이를 명확히 구분하려면 정확한 포인트컷 언어에 대한 정의를 알고 있고, 익숙하게 사용할 수 있도록 학습하는 것이 절실히 필요하다.


<리스트-4> 어노테이션을 이용한 다양한 포인트컷 정의

execution(@Transactional * *.*(..))
execution((@Transactional *) *.*(..))
execution(* (@Transactional *).*(..))

또 아직 이렇다할 표준이 없다는 점도 AOP 학습을 어렵게 하는 이유 중 하나이다. 따라서 AOP 툴에 따라 사용하는 용어가 조금씩 다르고 포인트컷이나 애스펙트를 정의하는 것이 제각각이다. 그로인해 용어의 혼동을 가져오기도 하고 하나의 AOP 툴을 충분히 이해했다하더라도 다른 AOP 솔루션을 사용하면, 또다시 새로운 학습이 필요하게 되는 어려움이 있다. AOP의 표준을 만들거나 스펙의 연합체를 만들려는 노력은 계속 있어왔다. 대표적으로 SpringAOP가 참여한 AOP Alliance는 Java/JEE의 AOP 표준을 만들기 위한 시도 중의 하나였다. 하지만 여전히 명확히 정의되고 모든 AOP 솔루션이 따르고 있는 표준은 존재하지 않는다. 그나마 다행스러운 것은 AOP 툴 사이에 여러 가지 호환을 위한 노력이 진행되고 있다는 점이다. AspectJ의 포인트컷 언어를 사용하는 Spring2.0 AOP의 예가 그런 긍정적인 노력의 하나라고 할 수 있다.


AOP의 학습이 쉽지 않은 것은 사실이지만 그 필요성을 절실히 인식하고 기술습득에 대한 충분한 투자를 할 만한 가치를 인정하는 것이 필요하다.


AOP도입전략의 부재


세 번째는 적절한 AOP 도입전략을 가지고 있지 못하기 때문이다.


AOP를 도입하려는 개발자나 기업의 가장 잘못된 접근방법은 단번에 AOP를 도입하려는 욕심을 내는 데 있다. AOP를 전부 아니면 전무(All-or-nothing) 개념으로 접근하는 것은 바람직하지 않다. AOP는 패러다임의 전환이고 상당한 노력과 시간이 필요하다. 또 그 적용은 팀 내 모든 개발자에게 영향을 미친다. 섣부른 AOP의 도움은 오히려 AOP에 대한 실망만 안겨주고 그다지 좋은 결과를 얻지 못할 수 있다.


AOP 도입의 가장 이상적인 방법은 단계적인 접근이다. 처음에는 가볍고 쉽게 시작할 수 있는 것부터 출발해서 지속적인 학습과 검증을 병행해 가면서 난이도가 높은 단계로 발전해나가는 것이다. AOP는 OOP 전체가 아닌 일부를 대체하기 때문에 그 정도를 조절하는 것이 가능하다.


사용자 삽입 이미지

<그림 2> AOP의 단계적인 도입 전략


<그림 2>는 AOP 전문가들이 추천하는 AOP 도입전략이다. 크게 4단계로 구분해서 1단계부터 단계적으로 적용하고 성공적인 적용이 끝나면 다음 단계로 나아가는 식이다.


첫 번째 단계는 프레임워크 등을 통해서 이미 구현되어있는 애스펙트를 사용하는 것이다. 가장 대표적인 것으로 SpringAOP를 이용한 서비스 레이어에 대한 트랜잭션 애스팩트의 적용이 있다. 스프링프레임워크를 사용하는 개발자는 AOP에 대한 지식 없이 간단한 설정만으로 AOP의 혜택을 누릴 수 있다. 사실 대부분의 스프링 사용자는 SpringAOP를 직접 사용하지 않는다. 하지만 프레임워크에서 제공하는 AOP 기능을 충분히 누릴 수 있다. 이처럼 손쉽게 쓸 수 있는 애스펙트를 적용하면서 AOP로 인해서 단순해진 핵심로직의 편리함을 누리는 것이 AOP의 매력을 느끼고 다음단계로 나아갈 수 있는 출발이 될 수 있다.


두 번째 단계는 정책의 검사와 강제(exploration and enforcement)에 AOP를 적용하는 것이다. 시스템 전반에 걸친 정책을 일괄적으로 적용하는 것은 쉬운 일이 아니다. 모든 개발자들이 정책을 명확히 인지하고 있어야 하고 그것을 위반하지 않고 개발에 적용해야 하기 때문이다. 대형프로젝트에는 수십 페이지의 정책문서가 작성되고 개발자들에게 공개된다. 개발자들은 그 규정에 따라 코드를 작성해야 한다. 그렇게 만들어진 코드에 정확히 정책이 적용되었는지를 검증하는 것 또한 큰일이다. 품질팀의 사람들이 여러 단계에 걸쳐서 그것을 검토하지만 끊임없이 수정되고 발전하는 코드를 지속적으로 체크하는 것은 어려운 일이다. AOP는 이런 전체적인 정책을 검사하고 필요에 따라 강제할 수 있는 효과적인 수단을 제공한다. 이 단계에서는 아직 모든 개발자가 AOP에 대한 많은 지식을 가질 필요도 없다. 정책을 결정하고 부여하는 역할을 담당하는 소수의 사람에 의해서 적용이 된다. 더 나아가서 필요하면 정책의 변경도 손쉽게 할 수 있다.


레이어기반의 아키텍처(Layered Architecture)에서 레이어 간의 의존성과 메소드의 호출 정책은 중요한 요소이다. 얼마 전에 필자는 대형프로젝트의 아키텍트로부터 개발자들이 이 정책을 지키지 않고 편의에 따라서 뷰(JSP)에서 데이터 레이어(DAO)의 인스턴스를 만들어서 사용하고 있는 것 때문에 고민이라는 얘기를 들었다. 시스템 레벨에서 일괄적으로 이것을 막을 수 있는 방법에 대한 조언을 구했기에 간단히 AOP를 적용하도록 조언해줬다.
AOP를 이용하지 않고 이런 적용을 하려면 매우 복잡할 것이다. 모든 DAO 메소드에 현재 쓰레드의 호출스택(Call Stack)을 조사해서 지정된 레이어가 아닌 곳에서 호출이 있다면 이를 막는 코드를 다 삽입해야 할 것이다. 큰 프로젝트라면 수백, 수천 개의 메소드에 이 코드를 모두 삽입해야 한다. 하지만 AOP를 적용하면 간단히 이런 정책을 강제할 수 있다. <리스트 5>의 AOP 코드만으로 어떤 뷰에서도 DAO를 사용할 수 없도록 제한 할 수 있다. 더 나아가서 정책이 변경된다면 언제든지 이 애스펙트를 수정하는 것만으로 전체 시스템에 정책의 변경을 적용할 수 있다.


<리스트 5> 레이어간 호출 정책 애스펙트

aspect LayerCallingPolicy {
pointcut inView() : within(view..*);
pointcut daoCall(): call(* dao..*(..)) &&!inDao();

declare error : daoCall() && inView(): "View에서 DAO를 직접 호출할 수 없습니다";

}

try/catch를 사용한 모든 코드에서 catch 후 아무 것도 하지 않는 dummy catch를 찾아내는 것을 생각해보자. 코드리뷰를 통해서 하려면 엄청난 작업이다. 반면에 AOP를 이용하면 세 줄 정도의 코드만 추가하면 충분하다.


세 번째 단계는 인프라스트럭처 애스펙트의 적용이다. 인프라스트럭처 서비스를 사용하는 것은 개발자들이 항상 해야 하는 일이기 때문에 이 단계에서 AOP를 적용하는 것은 팀 내 모든 개발자들이 분명하게 인지하고 있어야 한다. 대부분의 인프라스트럭처 서비스나 엔터프라이즈 서비스를 이 단계에서 AOP를 통해서 모듈화 할 수 있고 개발자들은 결과적으로 OOP에 충실하게 핵심로직을 개발하는 것이 가능해진다. 로깅, 에러핸들링, 모니터링, 트랜잭션, 세션관리, 캐슁 등의 많은 서비스들을 AOP에 적용할 수 있다.


세 번째 단계가 되면 시스템 전체적으로 AOP가 중요한 역할을 담당하게 된다. 이쯤 되면 본격적으로 AOP를 사용하고 있다고 생각할 수가 있다. 어느 정도 사용패턴이 정해져 있기 때문에 전략을 잘 세우고 소수의 AOP 개발자들만 수고하면 나머지 개발자들은 매우 편리하게 OOP에 집중해서 핵심로직을 개발할 수 있을 것이다.


마지막 네 번째 단계는 각 도메인에 특화된 AOP를 적용하는 단계이다. 이때부터는 개발자들이 매우 깊게 AOP를 사용하게 된다. 도메인이나 시스템에 특화된 로직의 많은 부분에 애스펙트를 적용할 수 있기 때문이다. 이때는Introduction(Inter-type declaration)이나 Mixin같은 고급기법을 이용할 수도 있다. 여기서 AOP와 OOP가 교차하는 부분이 생긴다. OOP의 추상화를 이용한 방법이 가능하지만 그것이 시스템의 설계를 복잡하게 한다면 AOP를 이용하는 것도 고려해볼만 하다. 예를 들어 OO디자인 패턴의 대표적인 옵저버패턴(Observer Pattern)은 객체지향적인 패턴을 활용해서 OOP로 구현할 수 있지만 경우에 따라서는 AOP를 사용하면 매우 간단하게 구현할 수도 있다.


단계적인 AOP 적용과 학습을 통해서 전체 팀이 AOP에 익숙해진 시점에서 이 네 번째 단계를 적용하기 시작하면 된다. AOP의 성공적인 적용에는 반드시 단계가 있다. 충분한 이해와 경험이 없이 처음부터 모든 영역에 AOP를 사용하려고 하는 것은 반드시 문제를 일으키게 마련이다.



AOP의 미래



필자가 AOP에 대한 기사를 처음 썼던 때에 비하면 지금은 AOP가 훨씬 많이 사용되고 있다. 특히 스프링프레임워크는 AOP를 널리 알리고 손쉽게 실무에 적용하게 해준 일등공신이다. 또 AspectWertz 프로젝트와 통합되고 Java5의 많은 특성을 적용한 AspectJ 5의 발전은 AOP 도입의 견인차 역할을 톡톡히 하고 있다. 최근엔 AspectJ와 SpringAOP도 긴밀하게 연동이 되어서 AOP 툴 간의 비호환 문제가 많이 해소되었다.


AOP를 이용하기 시작한 개발자들의 공통적인 반응은 “AOP가 없을 때는 어떻게 개발을 했는지 모르겠다”고 할 정도로 매우 뜨겁다. 최근에 등장한 많은 기술들의 공통점은 개발자들이 기술적인 것보다는 도메인의 비즈니스로직에 더 집중할 수 있도록 하는 것과 객체지향의 기본정신에 충실할 수 있도록 돕는 것이다. POJO 기반의 개발전략과 기술이 바로 그런 것들이다. 동시에 POJO 중심의 개발에서 AOP는 빠질 수 없는 필수 도구이다.


점점 복잡해지는 시스템의 요구사항을 기술적인 타협을 통해서 적당히 해결하고 후에 많은 문제를 떠안고 갈 것인가 아니면 처음엔 조금 부담이 되지만 AOP라는 멋진 대안을 선택해서 개발의 즐거움을 누릴 것인가 하는 것은 이제 개발자들의 선택에 달려있다.


참고자료

1) Aspect In Action, Ramnivas Laddad, Manning(2003)
2) One-on-one J2EE Development without EJB, Rod Johnson, Wrox(2004)
3) The AspectJ Project, http://www.eclipse.org/aspectj/
4) Spring 2.0 reference manual, http://static.springframework.org/spring/docs/2.0.x/reference/index.html
5) AOP@Work: AOP myth and realities, http://www-128.ibm.com/developerworks/java/library/j-aopwork15/
6) Introduction to Practical AspectJ Programming, SpringOne 2006
7) AOP in the enterprise, SpringOne 2006
8) Avoing AOP pitfalls, The Spring Experience 2006
9) AsoectJ for Spring Developer, The Spring Experience 2006

출처명 : 한국마이크로소프트 [2007년 3월호]
:
Posted by 뽀기
2007. 4. 9. 14:29

AOP(Aspect Oriented Programming) 용어들 그거/Tech2007. 4. 9. 14:29

aspect : 구현하고자 하는 횡단 관심사의 기능
cross cutting : 끼어들기
primary concern : 구현할 business logic
cross-cutting concern : 시스템 전반적으로 산재된 기능(보안, 인증, 로그 ...)
Advice : cross-cutting concern을 구현한 code(aspect의 실제 구현체)
point-cut : advice가 어떤 joint point에 적용되어야 하는지에 대한 정의
join-point : aspect를 플러그인할 수 있는 application의 실행지점
advisor : advice + point-cut
introduction : 기존의 클래스에 새로운 메소드나 속성을 추가
target : advice를 받는 클래스
proxy : 대상객체에 advice가 적용된 후 생성되는 객체
weaving : aspect를 대상객체에 적용하여 새로운 proxy 객체를 생성하는 과정

// 어렵다 -_-;

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

SunOS 5.9에 64bit JDK 1.5.0.11 설치  (0) 2007.04.16
선택이 아닌 필수 AOP(Aspect Oriented Programming)  (0) 2007.04.09
Aspect Oriented Programming (AOP)  (0) 2007.04.09
JSTL(JavaServer Pages Standard Tag Library)  (0) 2007.03.29
Velocity  (0) 2007.03.29
:
Posted by 뽀기
2007. 4. 9. 13:33

Aspect Oriented Programming (AOP) 그거/Tech2007. 4. 9. 13:33

저자: 김대곤(private @ roadtohome.com)

본 기사는 Aspect Oriented Programming에 대해 간략한 소개글이다. 아직까지는 생소한 분야일 수 있겠지만, 점점 더 많이 듣게 되리라 생각된다. AOP를 설명하는데 있어서 자주 등장하는 네 개의 용어들(Aspect, Cross-cutting concern, Point-cut, Advice) 를 설명함으로서 AOP가 왜 등장하게 되었으며, AOP가 제시하는 해결책에 대해 살펴볼 것이다. 먼저 "Aspect", "Oriented", "Programming"에서 생소한 단어는 단연 "Aspect"일 것이다. 야후 사전의 정의에 따르면, "Aspect"은 "사물의 면, 국면, 관점"으로 정의되어 있다. 소프트웨어 시스템은 여러가지 관점에서 바라볼 수 있다, 또는 여러 가지 단면을 가지고 있고 있다. 예를 들어, 자금 이체를 하는 프로그램을 작성한다고 생각해 보자. 출금계좌와 입금계좌, 그리고 이체금액을 입력받아 SQL 문장 또는 함수 한 번 돌리는 것으로 끝나는가? 절대 아니다. 먼저, 해킹을 방지하기 위해 사용자가 적절한 보안 프로그램을 설치했는지 점검하는 코드도 있어야 하고, 사용자가 인증되었는지 점검하는 코드도 써야 하고, 상대방 은행에서 적절하게 처리되었는지도 점점해야 하고, 혹시 사용자가 이체버튼을 두 번 누른 것은 아닌가 체크해야 하고, 시스템 로그도 남겨야 한다. 즉, 구현하려고 하는 기능 뿐 아니라 보안, 인증, 로그, 성능와 같은 다른 기능들도 녹아 있어야 한다. 어쩌면 이체를 위한 코드보다 잡다한 다른 측면의 문제들을 다루는 코드가 더 길어질 수 있다. 이런 코드들은 입금이나 출금 같은 다른 곳에서 들어가야 한다. 구현하려고 하는 비즈니스 기능들을 Primary(Core) Concern, 보안, 로그, 인증과 같이 시스템 전반적으로 산재된 기능들을 Cross-cutting concern이라고 부른다. AOP는 Cross-cutting concern를 어떻게 다룰 것인가에 대한 새로운 패러다임이라고 할 수 있다.

AOP는 구조적 방법론에서 객체지향 방법론으로 전환처럼 시스템 개발에 관한 전체적인 변화는 아니다. Object-Oriented Programming이 Aspect-Oriented Programming으로 대체되는 일은 없을 것이다. AOP는 구조적 방법론에도 적용될 수 있고, 다른 방법론에도 다 적용될 수 있지만, 주로 객체지향방법론이 가지는 단점을 보완하는 것으로 묘사되고 있다. 그러면 객체지향 프로그래밍이 또는 다른 이전의 프로그래밍 기법들이 Cross-cutting Concern를 어떻게 다루는지 알아보자. 매우 간단하다. Primary Concern를 구현한 프로그램에 함께 포함시켰다. 그것이 단 한 줄의 메소드 호출이라 하더라도. 많은 프로그래머들은 거의 모든 프로그램에 산재된 로그하는 단 한 줄의 코드를 찾아서 바꾸어 본 경험이 있을 것이다. 또는 간단하게 생각하고 프로그램을 수정하려고 했는데, 도데체 어디를 수정해야 되는지 모르게 코드가 길고, 알 수 없는 코드들이 자리를 차지하고 있을 때의 난감함. Primary concern, Cross-cutting concern이 하나의 프로그램 안에 들어가게 되면, 프로그램을 이해하기가 힘들고, Cross-cutting concern 코드가 여기저기에 산재되어 수정하기 힘들게 된다. 당연히 생산성 떨어지고, 품질 떨어지고, 유지보수 비용 많이 들게 된다.

그럼 AOP는 Cross-cutting concern를 어떻게 처리하는가? 이것도 매우 간단하다. 새로운 아이디어라고 할 수도 없다. Primary Concern 구현하는 코드 따로, Cross-cutting concern 구현하는 코드 따로 쓰고, 나중에 두 개 조합하게 완벽한 어플리케이션 만들겠다는 것이다. 기술 용어로 쓰면, Advice(Cross-cutting concern 구현한 코드)와 Primary concern 구현한 코드를 Point-cut 정보를 이용해서 Weaving(조합)하는 것이 AOP가 이 문제를 다루는 방법이다.

사용자 삽입 이미지


기술적 용어로서의 "Aspect"은 "Advice"와 "Point-cut"을 함께 지칭하는 단어이다. Point-cut은 어떤 Advice를 Code 어느 위치에 둘 것인가 하는 것이다. 예를 들면, 로그 기능을 구현한 Advice는 Code 속에 있는 모든 public 메소드가 수행되고 나면, 그 마지막에 실행되어라 라고 지정한 것이라 할 수 있다.

이전까지의 객체지향 프로그래밍은 Cross-cutting concern을 정적으로 어플리케이션에 결합시킨 반면 AOP는 동적으로 Cross-cutting concern를 다룬다고 표현하기도 합니다. 용어에서도 알 수 있듯이 AOP는 소프트웨어 엔지니어링 원칙 중에 하나인 "Separation of concern"를 구현하려고 하고 있습니다. 이러한 문제들을 다루고 있는 분야 중에 하나는 디자인 패턴할 수 있고, 예를 들어, Visitor 패턴은 정적인 구조를 동적으로 바꾸려고 합니다. AOP가 현재까지 나온 방법들 중에서 Cross-cutting concern를 다루는 가장 좋은 방법인가 하는 질문엔 아직 답하긴 힘들 것 같습니다. 그럼에도 분명 언제가는 책상 위에 관련 서적 한 권 있어야 할 것 같은 분야가 될 것 같습니다.

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

선택이 아닌 필수 AOP(Aspect Oriented Programming)  (0) 2007.04.09
AOP(Aspect Oriented Programming) 용어들  (0) 2007.04.09
JSTL(JavaServer Pages Standard Tag Library)  (0) 2007.03.29
Velocity  (0) 2007.03.29
DOM(Document Object Model)  (0) 2007.03.20
:
Posted by 뽀기
2007. 3. 29. 11:20

JSTL(JavaServer Pages Standard Tag Library) 그거/Tech2007. 3. 29. 11:20

JSTL

소개

  • J2EE 소형 클라이언트 기술인 JSP(JavaServer Pages)가 지난 몇 년 동안 널리 일반화되면서 독립적인 개발자들은 많은 사용자 정
    의 JSP 태그 라이브러리를 만들었습니다. 이러한 태그 라이브러리는 대부분 서로 다른 목표를 달성하도록 작성되었지만 반복, 조건 등의 일
    반적인 작업을 위한 유사한 솔루션을 제공하는 경향이 있습니다.
    유사하고 일반적이 문제점을 해결하는 독립적인 태그 라이브러리에 대한 필요성을 줄이기 위해 Java Community Process(JSR 52)의 지
    원하에 JSTL(JavaServer Pages Standard Tag Library)이 개발되었습니다. JSTL은 이러한 일반 기능을 처리하는 하나의 표준 솔루션
    을 제공합니다. (말그대로 표준태그라이브러리)
  • JSTL의 주요 강점 중 하나는 서블릿 컨텍스트에 저장된 데이타 같은 애플리케이션 데이타를 액세스 및 조작하는 쉬운 방법을 제공하는 간
    단한 EL을 사용할 수 있다는 것입니다.

EL 에 대하여

설치 방법

  • http://cvs.apache.org/builds/jakarta-taglibs/nightly/ 에서 다운
    \jakarta-taglibs-20051024\jakarta-taglibs\standard\lib
    에서 jstl 과 standard 파일 을 이 두개의 jar 파일을 우리의 웹애플리케이션의 /WEB-INF/lib 폴더에 넣습니다
    그 다음 tld 폴더의 tld 파일을 /WEB-INF/lib/tld 폴더 아래 넣습니다.
  • web.xml 에
     
        <!-- jstl 1.2 taglig -->
         <taglib>
           <taglib-uri>jstl-c</taglib-uri>
           <taglib-location>/WEB-INF/tlds/jstl/c.tld</taglib-location>
        </taglib>
        <taglib>
           <taglib-uri>jstl-fmt</taglib-uri>
           <taglib-location>/WEB-INF/tlds/jstl/fmt.tld</taglib-location>
        </taglib>
        <taglib>
          <taglib-uri>jstl-fn</taglib-uri>
          <taglib-location>/WEB-INF/tlds/jstl/fn.tld</taglib-location>
        </taglib>
        

    를 추가한다.

  • jsp 에 추가
     
         <%@ taglib uri="jstl-c" prefix="c" %>
        <%@ taglib uri="jstl-fmt" prefix="fmt" %>
        <%@ taglib uri="jstl-fn" prefix="fn" %>
         

사용 예(기본 문법)

Area Subfunction Prefix Description
Core Variable support c 변수지원
Core Flow control c 흐름제어
Core URL management c URL 처리
Core Miscellaneous c  
XML Core x XML 코어
XML Flow control x 흐름 제어
XML Transformation x XML 변환
I18n Locale fmt 지역
I18n Message formatting fmt 메시지 형식
I18n Number and date formatting fmt 숫자 및 날짜 형식
Database SQL sql SQL
Functions Collection length fn 콜렉션 처리
Functions String manipulation fn String 처리

http://java.sun.com/products/jsp/jstl/1.1/docs/tlddocs/index.html 참고

  • 변수지원태그
    set: <c:set var="varName" scope="session" value="someValue">
    var속성은 값을 지정할 변수의 이름
    <c:set t a r g e t ="" property="userName" value="someValue">
    target : 빈 프로퍼티나 맵 값을 설정한다.(Object)
    var와 target을 동시에 사용할 수 없다.
    scope속성은 변수가 위치하는 영역(page,request,session,application)
    value :저장하는 값
    remove :<c:remove var="varName" scope="session">
    var :삭제할 변수의 이름
    scope 속성은 삭제할 변수의 영역
    out : <c:out value="">
    value속성은 출력하려는 값
    catch : <c:catch var="">
    </c:catch>
    예외사항이 한번이라도 발생시 </c:catch>로 점프
    var에 정의 된 객체를 페이지 생존범위에 자동으로 묶어 나중에 var에 정의된 변수이름을 사용할 수 있다.
    예외 발생시
    : var 속성 사용시 exception 객체를 설정.
    <c:catch> 문 밖으로 제어가 떨어진다
 
    <c:set var="num1" value="${20}" />
	<c:set var="num2">
		10.5
	</c:set>
	
	<c:set var="today" value="<%= new java.util.Date()%>" /><br>

	변수 num1 = ${num1} <br>
	변수 num2 = ${num2} <br>
	num1 + num2 = ${num1+num2}<br>
	오늘은 ${today}입니다.
	
	<c:remove var="num1" scope="page" />
	
	<p>
	삭제한 후의  num1=${num1} <br>
	삭제한 후의 num1 + num2 = ${num1 + num2}
   
 
       <c:catch var="myException">
     	It's catch
         <% int x = 10/0; %>
         실행안됨.
       </c:catch>
       <c:if test="${myException != null}">
           발생된 예외는 : ${myException.message} <br>
       </c:if>
     
  • URL 관련
    import : <c:import url=""/>
    url속성에 명기한 파일을 현재 컨텐츠에 포함
    param : <c:param name="" value=""/>
    <jsp:param />과 같은 역할
    url : <c:url value="" var=""/>
    value에 들어있는 상대 경로 뒤에 jsessionid를 추가한다.(쿠키를 사용 못하는 경우)
    var : 옵션 속성으로 url을 참조하기 위해쓴다.
    redirect :<c:redirect url="' context=""/>
    context : url경로의 이름
 
    <c:import url="Header.jsp" >
	<c:param name="subTitle" value="This is subTitle"/>
    </c:import>
   
  • 흐름제어 태그
    if : <c:if test="조건"> </c:if>
    test속성의 값에는 "조건"이 오는데 이 조건문의 결과값이 true 면 처리
     
       <c:if test="true">
         무조건 수행<br>
        </c:if>
    
        <c:if test="${param.name == 'bk'}">
          name 파라미터의 값이 ${param.name}입니다 <br>
        </c:if>
    
        <c:if test="${param.name eq 'bk'}">
          name 파라미터의 값이 ${param.name}입니다 <br>
        </c:if>
    
        <c:if test="${18 < param.age}">
    	 당신의 나이는 18세 이상입니다.
        </c:if>   
       

    choose,when,otherwise : <c:choose>
    <c:when test="조건">
    </c:when>
    <c:otherwise>
    </c:otherwise>
    </c:choose>
    choose 태그는 자바의 switch 문과 if-else 의 혼합한 형태, 다수의 조건문을 하나의 블록에서 수행하고자 할때 사용
    -> switch문과의 차이점은 중간에 빠져 나가지 못한다는 것이다.
    -> when 의 어느 하나에도 실행되지 않을때 otherwise 실행
    -> otherswise 태그가 반드시 있어야 하는것은 아니다.

 
     <c:choose>
       <c:when test="${param.name == 'bk' }">
	<li>당신의 이름은 ${param.name}입니다.
     </c:when>
     <c:when test="${param.age > 18 }">
	<li>당신은 18세 이상입니다.
     </c:when>

     <c:otherwise>
	<li> 당신은 'bk'가 아니고 18세 이상이 아닙니다.
     </c:otherwise>
     </c:choose> 
    

forEach : <c:forEach var="변수" items="아이템" begin="시작값" end="끝값" step="증가값">
</c:forEach>
item 속성에 올수 있는 것들로는 Map,배열,Collection 이 있다.
varStatus는 javax.servlet.jsp.jstl.core.LoopTagStatus 객체 인스턴스변수를 만들며 count라는 프로퍼티가 있어 몇번의 회전인지 알 수있다.

 
       <c:forEach var="i" begin="1" end="9">
	<li>4 *${i} = ${4 *i}
       </c:forEach>

       <h4>int 형 배열</h4>

       <c:forEach var="i" items="${intArray}" begin="2" end="4">
	[${i}]
       </c:forEach>

       <h4>Map</h4>
       <c:forEach var="i" items="${map}">
	  ${i.key} = ${i.value}<br>
       </c:forEach>

       <c:forEach var="member" items="${memberList}" varStatus="memberLoopCount">
	  회원 $(memberLoopCount.count} : ${member} <br>
       </c:forEach>
     

forTokens : <c:forTockens var="token" items="문자열" delins="구분자">
</c:forTockens>
forTokens 태그는 StringTokenizer 와 같은 기능을 제공한다.

 
       <c:forTokens var="token" items="빨강색, 주황색, 노란색, 초록색, 파랑색, 남색, 보라색" delims=",">
     	${token}<br>
       </c:forTokens>
     
  • 숫자 및 날짜 지원 형식
    The JSTL formatting actions allow various data elements in a JSP page, such as numbers,dates and times
    to be formatted and parsed in a locale-sensitive or customized manner.

formatNumber : 숫자 형식을 표현

 
 
      number  : <fmt:formatNumber value="9876543.61" type="number"/>
      currency: <fmt:formatNumber value="9876543.61" type="currency"/>
      percent : <fmt:formatNumber type="percent">9876543.61</fmt:formatNumber>

      pattern=".000"    :<fmt:formatNumber value="9876543.61" pattern=".000" />
      pattern="#,#00.0#":<fmt:formatNumber value="9876543.612345" pattern="#,#00.0#"/>
    

parseNumber : 정해진 패턴을 문자열에서 수치를 파싱해내는 태그
formatDate :날짜 형식을 표현

 
      <jsp:useBean id="now" class="java.util.Date"/>
	
         <c:out value="${now}"/>
           date: <fmt:formatDate value="${now}" type="date"/>
           time: <fmt:formatDate value="${now}" type="time"/>
           both: <fmt:formatDate value="${now}" type="both"/>

           default:<fmt:formatDate value="${now}"
                        type="both" dateStyle="default" timeStyle="default"/>
           short  :<fmt:formatDate value="${now}"
                        type="both" dateStyle="short"   timeStyle="short"  />
           medium :<fmt:formatDate value="${now}"
                        type="both" dateStyle="medium"  timeStyle="medium" />
           long   :<fmt:formatDate value="${now}"
                        type="both" dateStyle="long"    timeStyle="long"   />
           full   :<fmt:formatDate value="${now}"
                        type="both" dateStyle="full"    timeStyle="full"   />

          pattern="yyyy년MM월dd일 HH시mm분ss초"
             <fmt:formatDate value="${now}" type="both"
                             pattern="yyyy년MM월dd일 HH시mm분ss초"/>
            
         <fmt:formatDate value="${now}" pattern="yyyy/MM/dd" />

parseDate :정해진 패턴의 문자열에서 날짜를 파싱해내는 태그
timeZone : <fmt:timeZone value=""/>

setTimeZone : <fmt:timeZone value="" var="" scope=""/>

  • 국제화
    message <fmt:message
    setLocale <fmt:setLocale
    bundle <fmt:bundle
    setBundle <fmt:setBundle
    param <fmt:param
    requestEncoding <fmt:requestEncoding
  • SQL
    <sql:query sql="sqlQuery" var="varName" [scope="{page|request|session|application}"]
    [dataSource="dataSource"] [maxRows="maxRows"] [startRow="startRow"]>
    <sql:param>
    </sql:query>
         <sql:query var="customers" dataSource="${dataSource}">
          SELECT * FROM customers
          WHERE country ='China'
          ORDER BY lastname
         </sql:query>
         
         <table>
          <c:forEach var="row" items="">
             <tr>
               <td><c:out value="${row.lastName}"/></td>
               <td><c:out value="${row.firstName}"/></td>
               <td><c:out value="${row.address}"/></td>
             </tr>
          </c:forEach>
         </table>
       

<sql:update>
<sql:setDataSource>
<sql:param>
<sql:dateParam>

  • XML 코어
    <x:parse>
    <x:out>
    <x:set>
  • 흐름제어
    <x:if>
    <x:choose>
    <x:when>
    <x:otherwise>
    <x:forEach>
  • XML 변환
    <x:transform>
    <x:param>
  • function
    contains
    containsIgnoreCase
    endsWith
    escapeXml
    indexOf
    join
    length
    replace
    split
    startsWith
    substring
    substringAfter
    substringBefore
    toLowerCase
    toUpperCase
    trim
         <c:if test="${fn:contains(name, searchString)}">
         <c:if test="${fn:containsIgnoreCase(name, searchString)}">
         <c:if test="${fn:endsWith(filename, ".txt")}">
         ${fn:escapeXml(param:info)}
         ${fn:indexOf(name, "-")}
         ${fn:join(array, ";")} 
         You have ${fn:length(shoppingCart.products)} in your shopping cart.
         ${fn:replace(text, "-", "•")}
         ${fn:split(customerNames, ";")}
         <c:if test="${fn:startsWith(product.id, "100-")}">
         P.O. Box: ${fn:substring(zip, 6, -1)}
         P.O. Box: ${fn:substringAfter(zip, "-")}
         Zip (without P.O. Box): ${fn:substringBefore(zip, "-")}
         Product name: ${fn.toLowerCase(product.name)}
         Product name: ${fn.UpperCase(product.name)}
         Name: ${fn.trim(name)}
       

출처 : 인터넷

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

AOP(Aspect Oriented Programming) 용어들  (0) 2007.04.09
Aspect Oriented Programming (AOP)  (0) 2007.04.09
Velocity  (0) 2007.03.29
DOM(Document Object Model)  (0) 2007.03.20
패턴과 프레임워크  (0) 2007.03.14
:
Posted by 뽀기
2007. 3. 29. 11:18

Velocity 그거/Tech2007. 3. 29. 11:18

Velocity

Velocity

소개

  • 벨로시티란 자바 기반의 템플릿 엔진입니다.
    벨로시티를 활용하면 간단하면서도 강력한 템플릿 언어를 통하여 자바 코드에 정의된 객체를 액세스할 수 있습니다.
    벨로시티를 웹 개발에 사용하면, 웹 디자이너는 자바 프로그래머와 병렬로 작업을 할 수 있으며 MVC(모델-뷰-컨트롤러) 모델에 따라 웹 사이트를 개발할 수 있습니다. 더 자세히 설명하면 웹 페이지 디자이너의 경우 보기 좋은 사이트를 만드는 데만 집중하면 되고, 프로그래머는 잘 동작하는 코드를 만드는 데만 집중하면 된다는 뜻입니다.
    벨로시티는 웹 페이지와 자바 코드를 분리하여, 장기적인 측면에서 볼 때 웹 사이트를 손쉽게 유지보수할 수 있도록 하고, 자바 서버 페이지 (JSP) 또는 PHP를 대체할 수 있는 방안을 제시합니다. 벨로시티의 쓰임새는 웹 사이트에 국한되지 않습니다. 예를 들면, 템플릿으로부터 SQL이나 포스트스크립트, 또는 XML(XML 변환에 대해서는 벨로시티 툴 중 하나인 아나키아(Anakia)를 참조)문서를 생성하는 데 쓰일 수 있습니다. 벨로시티는 스탠드얼론 유틸리티처럼 사용하여 소스 코드나 리포트를 생성할 수도 있고, 다른 시스템의 컴포넌트로 통합할 수도 있습니다. 또한 벨로시티는 터빈 (또다른 자카르타 서브 프로젝트 중 하나) 웹 애플리케이션 프레임웍에 템플릿 서비스를 제공합니다. 벨로시티와 터빈을 조합하면 진정한 MVC 모델에 따라 웹 애플리케이션을 개발할 수 있습니다

설치 방법

  • web.xml 수정
     
       <servlet>
         <servlet-name>velocity</servlet-name>
            <servlet-class>org.apache.velocity.tools.view.servlet.VelocityViewServlet
         </servlet-class>
         <init-param>
            <param-name>org.apache.velocity.toolbox</param-name>
            <param-value>/WEB-INF/velocity-toolbox.xml</param-value>
         </init-param>
         <init-param>
            <param-name>org.apache.velocity.properties</param-name>
            <param-value>/WEB-INF/velocity.properties</param-value>
           </init-param>
         <load-on-startup>10</load-on-startup>
       </servlet>
       <servlet-mapping>
          <servlet-name>velocity</servlet-name>
          <url-pattern>*.vm</url-pattern>
       </servlet-mapping>
     
  • 파일 생성
    velocity.properties 파일을 생성한 후 web_inf 아래 둡니다.(생성하지 않아도 무관합니다.)
    velocity-toolbox.xml 을 생성 한 후 web_inf 아래 둡니다.
    velocity-toolbox.xml
    <?xml version="1.0"?> 
    <toolbox>
    <data type="number">
    <key>version</key>
    <value>1.1</value>
    </data>
    <tool>
    <key>date</key>
    <scope>application</scope>
    <class>org.apache.velocity.tools.generic.DateTool</class>
    </tool>
    </toolbox>

사용 예(기본 문법)

Velocity Template Language(VTL) 은 Template 에서 사용되는 Velocity 고유의 언어를 의미합니다.

  • References(참조형)
    Variables(변수) – 다음과 같이 $를 먼저 쓰고 그 뒤에 식별자를 적어주는 방식으로 사용
    ex) $foo
    Property(특성) – $ 다음에 식별자를 쓰고, 마침표(.)후에 다시 식별자의 형태로 사용
    ex) $foo.name
    Method(메소드) - $다음에 식별자를 쓰고 마침표 후에 호출할 메소드의 이름을 적는다
    ex)$foo.getName()
  • Directive(지시형)
    #set – reference 의 값을 설정한다.
    #if/elseif/else – 조건문 제어
    #foreach –반복문 제어
    #include – velocity 로 파싱되지 않는 파일의 출력
    #parse –velocity 로 파싱된 파일 출력
    #stop –template 엔진의 정지
    #macro – 반복적으로 사용할 vm정의
  • Comment (주석)
      ## - 한줄짜리 주석
      #* … *# 여러줄 짜리 주석
      
     ##This is an example velocity template
     #set($this = “Velocity”)
       $this is great! But It’s so hard.

     #foreach($name in $list)
       $name is great!
     #end

     #set($condition = true)
     #if ($condition)
        The condition is true!
     #else
       The condition is false
     #end
  

http://velocity.apache.org/engine/releases/velocity-1.5/getting-started.html

Tool box 에 대해

VelocityTools is a collection of Velocity subprojects with a common goal of creating tools and infrastructure for building both web and non-web applications using the Velocity template engine.

  • Generic Tool (http://jakarta.apache.org/velocity/tools/generic/)
    *DateTool : A tool for manipulating and formatting dates
    *MathTool :A tool for performing floating point math.
    *NumberTool :A tool for formatting numbers
    *IteratorTool :A convenience tool to use with #foreach loops. It wraps a list to let the designer specify a
    condition to terminate the loop, and reuse the same list in different loops.
    *RenderTool:A tool to evaluate and render arbitrary strings of VTL (Velocity Template Language).
          Example uses:
      $date                         -> Oct 19, 2003 9:54:50 PM
      $date.long                    -> October 19, 2003 9:54:50 PM PDT
      $date.medium_time             -> 9:54:50 PM
      $date.full_date               -> Sunday, October 19, 2003
      $date.get('default','short')  -> Oct 19, 2003 9:54 PM
      $date.get('yyyy-M-d H:m:s')   -> 2003-10-19 21:54:50
     
      $myDate                        -> Tue Oct 07 03:14:50 PDT 2003
      $date.format('medium',$myDate) -> Oct 7, 2003 3:14:50 AM 
    
    

출처 : 인터넷, (일부 최근 정보에 맞게 수정하였습니다.)

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

Aspect Oriented Programming (AOP)  (0) 2007.04.09
JSTL(JavaServer Pages Standard Tag Library)  (0) 2007.03.29
DOM(Document Object Model)  (0) 2007.03.20
패턴과 프레임워크  (0) 2007.03.14
Crossing borders: JavaScript의 특징 (한글)  (0) 2007.03.13
:
Posted by 뽀기
2007. 3. 20. 10:36

DOM(Document Object Model) 그거/Tech2007. 3. 20. 10:36

DOM

DOM(Document Object Model)

  • W3C's definition

    "The Document Object Model is a platform- and language-neutral interface that will allow programs and scripts to dynamically access and update the content, structure and style of documents.

  • DOM 의 역할
    • dynamic access and update
      • XML/HTML 문서의 접근 및 수정 => 내용/구조/스타일 정보의 검색 및 수정
      • 대상 문서 : XML1.0 또는 HTML4.0, 기타 웹문서
      • 컨텐츠의 조작 : 문서 요소에서 text 등 컨텐츠의 검색/질의, 추가/수정/삭제
      • 구조의 탐색 및 조작 : 각 요소와 속성에 대한 검색/질의, 추가/수정/삭제
    • interface
      • 응용 프로그램 인터페이스 (API) - 각종 메소드 및 속성을 정의
      • 플랫폼 및 언어 중립적(스크립트 포함) : Java, JavaScript, ASP, ...
  • DOM level
    • DOM level 1 : 1998.10 W3C 표준안
    • DOM level 2 : 2000.11 W3C 표준안
    • DOM level 3 : 2001.8 W3C Working Draft

DOM and XML Parser

  • Parser 의 역할
    • XML 문서를 읽고 해석 : well-formed, valid 검사
    • 응용프로그램 개발시 파서 사용 이유
      • 파서가 메모리에 DOM 트리를 생성 : XML 문서트리와 일치
      • 세부적인 XML 문법으로부터 프로그램 격리

  • [참고] SAX 기반 Parser

DOM 구조적 모델

  • DOM 트리에서 노드/객체의 종류
    • Document : 문서 객체, 최상위 노드
    • Element, Attribute : 문서의 구조를 구성하고 있는 요소
    • Text : 컨텐츠의 내용, 항상 단말 노드
    • Collection : 일종의 노드 집합
  • DOM 트리의 예

    <parent>
        <child  id="123">text here</child>
     </parent>

     

  • 인터페이스(API)의 예
    • 객체의 속성과 메소드를 사용하기 위한 사양
    • DOM 인터페이스 예
      - 문서.childNodes[1].nodeName
      - 문서.firstChild.firstCild.firstCild.nodeName
      - 문서.firstChild.firstCild.firstCild.nodeValue
  • DOM Core Interface (Object Hierachy) 교재 p.414 (표9-3)

DOM 주요 API

  • DOM 인터페이스의 공통 속성
    • type, name, value
    • 예) Node 객체의 경우 nodeType, nodeName, nodeValue 속성
      • nodeType은 위 그림과 같이 여러 가지가 있다 (p.418 표 9-6, 9-7)
  • 주요 객체/속성/메소드 - 진행하면서 지속적으로 참조

    객체(Object)

    속성(Properties)

    메소드(methods) 

    Node 객체

    nodeName, nodeType, nodeValue, childNodes, parentNode, childNode, firstChild, lastChild, previousSibling, nextSibling, attributes, ownerDocument, ..., 
    (
    text, xml) [표9-8]

    [표9-10] 노드 정보 구하기
    getNodeName, ..., getAttributes, ...
    [표 9-11] 문서 조작
    appendChild, insertBefore,
    removeChild, replaceChild, cloneNode
    [표 9-12] 트리 순회 관련
    getParentNode, getChildNode, ...
    hasChildNodes, ...

    Document 객체

    doctype,
    documentElement, implementation, ...

    (async, readyState)
    * W3C 표준이 아니라  MS에서 제공하는 인터페이스

    [표9-13] 문서관련 정보
    egtDoctype, getImplementation, ...
    [표 9-14] 트리 순회 관련
    getDocumentElement, getElemenmtByID,
    getElementByTagName, ...
    [표 9-15] 문서 작성
    createElement, createAttribute, createTextNode, createCDATASection, createComment, createEntityReference, ...

    DOMImplementation 객체


    [표 9-16] hasFeature, createDocument, ...

    DocumentFragment 객체

    * Node 객체와 동일

    * Node 객체와 동일

    NodeList 객체

    length

    [표 9-17] getLength, item

    Element 객체

    tagName

    [표 9-18] Element의 속성에 접근
    getAttribute, setAttribute, getAttributeNode, setAttributeNode, removeAttribute, ...,
    [표 9-18]Element 객체에 접근
    getTagName, hasAttribute, ...

    NamedNodeMap 객체

    length

    [표 9-19] getNamedItem, setNamedItem, removeNamedItem, item, getLength

    Attribute 객체

    name, value

    [표 9-20] getName, getValue, setValue, ...

    CharacterData 객체

    data, length

    [표 9-21] appendData, deleteData, insertData, replaceData, substringData, ...

DOM 프로그래밍 시작 - Document 객체

  • 문서 객체 새로 만들기 : DOMDocument 객체를 생성
    • Msxml.DOMDocument 객체를 새로 만들기
    • 또는 HTML에서 <xml> 태그 이용

    JavaScript (JScript)

    <Script language="Javascript">
       
    var xdoc1,xdoc2
       
    xdoc1 = new ActiveXObject("Msxml.DOMDocument");
       
    xdoc2 = new ActiveXObject("Msxml.DOMDocument");
        ...
    xdoc1.load("ex08.xml");  xdoc2.load("ex09.xml");
    </script>
    VBScript 의 경우 <Script language="VbScript">
      Dim xdoc1,xdoc2
      Set xdoc1 = CreateObject("Msxml.DOMDocument")
      Set xdoc2 = CreateObject("Msxml.DOMDocument")
      ...
    xdoc1.load("ex08.xml"); xdoc2.load("ex09.xml");
    </Script>

    HTML에서 <xml>  태그 이용

    - MSXML 파서 설치안한 경우

    <HTML> <HEAD>
        <Script language="Javascript">
            xdoc.load("ex08.xml");    </script>
    </HEAD>
    <BODY>
        <xml id="xdoc1"></xml>
        <xml id="xdoc2" src="ex09.xml"></xml>
    </BODY> </HTML>

    Java의 경우

    import java.xml.parsers.*;
    ...
    class xxxxxx
    {
      public static void main(String[] args) throws Exception
      {
        DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance();
        DocumentBuilder  db = dbf.newDocumentBuilder();
        Document  xdoc1 = db.parse(new FileInputStream(arg[0]));
    ...

  • 기존의 XML 문서 읽기
    • DOMDocument 객체 읽기 - async 속성, load 메소드, xml 속성 : MSXML에서 제공
    MSXML 파서 설치한 경우   MSXML 파서 설치안한 경우  
    <HTML>
    <HEAD>
    <Script language="Javascript">
    function xload0()
    {
       
    var xdoc = new
            ActiveXObject("Msxml.DOMDocument");
        xdoc.
    async = false;    xdoc.load("ex08.xml");
       
    alert(xdoc.xml);
    }
    </script>
    </HEAD>
    <BODY>
    <input type="button" value="XML 로드0"
        onClick="xload0()">
    </BODY>
    </HTML>  
    <HTML>
    <HEAD>
    <Script language="Javascript">
    function xload1()
    {
        xdoc.async = false;
        xdoc.load("ex08.xml");

         alert(xdoc.xml);
    }
    </script>
    </HEAD>
    <BODY>
    <input type="button" value="XML 로드1"
        onClick="xload1()">
    <xml id="xdoc"></xml>
    </BODY>
    </HTML>
    VBScript 의 경우
    <Script language="VbScript">
      Dim xdoc
      Set xdoc = CreateObject("Msxml.DOMDocument")
      xdoc.
    async = False;
      xdoc.
    load("ex08.xml");  MsgBox  xdoc.xml
    </Script>
  • 신규 XML 문서의 작성
    • loadXML 메소드  
     xdoc.async = false;
     xdoc.
    loadXML( "<book><title>XML 입문</title><author>일지매</author></book>");
     alert(xdoc.xml);  
     xdoc.async = false;
     xdoc.
    loadXML( "<book> <title> XML 입문 </title> <author> 일지매 </author> </book>");
     alert(xdoc.xml);  
  • 공백의 처리 : preserveWhiteSpace 속성
     xdoc.async = false;
     xdoc.preserveWhiteSpace = true;
     xdoc.loadXML( "<book> <title> XML 입문 </title> <author> 일지매 </author> </book>");
     alert(xdoc.xml);  
     xdoc.async = false;
     xdoc.preserveWhiteSpace = true;
     xdoc.load("ex08.xml");
     alert(xdoc.xml);  
  • XML 문서의 저장 : save 메소드
  • 에러 처리 : parseError 객체
    • parseError.errorCode, parseError.line, parseError.linepos, parseError.reason
     xdoc.async = false;
     xdoc.loadXML( "<book> <title> XML 입문 </title> <author> 일지매 </authors> </book>");
     alert(xdoc.xml);  
     xdoc.async = false;
     xdoc.loadXML( "<book> <title> XML 입문 </title> <author> 일지매 </authors> </book>");
     if (xdoc.parseError)
        alert("에러 위치 : " + xdoc.parseError.line + "번째 라인 " + xdoc.parseError.linepos 
                  + "번째 문자 에러 이유 : " + xdoc.parseError.reason);
     else
    alert(xdoc.xml);  
  • 루트 노드 찾기 (루트 에리먼트)
    • documentElement 속성
     xdoc.async = false;
     xdoc.load("ex08.xml");
     var xroot = xdoc.
    documentElement;
     alert(xroot.nodeName);  
     xdoc.async = false;
     xdoc.loadXML( "<book> <title> XML 입문 </title> <author> 일지매 </author> </book>");

     var xroot = xdoc.
    documentElement;
     alert(xroot.nodeName);  

Node 객체의 정보구하기 - Node 객체

  • 속성 : nodeName, nodeType, nodeValue, attributes, text 속성
  • nodeType
    • 1 (element), 2 (attribute), 3 (text) , 4 (CDATA), 5 (Entity Reference)...
  •  xdoc.load("ex08.xml");
     var xroot = xdoc.documentElement;
     alert('nodeName: '+xroot.nodeName+' nodeType: '+xroot.nodeType+
           ' nodeValue: '+xroot.nodeValue+' attributes: '+xroot.attributes.length);
     alert('xroot.text : ' + xroot.text);
     

실습 프로그램

  • 파일 LOAD 및 DOM 명령 실행 (교재 s20-03.htm 과 s20-05.htm 혼합)

    XML 파일 경로를 직접 입력하거나  '찾아보기'로 선택, 'LOAD'로 파일을 메모리에 로드
    경로 :  =>

    또는 미리 작성되어 있는 파일 사용하기  

    사용하고자 하는 DOM 구문을 아래에 입력하고 '확인' 버튼을 클릭  (예 : xdoc.text, xdoc.documentElement.firstChild.nodeName, ...) 구문 :    

     <SCRIPT language="Javascript">
            var xdoc, rootNode;
            function
    FileLoad(filename)
            {
                    xdoc = new ActiveXObject("Msxml.DOMDocument");
                    xdoc.async = false;
                    if (filename) xdoc.load(
    filename);
                    else xdoc.load(
    "file://"+path.value);
                    if (xdoc.parseError.errorCode != 0)
                        alert("파일을 메모리로 로드하는데 실패하였습니다 : " + xdoc.parseError.reason);
     
                    rootNode = xdoc.documentElement;
                    alert("[파일로드 성공] 루트 엘리먼트 : " + rootNode.nodeName);
            }
            function Execute()
            {
                    var selectionString;
                    try {
                            selectionString = eval(syntax.value);
                    } catch(e) {
                            selectionString = null;
                    }
                    alert(selectionString);
            }
    </SCRIPT>  
     ...
    <BODY>
     ... 경로 : <input  type="file"  size="40"  id="
    path">
         <input  type="button"  value="LOAD"  onclick="
    FileLoad()">
     ... 미리 작성되어 있는 파일
         <input  type="button"  value="서점 책 list"  onclick="
    FileLoad("ex08.xml")">
     ... 구문 : <input  size="50"  id="
    syntax"  value="xdoc.">&nbsp;
         <input  type="button" onclick="
    Execute()" value=확인>
     ...
    </BODY>

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

JSTL(JavaServer Pages Standard Tag Library)  (0) 2007.03.29
Velocity  (0) 2007.03.29
패턴과 프레임워크  (0) 2007.03.14
Crossing borders: JavaScript의 특징 (한글)  (0) 2007.03.13
JSON vs. XML  (0) 2007.03.13
:
Posted by 뽀기