태터데스크 관리자

도움말
닫기
적용하기   첫페이지 만들기

태터데스크 메시지

저장하였습니다.

현대 프로그래밍 언어나 프래임웍중에 스캐폴딩이라는 개념이 있다. 


이게 원래의 사전적인 의미는 다음과 같다.



출처: http://experientia.co.kr/?p=244


스캐폴딩(scaffolding)의 개념

– 스캐폴딩이란 교수자가 학생에 대하여 적절한 교수학습적 도움을 제공함으로써 유의미한 학습을 수행할 수 있도록 하는 것으로, 특히 학습자들로 하여금 자신의 현재 기능수준을 넘어서 다음단계의 수준까지 이르도록 돕기 위해 제공되는 지원


프로그래밍 언어에서의 스캐폴딩은 Data Model을 이용해서 CRUD(Create/Read/Update/Delete) 동작을 자동으로 제어해주는 코드를 생성해주거나 기능을 제공하는 것을 이야기한다.  Ruby on rails 에서 이런 기능을 제공하면서 인기를 끌게 되면서 다른 언어나 프레임웍에서도 비슷한 기능들을 제공하기 시작했다. 


PHP yii 프레임웍이나 나 groovy on grails에서도 이런 비슷한 기능들이 제공된다. 


사전적 의미에서 처럼 이런 기능을 제공하는 의미는 기본적인 코드를 제공함으로써 생산성 향상과 튜토리얼의 의미 모두를 만족시키기 위함이 아닐까 생각이 된다. 


웹개발에서 이런 CRUD개발이 50%이상의 시간을 쓰는점을 본다면 이 코드를 기반으로 개발을 시작하게 되면 좀더 빠른 생산성을 가져갈수 있을것이다.


또한 관리자툴 같은곳에서는 이 기능만 활용해도 거의 대부분의 기능을 구현해 낼수가 있다. 


현재 진행하는 프로젝트에서도 운영자툴을 위하여 이러한 솔루션을 몇가지 검토를 해보았는데, 


Spring Roo와 OpenXava 이 두가지가 가장 좋아 보였다. 



두 프로젝트 모두 Model 클래스를 통하여 CRUD기능을 하는 화면까지 생성을 해준다. 


OpenXava의 경우는 코드를 생성해주는 방식은 아니고 모델을 런타임에 분석하여 해당 기능들을 제공하는 방식이다. 





이 프로젝트들의 단점은 무엇보다 도큐먼트의 부족이다.


특히 Spring roo의 도큐먼트는 매우 부족한 수준이다. 

그리고 프로젝트 초기에 이런 프레임웍으로 개발하겠다고 결정하고 시작을 해야 생산성 향상에 도움이 될것 같다. 프로젝트 중간에 이런 툴들을 integration시키려면 여러가지 문제가 속출해서 그것을 해결하다가 보니 오히려 생산성 저하가 오는것 같다. 


여러가지 이유로 OpenXava를 도입하려고 했었는데 이것도 이것 자체가 웹프레임웍이라서 spring MVC와 연동을 하기에는 쉽지 않고, 커스터마이징 기능을 개발하기에도 이미 개발해놓은 코드들을 재사용하지 못해 불편함이 따른다. 


관리자툴 개발에 시간을 절약하기 위하여 도입검토를 한건데 이런식이니 그냥 간단하게 비슷한 역할을 해주는 것을 개발하기로 결정했다. 


향후 시작하는 프로젝트에서는  시작전에 미리 검토를 하고 시작해야 겠다. 







Posted by Breeze.Kang

댓글을 달아 주세요






일명 배치잡(Batch Job) 이라고 하는 주기적으로 구동시켜야 하는 작업들을 Spring에서 구동 시키기 위하여서는 Spring Batch 라는 프로젝트가 따로 존재 합니다. 




Spring Batch가 배치작업의 끝판왕이라면 Spring이 주로 사용되기 이전부터 Quartz라는 라이브러리를 java쪽에서는 많이 사용을 했었는데요. 


Spring에서도 이 Quartz를 지원해서 간단하게 주기적인 배치작업을 구동시킬수 있습니다. 




이렇게 스케줄러를 이용해서 Spring project상에서 배치작업을 Invoke시키는 방식은 통합적인 코드로 스케줄 작업들을 관리할수 있기 때문에 관리 측면에서 유리한 점이 있습니다. Spring 프로젝트 배포 시스템에서 따로 설정을 해주거나 관리를 해줄 필요가 없구요. 


반면 WAS서버가 여러대가 동시에 구동되는 실제 대용량 환경을 고려해본다면 실제 배치잡이 한군데에서만 구동되어야 하는 경우가 있을 것입니다. 두개의 인스턴스가 동시에 구동이 된다면 로직상에 문제가 생길만한 작업의 경우에 말이죠. 


예를 들면 주기적으로 보상을 해주거나 리셋을 해야 한다거나 통계를 내야 한다거나 하는 작업들이 있겠네요. 


이렬경우는 배치작업이 셋팅된 프로젝트를 따로 분리하여 한대의 서버에만 설치를 한다거나,  한대의 서버에서만 배치작업이 구동되게 설정을 바꾼다거나 하는 작업이 필요 해집니다. 


이럴경우 좀 번거롭게 되는것이 사실입니다. 


그래서 그냥 간단하게 외부 라이브러리 없이 Async하게 작업을 구동시키는 컴포넌트를 만들고,  내부URL에서만 접속가능한 Controller를 만들어서 패키지를 모든 배포하고 cronjob을 한대의 서버에서만 셋팅해서 내부 URL을 원하는 타임에 호출하게 해주면 어떨까 생각해 봤습니다.


interface 

PvpRankingRewardWorker.java 



Implementation
PvpRankingRewardWokerImpl.java


Controller코드는 간단하므로 생략했고, 

cronjob에는 wget같은 유틸을 이용해서 wget URL 같이 등록을 해주면 될것 같습니다.


이런방식이 Spring batch나 Quartz를 사용하는 방식보다 간단해 보이는데 어떻게 생각하시나요?






Posted by Breeze.Kang

댓글을 달아 주세요

맨날 헷갈리는 부분이다.


한눈에 정리되는 HashMap TreeMap LinkedHashMap의 차이점 


출처는 아래와 같다. 



출처 : 

http://stackoverflow.com/questions/2889777/difference-between-hashmap-linkedhashmap-and-treemap




결론 :  Sorting이 필요 없으면 괜히 TreeMap 같은거 쓸필요 없다. 


Posted by Breeze.Kang

댓글을 달아 주세요

CSV 파일을 파싱해서 처리해야 하는 경우 선택가능한 라이브러리가 여러개 있는게 가장 많이 사용하시는게 3가지 정도 있는것 같습니다. 


1. commons-csv : apache commons의 csv 라이브러리  


http://commons.apache.org/proper/commons-csv/


2. openCSV  :  http://opencsv.sourceforge.net


3. superCSV :  http://supercsv.sourceforge.net/



그중에 CSV파일을 Bean 오브젝트로 간단하게 변환하는 것을 제공하는 것은  OpenCSV입니다. 


아래 같이 사용하시면 되는데요.



    public List getCsvObjectList(File file, Class className) throws IOException {

        InputStreamReader in = new InputStreamReader(new FileInputStream(file), Charset.forName("UTF-8"));

        CSVReader reader = new CSVReader(in);


        HeaderColumnNameMappingStrategy start = new HeaderColumnNameMappingStrategy();


        start.setType(className);


        CsvToBean csv = new CsvToBean();

        List list = csv.parse(start, reader);

        reader.close();

        return list;

    }



javadoc을 참조하시면 총 3가지의 MappingStrategy를 제공하고 있습니다.


ColumnPositionMappingStrategy 

HeaderColumnNameMappingStrategy 

HeaderColumnNameTranslateMappingStrategy



ColumnPositionMappingStrategy :  이건 컬럼 순서대로 읽어서 수동으로 처리를 해줘야 하는 방식.


HeaderColumnNameMappingStrategy  : CSV 한줄을 bean으로 변환하는 방식 


HeaderColumnNameTranslateMappingStrategy  : CSV의 특정 컬럼만을 뽑아내서 bean으로 매핑하는 방식 



이렇게 지원을 합니다.


직접해보시면 아주 간단합니다. 





Posted by Breeze.Kang

댓글을 달아 주세요


ProtoBuffer와 spring을 이용하여 Rest 서비스 구현하기

Protobuffer를 차기 프로젝트에서 사용 예정인데, 이를 이용하여 바로 RESTful한 API를 구현할수 있는 방법이다.   

딱 내가 찾던 방식인데 실제 서비스에서 사용할지는 미지수. 


http://www.slideshare.net/mokeefe/javaone-2009-ts5276-restful-protocol-buffers





Posted by Breeze.Kang

댓글을 달아 주세요


DEView2012에 발표한 Netty Internals and Core Concepts 


Posted by +쇠돌이+

댓글을 달아 주세요

요새 Groovy on Grails를 공부하고 있는데요, 재미있군요.

java base 에서Rails 스타일로 프로그래밍을 가능하게 해줘서 생산성을 높여주는데 촛점이 맞춰진 언어입니다.

기본적인 내용들은 자료가 많이 있으니 찾아 보시면 될 것 같구요.


오늘은 Maven을 Integration하는 방법을 정리 해보려고 합니다.


기본적인 내용은 일단 아래 링크를 참조하시구요.

http://grails.org/doc/latest/guide/4.%20The%20Command%20Line.html#4.5%20Ant%20and%20Maven

간략하게 핵심만 정리해 보도록 하겠습니다.



기본적인 컨셉은 다음과 같습니다.

1. grails maven plugin을 이용하여 프로젝트 생성
2. 프로젝트 초기화
3. maven을 이용하여 Grails controller등 생성
4. maven을 이용하여 실행


프로젝트 생성

아래 명령에서 적절한 group id와 artifact id를 지정해줍니다.

mvn archetype:generate -DarchetypeGroupId=org.grails -DarchetypeArtifactId=grails-maven-archetype -DarchetypeVersion=1.3.4 -DgroupId=com.my-group -DartifactId=my-project

명령을 수행하면 버전 정보를 묻는 창이 뜨는데,

0.0.1-SNAPSHOT 같은 적당한 버전 이름을 지정해주면 됩니다.

성공적으로 수행이 되면 pom.xml 파일이 생성되고 기본적인 프로젝트 트리가 생성됩니다.



프로젝트 초기화

생성된 프로젝트 폴더로 가서 initialize를 시킵니다. 실행하면 grails 폴더 트리들이 자동으로 생성됩니다.
cd my-project
mvn initialize



Maven을 이용하여 Grails 명령어 실행

이제부터 Grails명령어들을 이용하기 위하여서는 Maven을 사용해야 합니다.

Bbs 모델 class가 이미 존재한다고 가정하고,

컨트롤러 및 기타 생성
mvn grails:exec -Dcommand=generate-controller -Dargs=com.my-project.common.model.entity.Bbs
mvn grails:exec -Dcommand=generate-all -Dargs=com.my-project.common.model.entity.Bbs

어플리케이션 실행
mvn grails:run-app


이런식으로 실행을 하면 됩니다.


이제부터는 pom.xml 파일에 Maven repository에 있는 레거시 Jar 패키지들을 Dependency 추가해주면 그대로 사용이 가능합니다.
물론 grails 파일들 외에 기존 사용하던 Java class들도 Dependency를 추가해주면 그대로 사용이 가능합니다.










Posted by Breeze.Kang

댓글을 달아 주세요

Cassandra GUI Client 프로그램을 찾던중에 괜찮은 녀석을 발견해서 올려봅니다.

Java 기반이고 war를 tomcat의 webapp 디렉토리에 넣어주기만 하면 됩니다.

처음실행시

Cassrandra Host , Thrift Port (보통 9160) , JMX port 를 입력해주시면 접속이 됩니다.

왼쪽의 Configuration 메뉴에서도 변경 가능하구요.



다운로드 페이지
http://github.com/suguru/cassandra-webconsole



소개하는 블로그
http://nosql.mypopescu.com/post/611576467/cassandra-web-console



스크린샷











Posted by Breeze.Kang

댓글을 달아 주세요

${RequestParameters['test']}

또는

${RequestParameters.test}

로 사용하면 HttpRequest의 모든 Parameter를 사용할 수 있다.



또한 httpRequest의 Attribute를 사용하고 싶다면 다음과 같이 하면 된다.

${Request['AttributeName']}

또는

${Request.AttributeName}

Posted by Breeze.Kang

댓글을 달아 주세요


Effective Java에 나오는 내용이긴 합니다만, 우리가 자주 저지르는 실수가 있습니다.

프로그램을 하다가 보면 자주 난수(Random number)를 발생시켜야 할 때가 있습니다.

아래와 같은 코드를 사용하여 많이 발생을 시키죠.


private static final Random rnd = new Random();

static int random(int n) {
   return Math.abs(rnd.nextInt()) % n;
}


논리적으로는 아무런 문제가 없어 보입니다.

그렇지만 실제적으로 다음과 같은 코드를 사용하여 테스트를 해보도록 합시다.

public static void main(String[] args) {
   int n = 2 * (Integer.MAX_VALUE / 3);
   int low = 0;
   for (int i = 0; i < 1000000; i++)
       if (random(n) < n/2)
           low++;
   System.out.println(low);
}

위의 코드데로라면  100만번 테스트를 해서 기준값보다 작은 수가 나오는 경우는 50% 가깝게 나와야 정상입니다.

그렇지만 실제로 돌려보면 항상 666,666에 가까운 수가 나옵니다.

난수가 제대로 발생하고 있지 않다는 이야기죠.

실제적으로 이런식의 버그는 찾아내기가 쉽지 않습니다.

로직상으로는 아무런 문제가 없기 때문이죠.


따라서 저런 프로그램으로 Load Balancing하거나 데이터를 분산하고 있다면 제대로 처리가 되고 있지 않을 것입니다.

java 1.2부터 추가된 Random.nextInt(int) 를 사용하거나,  apache commons의 RandomUtils.nextInt()를 사용해야 제대로 동작을 합니다.



Posted by Breeze.Kang

댓글을 달아 주세요