[ClassLoader] Thread의 getContextClassLoader()

pdf: http://www.theserverside.com/tt/articles/content/dm_classForname/DynLoad.pdf

먼저 간단한 퀴즈로 시작하는.. 긴 문서..


이 코드가 문제를 일으킬 수 있는 상황은 저 클래스를 자바 Extentions 폴더 (<JDK>/jre/lib/ext 또는 <JRE>/lib/ext)에 놓았을 경우입니다. 그렇지 않고 그냥 CLASSPATH에 둘었다면 별 문제가 없을 가능성이 큽니다. 대체 저런식으로 동적으로 로딩하려는 클래스들이 App CL의 CLASSLOADER에 있기 마련일 테니까요.

하지만, Extention 폴더로 이동하면 무슨일이 벌어질까요. ClassNotFoundExcepion이 발생합니다. 그 원인을 정확히 모르신다면 ClassLoader에 대한 아주 기본적인 개념이 부족한거라 봐도 무방하겠습니다.

이 글에서 그에 대한 해결책 두 개를 제공해주며, 그 중 하나가 바로 쓰레드 컨텍스트 클래스로더를 사용하는 것입니다.

http://java.sun.com/javase/6/docs/api/java/lang/Thread.html#getContextClassLoader%28%29

Returns the context ClassLoader for this Thread. The context ClassLoader is provided by the creator of the thread for use by code running in this thread when loading classes and resources. If not set, the default is the ClassLoader context of the parent Thread. The context ClassLoader of the primordial thread is typically set to the class loader used to load the application.

First, if there is a security manager, and the caller’s class loader is not null and the caller’s class loader is not the same as or an ancestor of the context class loader for the thread whose context class loader is being requested, then the security manager’s checkPermission method is called with a RuntimePermission(“getClassLoader”) permission to see if it’s ok to get the context ClassLoader.

별짓을 하지 않는다면 쓰레드 컨텍스트 클래스로더는 System(또는 Application) CL이 됩니다. 따라서 저 위의 코드를 자바 extentions 폴더에 넣고 실행하더라도 쓰레드 컨텍스트 클래스로더를 가져와서 loadClass를 호출하면 아무런 문제없이 동작합니다.

안그래도 대체 Thread Context ClassLoader가 왜 생긴건지 궁금했는데 이 글을 통해서 짐작할 수 있게 됐습니다. 클래스로더 계층 구조를 만들 때, 예를 들어 톰캣의 Common CL 같은 녀석은 자기 하위의 CL인 WebAPP CL에 어떤 클래스들이 들어올지 모를겁니다. 하지만 그런 상태에서 Common CL에서 WebAPP CL에서만 로딩할 수 있는 어떤 클래스를 로딩할 필요가 있다면… WebAPP CL을 직접 참조하지 않고 바로 Thread Context ClassLoader를 사용해서 로딩하도록 코딩을 하면 로딩할 수 있게 되는거죠.

흠.. 하지만 이.. 미적지근한 느낌은 뭘까요.. 왠지 상위 CL에서 하위 CL의 클래스패스에 있는걸 로딩해버리면 왠지 꼬여버릴 것만 같은 이 기분… 아.. 불안해. 왜 그럴까나..

[구글 Visualization API] Fluent Interface 적용하기

http://martinfowler.com/bliki/FluentInterface.html

이름은 좀 멋지지만 별거 아니라는…

        Statistics statistics = new Statistics(StatisticsChartType.MOTION, “스터디 참석자 통계 그래프”);
        statistics.addColumn(StatisticsColumnType.STRING, “스터디 이름”);
        statistics.addColumn(StatisticsColumnType.DATE, “모임 날짜”);
        statistics.addColumn(StatisticsColumnType.NUMBER, “참석자 수”);
        statistics.addColumn(StatisticsColumnType.NUMBER, “모임 횟수”);
        statistics.addColumn(StatisticsColumnType.NUMBER, “참석율”);
        statistics.addColumn(StatisticsColumnType.STRING, “스터디 상태”);
        statistics.addColumn(StatisticsColumnType.STRING, “모임명 이름”);

이 코드는 얼마전에 봄싹 사이트에 Statistics 메뉴를 추가하면서 코딩한 것인데 오늘 보다가 갑자기 Fluent Interface가 떠올랐습니다. 그래서 적용해봤지요.

        Statistics statistics = new Statistics(StatisticsChartType.MOTION, “스터디 참석자 통계 그래프”)
            .withColumn(StatisticsColumnType.STRING, “스터디 이름”)
            .withColumn(StatisticsColumnType.DATE, “모임 날짜”)
            .withColumn(StatisticsColumnType.NUMBER, “참석자 수”)
            .withColumn(StatisticsColumnType.NUMBER, “모임 횟수”)
            .withColumn(StatisticsColumnType.NUMBER, “참석율”)
            .withColumn(StatisticsColumnType.STRING, “스터디 상태”)
            .withColumn(StatisticsColumnType.STRING, “모임명 이름”);

조금 fluent 하게 보이나요??ㅋ 객체를 다 만든 다음에 추가하는게 아니라 객체 만들 때 메서드를 연달아 호출하는 모양이여서 메서드 이름을 add -> with로 변경했습니다. 메서드 리턴 타입도 당연히 바꿨지요.

장점은 저렇게 조금 API가 이쁘고 사용하기 편해진다는 것이고 단점은 디버깅할 때 조금 불편할 수 있다는데.. 머.. 글쎄요. 그다지…