웹 개발2018. 9. 4. 13:13

Tomcat 4.1, JDK 4인 환경에서 기능 추가하려니 맞는 라이브러리 찾기가 무척 힘듭니다.


jakarta-taglibs-standard-1.0.6.zip


JSTL을 쓰려고 해도 해당 환경에서 돌리려고 해도 계속 에러가 나와서 JSTL 1.0까지 떨어트리니 작동했습니다.


또 Servlet 버전에 따라 jsp에 적는 taglib uri를 다르게 해야 합니다. ( web.xml의 web-app version 참조 )



버전 2.3 : <%@ taglib uri="http://java.sun.com/jstl/core" prefix="c" %>

버전 2.4 : <%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %>


Tomcat 4.1는 Servlet version 2.3이기에 EL 언어를 사용하기 위해 아래와 같이 isELIgnored="false"를 추가하면 될 것 같은데,


<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8" isELIgnored="false" %>

실제로 돌려보면 해당 속성을 지원하지 않는다는 에러가 발생합니다.


org.apache.jasper.JasperException:  Page directive has invalid attribute: isELIgnored


좀 더 조사해 보니 isELIgnored 속성은 JSP 2.0부터 지원하고, Tomcat 4.1은 JSP 1.2까지만 지원해서 EL 언어를 바로 사용하지 못하게 되어있습니다.


EL 언어를 직접 쓰지 못하기 때문에 EL 언어를 사용하려면 JSTL 안에서 사용해야 정상작동합니다.


${attribute} : (X)

<c:out val="${attribute}" /> : (O)


참고로 Tomcat 버전에 따른 최소 Java 버전이나 JSP, Servlet 버전 등이 정리된 표가 있어서 덧붙입니다.


Servlet SpecJSP SpecEL SpecWebSocket SpecJASPIC SpecApache Tomcat VersionLatest Released VersionSupported Java Versions
4.02.33.01.11.19.0.x9.0.118 and later
3.12.33.01.11.18.5.x8.5.337 and later
3.12.33.01.1N/A8.0.x (superseded)8.0.53 (superseded)7 and later
3.02.22.21.1N/A7.0.x7.0.906 and later
(7 and later for WebSocket)
2.52.12.1N/AN/A6.0.x (archived)6.0.53 (archived)5 and later
2.42.0N/AN/AN/A5.5.x (archived)5.5.36 (archived)1.4 and later
2.31.2N/AN/AN/A4.1.x (archived)4.1.40 (archived)1.3 and later
2.21.1N/AN/AN/A3.3.x (archived)3.3.2 (archived)1.1 and later


 ( 해당 표 출처 : http://tomcat.apache.org/whichversion.html )

'웹 개발' 카테고리의 다른 글

CVS commit시 sticky tag 에러, Eclipse에서 해결하기.  (0) 2018.09.03
context path 구하기  (0) 2018.02.22
Malformed fileattr.xml  (0) 2013.02.04
SCRIPT5009: 'JSON' is undefined  (0) 2013.02.01
prototype과 jquery 함께 사용하기.  (0) 2012.02.06
Posted by net4all
웹 개발2018. 9. 3. 16:58

이클립스 사용중 CVS의 commit을 하면 아래와 같은 에러를 만나는 경우가 가끔 있습니다.



cvs commit: sticky tag `1.1' for file `filename' is not a branch


이에 대한 해결책으로 override and update 해서 버전을 맞춘 뒤 백업해 놓은 변경분을 적용해 다시 커밋을 시도하거나 Enries 파일의 Tag 부분을 지운 뒤 작업을 진행하는 등 여러가지가 있지만 번거로운 절차를 거쳐야 했습니다.


가장 매뉴얼에 따른 절차로는 아래와 같이 -A 옵션으로 업데이트한 뒤 작업하는 것이지만 이것도 불편한 건 매한가지입니다.


cvs update -A filename


무엇보다 이클립스를 이용해서 편하게 사용하려고 하는데 위와 같은 명령을 직접내리는 것은 마음에 들지 않는 점입니다.


그래서 에러 메시지를 유심히 보던 중 '... not a branch'라는 메시지가 눈에 띄였습니다.

위 메시지대로면 브랜치만 맞춰주면 정상적으로 작동하겠다는 생각이 들어서 해당 명령어를 찾아봤습니다.


에러난 파일명에서 오른쪽 마우스 클릭 후 'Team' 명령어로 들어가면 나오는 

Switch to Another Branch or Version.. 명령어를 선택 후 


Select the tag from the following list에서 Matching tags:의 HEAD를 선택하면 해당 sticky tag가 사라집니다.

( update -A와 동일한 효과가 적용되는 거죠^^* )


에러난 파일들을 위 명령어로 tag를 다 바뀌어주고 난 뒤 oommit 명령어를 선택하면 정상적으로 작업을 진행할 수 있습니다. ( 이 방법은 백업 뒤 업데이트 작업 후 변경분 추가하는 번거로운 작업없이 현상태에서 tag만 바꾸고 바로 작업이 진행되기에 더욱 편합니다^^ )




'웹 개발' 카테고리의 다른 글

Tomcat 4.1, Java 1.4에서 JSTL 사용하기.  (0) 2018.09.04
context path 구하기  (0) 2018.02.22
Malformed fileattr.xml  (0) 2013.02.04
SCRIPT5009: 'JSON' is undefined  (0) 2013.02.01
prototype과 jquery 함께 사용하기.  (0) 2012.02.06
Posted by net4all
웹 개발2018. 2. 22. 13:58


context path가 소스내에 하드코딩 되어 있으면, context path 변경시 모든 파일에서 다 찾아서 변경해야 합니다. 

요즘은 에디터가 잘 되어있어서 쉽게 찾을 수 있지만 잘못된 문구가 치환되지 않기 위해 주의를 기울여야 하는 것도 번거럽기도 하고, 하드코딩은 가능한 피햐해 하죠.


소스 내에 자바스크립트를 사용할 수 있는 곳은 아래 함수를 이용하면 됩니다.


javascript:

function getContextPath() {

var hostIndex = location.href.indexOf( location.host ) + location.host.length;

return location.href.substring( hostIndex, location.href.indexOf('/', hostIndex + 1) );

}




이미지 경로 설정과 같은 경로를 직접 써야 할 경우 아래 표현식을 써주면 좋습니다.


jsp : EL(Expression Language) 사용


servlet 2.4 이상 ${pageContext.request.contextPath}

servlet 2.3 이하 <c:out value='${pageContext.request.contextPath}' />

위 방식을 사용하기 위해선 JSTL 1.2 라이브러리가 필요합니다.

예) <img src="${pageContext.request.contextPath}/images/logo.png" />

<img src="<c:out value='${pageContext.request.contextPath}' />/images/logo.png" />




그런데 자바스크립트를 사용 못하는 css 파일 내 경로의 경우 톰캣에서 컨텍스트 주소를 별도로 잡아서 적어주는 방법이 있습니다.


톰캣의 server.xml

<Context docBase="CTX" path="/ctx" reloadable="true" source="org.eclipse.jst.j2ee.server:CTX"/>

    <Context docBase="CTX/images" path="/ctxImg" reloadable="true" source="org.eclipse.jst.j2ee.server:CTX"/>

위에서와 같이 이미지 경로를 포함하는 주소를 docBase에 설정하고 path를 잡아주면 됩니다. 일반적인 경우는 인터넷에 많이 나와 있어서 이클립스에서 잡은 예를 들었습니다.


예) 

AS-IS : .pattern { background:url(/ctx/images/common/pattern.gif) repeat; }

TO-BE : .pattern { background:url(/ctxImg/common/pattern.gif) repeat; }

사실 톰캣에서 주소를 새로 설정해주었으면 모든 이미지 주소에 새로 설정한 값을 사용하는 것이 편할 수 있습니다. context path와 image path는 별개로 움직이니까요.




참고로 자바에서 컨텍스트 패스를 잡는 경우는 아래와 같이 하면 됩니다.


java : HttpServletRequest request

request.getContextPath()

Posted by net4all
웹 개발2013. 2. 4. 14:52

Malformed fileattr.xml file  ... Please fix or delete this file


CVS를 사용하다 보면 위와 같은 에러가 나서 당황스러울 때가 있습니다.


알려져 있기는 CVSNT 2.0.58d와 몇몇 이전버전의 버그로 알려져 있어서 보통 다른 버전을 설치해서 해결하곤 합니다.


그렇다고 무턱대고 새로 설치하긴 그래서 찾아봤더니 파일명에 &(ampersand)가 들어가면 fileattr.xml이 깨진다고 하네요.

그리고 파일명에 한글이 들어간 경우에도 에러를 발생하고요.


폴더명과 파일명을 영숫자로 작성한다면 큰 문제없이 피해갈 수 있을 듯 합니다.

Posted by net4all
웹 개발2013. 2. 1. 15:19

SCRIPT5009: 'JSON' is undefined

 

크롬으로 개발하고 나서 IE에서 돌아가나 테스트하다보니 별의별 SCRIPT 에러가 발생하네요.

 

그중에 하나가 JSON이 정의되지 않았다는 것인데, IE 10에서는 잘 작동하나 이전 버전의 IE에서는 JSON 모듈을 포함되어 있지 않기에 별도록 해당 모듈을 추가해야 합니다.

 

json2.js를 추가하면 되는데, 최신 모듈은 아래 주소에서 볼 수 있습니다.

 

https://github.com/douglascrockford/JSON-js

 

그리고 json2.js를 프로젝트에 포함시킨 뒤 아래처럼 한 줄을 적어주면 에러가 발생하지 않습니다.

 

<script type="text/javascript" src="json2.js" ></script>

 

그럼, 행복한 프로그래밍하세요~

Posted by net4all
웹 개발2012. 2. 6. 21:07

기존에 사용된 prototype과 jquery의 $ 기호와 충돌을 일으키는 문제가 있어서 찾아봤습니다.

한 사이트에 소개한 jquery의 $표시를 무효화시키고 jQuery를 연산자로 사용하는 것도 괜찮습니다만,

jQuery.noConflict();

그보다는 다른 스크립트 라이브러리(여기선 prototype)보다 먼저 jQuery를 삽입하면 $가 프로토타입의 $ 기호로 작동하게 된다고 하네요.

jQuery 글자가 길기에 이를 다른 객체에 담아서 짧게 사용하는 것도 좋은 아이디어입니다.

var $j = jQuery;
$j(document).ready(function(){
    $j('.location').hide();
    $j('#title').click(function(){
    alert('jQuery is running!');
  });
 });
 


 
Posted by net4all
웹 개발2011. 7. 21. 09:43

프로그래밍 초보자가 알아야 하는 9 + 1 가지 Homo Coding 2011/07/20 11:11

크리에이티브 커먼즈 라이선스
Creative Commons License
지금와서 보면 초보 시절에 별것 아닌 것으로 많은 시간과 많은 밤들을 보내면서 나름 프로그래밍의 세계에서 매력아닌 매력에 빠지곤 했었습니다. 지금의 초보자들을 보면서 지난 프로젝트들을 되돌아보고 저 역시 저질렀거나 하지 말았으면 하는 것들에 대해서 한번 생각을 해보았습니다.


so I went for a walk to a place that I know.
so I went for a walk to a place that I know. by Sanctu 저작자 표시비영리동일조건 변경허락

이는 비단 프로그래밍에 갓 입문한 사람 뿐만 아니라, 다른 프로그래밍 언어를 처음 접할 때에도 적용되는 내용 같습니다.

1. 프로그래밍의 최대의 적은 Copy & Paste이다.

IT에서 일어나는 작업들은 모두 사람의 힘을 배가시키기 위함이고, 이를 위해서는 최대한 자동화를 시켜주는 것이 그 목적일 것입니다. 따라서, 프로그래밍이라는 작업 역시 일일히 모든 것들을 만들기보다는 최대한 재사용할 수 있는 것들을 활용하는게 작업 능률에 있어서 최적일 것입니다. 그 중에서 프로그래밍 작업 중에 제일 흔하게 일어나는 일이 Copy & Paste입니다. 하지만, 익숙한 프로그래밍 언어라도 Copy & Paste는 주의를 필요로 합니다. 더군다니 초보 시절의 Copy & Paste는 독이 될 수도 있으며, 결국 잘 이해되지 않은 코드의 Copy는 어떠한 문제를 일으킬지 잘 모릅니다.

Palacio de las Artes Reina Sofía C.A.C / Palace of Arts Reina Sofia  C.A.C./ Palais des Arts Reina Sofia C.A.C.
Palacio de las Artes Reina Sofía C.A.C / Palace of Arts Reina Sofia C.A.C./ Palais des Arts Reina Sofia C.A.C. by Iabcstm 저작자 표시비영리변경 금지


초보 시절은 동일한 코드라고 하더라도 직접 키보드를 통해서 입력하거나 도구를 정상적인 경로를 통해 사용해봄으로써 머리와 몸으로 코드의 원리를 체득해야 합니다. 심지어 수천라인의 XML 코드라고 하더라도 일일히 쳐보면서 그 의미를 하나하나 쫓아서 읽을 수 있어야 합니다. 무분별한 Copy & Paste는 발생하는 에러를 엉뚱한 곳에서 디버깅하게 만듭니다. 요즘에는 책을 사면 소스 코드를 다운받게 되어 있습니다. 해당 내용을 잘 알고 어느 정도 이해를 하고 있는 사람이라면 상관없겠지만, 초보자들에게는 이 역시 소스 코드를 다운받아서 실행하는 것보다는 직접 책에 있는 내용을 쳐보고 이해하는 것이 많은 도움이 됩니다. 혹 책의 내용이 이전 라이브러리들을 참고하고 있어서 지금의 환경에서 이를 적용하려면 더 많은 수고들을 거치는 경우도 발생됩니다. 이러한 과정을 거치게 되면 또 다른 지식들이 쌓이게 되니, 헛짓이나 삽질이라고 생각하시면 안됩니다.

초보 시절 저 역시 수많은 코드들을 직접 손으로 치면서 많은 것들을 이해할 수 있었고, 더 많은 연결된 부분들을 찾아볼 수 있었던 것 같습니다. 반복적인 작업이지만, 그 자체로 더 없는 많은 내용들을 얻게 될 것입니다.

2. 가이드 대로 따라하면 안된다.

프로젝트에서는 수많은 가이드들이 만들어지며, Step By Step 형식으로 나와있는 문서나 자료들도 많이 있습니다. 초보 시절에는 가이드를 보는 것 역시 중요하겠지만, 그 내용을 곧이 곧대로 믿어서도 안됩니다. 특히나 프로젝트 중후반부로 들어갈수록 초반에 만든 가이드는 잘못되거나 수정된 내용이 많아질 가능성이 많습니다. 가이드는 대략적인 내용들을 보면서 어떠한 흐름으로 프로그래밍을 하게 되는지에 대한 지식을 얻으면 됩니다. 또한, 프로그래밍은 가이드에 나타난 내용 이외에도 더 많은 지식과 기술을 필요로 합니다. 가이드는 말 그대로 안내하는 용으로만 사용해야지 가이드가 모든 것을 가르쳐준다고 생각해서는 절대 안됩니다.


Day 106 - I am a librarian
Day 106 - I am a librarian by cindiann 저작자 표시비영리변경 금지


가이드를 절대적으로 믿는 행위 중에 하나는 가이드에 표현되지 않은 것들에 대해서는 아무런 사고를 안하게 만든다는 것입니다. 예를 들어, 예상치 못한 에러나 경고 문구가 나타나도 이를 신경쓰거나 무엇이 문제인지를 찾으려는 노력을 하지 않게 만듭니다. 이는 가이드를 만들고 배포하는 측에서도 가이드만 보고 따라하라는 암묵적인 강압을 하는 태도도 일조한 결과입니다. 가이드를 맹신하지 마십시오.

3. 한가지 정답을 찾으려고 하지 마라.

우리가 학교에서 배운 것들은 모두 정답이라는 한가지 목적에만 초점을 맞추어 있습니다. 사회의 초년생들 역시도 정답이 아니면 안된다는 사고를 가진 사람들이 대부분인 것 같습니다. 정답은 늘 해당 상황에서만 정답이 될 뿐이며, 다른 상황에서는 오답이 될 가능성이 높습니다. 프로그래밍에서 어느 한가지로 해결했다고 해서 이것이 다른 프로젝트에서 똑같이 적용될 수는 없을 겁니다. 정답보다는 그 가능성에 대해서 염두해 두십시오. 가능성은 확률로도 표현이 가능하며, 가능성이 높다는 것은 다양한 전제 조건을 포함한다는 것입니다. 즉, Context라는 단어가 의미하는 바를 이해하고, 그 상황에서 다양한 전제 조건과 제약 조건들을 같이 고려하여 가능성이 제일 높은 해결책을 찾는 것이 프로그래밍을 배울 때 같이 익혀야 할 내용입니다.

현실에서는 프로그래밍으로 할 수 있는 것들이 의외로 많고, 그러한 내용들이 나타나는 프로그램은 단지 그 결과에 지나지 않습니다. 다른 사람의 정답을 참고할 수는 있겠지만, 그러한 정답이 지금의 프로젝트 내에서 반드시 정답이라는 보장을 하기 힘듭니다. 그 사람의 정답(결과)이 만들어지는 배경과 상황을 같이 이해해야 합니다.

4. 고민을 끌어안고 있지 말고, 터뜨려라.

프로그래밍의 초보 시절에는 단순한 문제라도 심각한 고민거리가 되기 쉽상입니다. 그러한 고민거리는 그 누구한테 물어보는 것조차 힘겨운 법입니다. 고민을 하는 것은 그리 나쁜 것은 아니지만, 주어진 시간과 다른 작업으로의 연결이라는 제약사항들이 있기 때문에 무한정 고민만 할 수는 없습니다. 고민을 하는 시간조차도 효율적으로 사용해야 하며, 정 고민이 해결되지 않는다면 터뜨리십시오. 누가 물어보기 전까지 고민을 안고 있다라는 것은 다른 사람으로 하여금 실력이 없다라는 충분한 오해를 받을 만한 일입니다. 본인의 실력이 드러날까봐 고민을 안고 불안한 마음에 있는 것보다는 고민을 터뜨려서 해결하는 방법을 채택하는 것이 오히려 전체적인 관점에서 더 좋은 결과가 나타날 수 있습니다.

고민을 터뜨리는 방식은 단순하게 이 부분을 모르겠다고 솔직하게 털어놓는 것도 있을 수 있지만, 좀 더 유연한 방식으로는 그동안 고민했던 내용들을 다시 정리해보고, 왜 문제가 있는지를 간략하게 설명할 필요가 있습니다. 이때 주의할 점은 본인의 입장에서만 고민을 정리하다보면 상대방이 고민에 대한 본질적인 문제보다 그 부수적인 내용이 더 이해하기 힘든 경우가 발생됩니다. 고민은 최대한 간략하고 이해할 수 있도록 정리하는 기술이 필요합니다.

5. 통용되는 정확한 용어를 사용하라.

초보 시절에는 모든 단어와 용어들이 낯설기만 합니다. 그러한 용어들은 선배들이 잘못 이해하여 사용되는 경우들도 있습니다. 용어를 들은대로만 사용하기보다는 그 뜻과 의미를 정확하게 알아보는 습관을 들이는게 좋습니다. 또한, 그러한 용어를 알아가는 과정이 전번적인 내용을 이해하는 과정이기도 하며 이는 프로그래밍에 대한 지식을 더 폭넓게 이해하게 만듭니다.

대표적인 예가 디자인 패턴의 패턴 이름인데, 이러한 패턴 이름들은 상당한 오랜 시간 동안 많은 사람들에게 사용되어 왔습니다. 단순하게 이름만을 듣고, 그에 대한 배경 지식이 없는 상태에서 프로그램된 상태만을 본다면 나중에 또 다시 잘못 사용하여 이를 듣는 사람에게 오해를 불러일으킬 수도 있을겁니다.

변수명, 클래스명, 메소드명 등 역시 정확한 (비즈니스) 용어를 사용해야 합니다. 귀찮다고 줄이거나 임의로 이해할 수 없는 용어를 만들면 다른 사람이 보기가 힘들어집니다. 예전 기술들이야 워낙에 HW 용량에 대한 문제가 커서 이름이나 용어에 대한 길이 한정 제약이 있었지만, 지금은 이러한 제약에서 훨씬 자유롭습니다. 의미있고 정확한 이름을 사용하기 바랍니다.

6. 프로그래밍의 내용은 한편의 시나리오(소설)를 보는 듯하게 만들어라.

초보 시절의 프로그래밍 습관은 상당히 중요합니다. 주석, 변수명, 메소드 호출 등에 있어서 다양한 습관들이 서로 다르게 표현될 수 있습니다. 특히, 프로그래밍의 내용은 비즈니스의 흐름과도 동일하며, 그 흐름은 마치 한편의 드라마를 보듯이 매끄럽게 이어져야 합니다. 변수 선언, 객체 생성, 리소스 획득, 메소드 호출 등이 뒤죽박죽 섞여 있어서 코드를 본다는 것 자체가 무슨 내용인지를 모르게 구성하는 경우들이 있습니다. 프로그래밍의 내용은 코드만을 보고도 어떠한 흐름으로 이어지는지를 알 수 있게 구성되어야 하며, 이는 마치 시나리오와 같습니다.

어떤 경우에는 초반에 만든 코드와 나중에 수정한 코드가 뒤범벅이 되어서 어떤 것이 실제로 사용하는지도 잘 모르게 구성되는 경우도 많습니다. 정 이전 코드를 참고하고 싶으면, 차라리 주석 처리하거나 별도의 메소드를 만들어서 구성하는 것이 더 깔끔할 겁니다. 그리고, 사용되는 것과 사용되지 않는 것들에 대한 구분 역시 필요하며, 이는 본인만 알아볼 수 있는 것이 아닌 공통적으로 누구나 알아볼 수 있는 표시를 해야 할겁니다.

코드는 실행되는 것도 중요하지만, 무엇보다 깔끔한 형태의 코드 결과도 상당히 중요합니다. 그 안에는 누구든지 해당 코드를 보고 쉽게 이해할 수 있고, 이름만 보더라도 무엇을 하는 객체인지를 알 수 있게 구성하는 것이 훨씬 중요합니다.

7. 처음부터 구조화시킬 필요는 없다.

처음에 프로그래밍 언어에 익숙해지기 시작하면서 코드에 대한 구조를 생각하게 되고, 초반부터 이해하지 못하는 구조를 만들기 시작하는 경우도 있습니다. 구조화는 코드가 동작하는 상태에서 의미가 있으며, 구조화되지 못한 코드라고 하더라도 동작한다면 크게 문제되지 않을 수 있습니다. 초반에 구조화를 시도하다가 많은 시간을 허비하는 경우도 있으며, 다양한 구조화를 시도하다가 실제 작업은 진행하지 못하는 경우도 많습니다. 또한, 초반에 너무나 많은 내용을 한꺼번에 프로그래밍을 하다보니, 테스트할 시간조차도 부족한 경우도 있습니다.

처음 객체를 만들때에는 가장 단순하고 적은 데이터만을 사용해서 전체적인 뼈대를 구성하는 것이 가장 좋습니다. 그리고, 이러한 뼈대에 차츰 살을 붙여서 완성해가는 것이 좋은 방법입니다. 뼈대가 구성되지 않은 상태에서 살을 붙일 수는 없을겁니다. 어느 정도의 구조화는 필요하겠지만, 여기에 모든 노력을 기울여 자신의 실력을 뽐낼 필요까지는 없습니다. 보여주고자 하는 데이터가 한두개 모자란다고 크게 문제되지는 않습니다. 이러한 문제들은 테스트를 통해서 수정하기만 하면 됩니다. (물론, 테스트를 할 수 있는 시간이 확보된다면 문제되지 않습니다.) 하지만, 동작하지도 않는 프로그래밍은 테스트를 하기도 어려우며, 그 결과를 예측하기도 어렵습니다.

8. 책을 쓰듯이 코드를 작성하라.

프로그래머에게는 코드 자체가 문서이자, 작업 대상이어야 합니다. 코드를 작성하는데 매번 만족할 만한 수준까지 작업하기는 힘들지만, 코드를 책을 쓰듯이 작성해야 합니다. 책을 쓴다는 것은 자신이 읽기 위함이 아닌, 다른 사람이 본다는 생각을 할 필요가 있습니다. 다른 사람이 자신이 만든 코드를 보고 이해하기 쉽게 짜임새 있게 구성되어야 하며, 각 부분들의 조각들이 맞추어져서 전체가 하나의 새로운 구성체로 조합되어야 합니다. 책을 쓴다는 것은 작성자의 signature를 포함한 내용들이 그 안에 있어야 하며, 코드에서 만큼은 본인이 구성한 의도가 충분히 포함되어야 합니다.

잘 짜여진 코드는 그 내용이 기승전결과 같은 일관된 구성으로 이루어지며, 심지어 표지나 서문과 같은 내용까지도 생각하여 다양한 데코레이션들을 코드 내에 삽입할 수 있습니다. 무엇보다 책 내용이 중요하듯이 그리고, 그 책의 구성 또한 소홀히 할 수 없듯이 코드는 그 내용을 중심으로 한권의 책을 보는 듯한 인상을 주어야 합니다.

9. 프로그램 내용을 시각적으로 표현해보라.

아무리 단순한 프로그램이라고 하더라도 최소한 몇개의 레이어나 시스템을 거쳐서 비즈니스 로직을 수행하기 때문에 전체적인 윤곽을 그려보는 것이 필요합니다. 또한 프로그램을 담고 있는 각 객체간의 관계가 어떻게 이루어지고, 무슨 작업을 수행할 때 필요한지는 파악해야 합니다. 이러한 모습들은 코드의 형태보다는 간단한 시각적인 표현 방식을 사용해서 파악하는 것이 필요합니다. 때로는 프로그램적으로 해결하기 어려운 부분들이 전체적인 구조를 파악하면 쉽게 해결되는 부분들도 있습니다.

즉, 자신이 만든 프로그램도 중요하지만, 남이 만든 프로그램 역시 전체적으로 어떠한 관계를 맺는지를 파악하는게 중요합니다. 자신이 만든 프로그램이 어떠한 용도로 사용되며, 어떠한 문제들을 일으키게 되는지도 사전에 파악하여 조치를 할 수 있어야 하며, 남이 만든 프로그램을 사용할 때 어떠한 부가 효과(side effect)가 발생할지에 대해서 알아야 합니다.

프로젝트를 혼자 모든 것을 처리하지 않는 이상, 다른 이가 만든 프로그램에 대한 영향, 내가 만든 프로그램에 대한 영향을 파악하는 시각을 키울 필요가 있습니다. 프로그램 자체가 단독으로 실행하는 것보다 서로 다른 사람들이 만들어서 같이 동작하게 만다는 작업은 상당한 노력을 필요로 하며, 이는 그리 단순하고 손쉬운 작업이 결코 아니라는 것을 인지할 필요가 있습니다.

이를 위해서는 자신이 만든 프로그램이라도 시각적으로 표현하고 이를 이해할 수 있어야 합니다.

9+1. 항상 원칙을 지키도록 노력하라.

프로젝트 현장에서는 다양한 기술과 노하우들이 펼쳐지고 난무하는 곳입니다. 이곳에서의 기술과 노하우들은 그 근본을 알 수 있는 것들도 있지만, 그렇지 못한 것들도 많이 있습니다. 원칙, 원리, 이론 등은 현장에서의 이러한 적용의 원리에 의해서 마치 구시대적이거나 학구적인 것으로 대접을 받아 상당한 무시와 천대를 받는 경우도 많습니다. 무엇이 되었든 간에 원칙은 반드시 존재해야 하며, 그러한 원칙을 기반으로 다양한 적용 원리가 나타날 수 있다는 것을 잊지 마시기 바랍니다.

원칙이라는 것이 거창하거나 복잡한 내용이 아니지만, 이를 이해하는 과정은 그리 쉬운 작업이 아닙니다. 하지만 원칙이 없는 프로그램 코드는 마치 모래 위에 성을 쌓는 것과 같습니다. 늘 환경에 의해서 흔들리고 다시 지어야하는 경우가 발생됩니다. 사소한 원칙이라도 무시하지 마시고, 그 원칙이 왜 생겨났는지를 곰곰히 생각하여 반드시 프로그래밍으로 표현할 때 같이 이해할 수 있어야 합니다.


위와 같은 내용은 어찌보면 엉뚱한 이야기인 것 같기도 하지만, 대부분이 현장에서 초보 프로그래머들과 접하면서 꼭 해주고 싶었던 이야기들입니다. 프로젝트 현장은 이와 같은 내용들을 세세하게 설명해주고 가르쳐줄 시간이 턱없이 부족하며, 똑같은 실수들을 후배들이 저지르는 것을 보는 것이 안타까울 따름입니다. 초보 시절의 습관은 경력이 쌓이더라도 쉽게 사라지지 않고 그 습관은 반드시 코드 상에 나타나게 되어 있습니다. 처음부터 힘들더라도 차근차근 실력을 쌓아가도록 노력하려는 자세가 필요합니다.
 
저작자 표시 비영리 동일 조건 변경 허락

Posted by net4all
웹 개발2009. 3. 5. 16:40

iBatis를 사용해서 구문을 만들 때 흔히 걸리는 것이 where title like "#title#%" 와 같은 구문을 적으면 에러를 발생한다는 것일 겁니다.

메뉴얼을 참고로 해서 where title like "$title$%"를 이용하면 title property가 저 자리에 그대로 치환되면서 사용할 수 있기는 합니다. 그런데 말 그대로 치환되는 것이기에 SQL injection 공격으로 인한 위험도 있습니다.

그래서 다른 방법으로 쓸 수 있는 것이 concat으로 프로퍼티와 %를 합쳐서 집어넣는 것입니다.

where title like concat ( #title# , '%' )

라고 적어주면 처음에 나온 의도했던 구문처럼 작동합니다.


참고로 실제로 SQL 구문이 어떻게 들어가고 있는지 보려면 log4j에서 java.sql을 DEBUG로 찍어보면 됩니다. Dynamic SQL을 사용할 때 제대로 작동하는지 확인할 때 변합니다.

log4j.properties 파일 안에

log4j.logger.com.ibatis=DEBUG
log4j.logger.java.sql.Connection=DEBUG
log4j.logger.java.sql.Statement=DEBUG
log4j.logger.java.sql.PreparedStatement=DEBUG
log4j.logger.java.sql.ResultSet=DEBUG


위와 같이 적어주면 관련된 명령 등을 대부분 볼 수 있을 겁니다.
SQL 쿼리문 관련해서는

log4j.logger.java.sql=DEBUG


이렇게 한 줄만 적어주어도 되고요.

위와 같이 찍어보면 #title#은 prepared statement에서 ?로 들어가고, $title$의 경우는 그 값이 그대로 치환되어서 들어가는 것을 확인할 수 있습니다.


Dynamic SQL 사용해보면 XML의 중첩 기능을 사용해서 다양한 구문을 만들어 낼 수 있는게 재미있습니다 :)

Posted by net4all
웹 개발2009. 3. 2. 10:18

JSP에서 알면 간단하면서도 모르면 고생하는 것이 날짜를 받아서 DB에 넣는 것이 아닐까요?

입력이 request를 통해 들어올 때 getParameter 메소드로는 문자열 밖에 못 받는데,
이걸 DB에서 인식하는 Timestamp 형태로 바꾸기 위해서

가장 무식한 형태는 년,월,일,시,분,초 형태로 짤라서 넣기 ㅡ.ㅡ;

이걸 Date로 넣을 경우 오류가 발생하기도 해서 java.sql.date를 안 쓰고 java.sql.timestamp를 쓰는데

정말 간단한 형태를 지원합니다.


Timestamp.valueOf("2009-03-01 10:10:05"); 형태로 집어넣는 겁니다.

그러기에 이를 쓰기 위한 메소드는 두 줄이면 끝납니다.

import java.sql.Timestamp;

public Timestamp getTimestamp(String str){
return Timestamp.valueOf(str);
}

위의 문자열 형태는 규칙에 따르면 yyyy-mm-dd hh:mm:ss.fffffffff 형식입니다.
위에서 보는 것과 같이 timestamp는 나노 세컨드까지 지원하는 반면,
java.sql.date는 밀리 세컨드까지만 지원합니다.
이 때문에 처리할 때 좀 달라지는 경우가 생깁니다.
Posted by net4all
웹 개발2009. 1. 30. 14:41

여러 사람이 공유하는 데이타일 경우 마지막으로 열람한 뒤 변경 여부를 어떻게 알 수 있을까?

이를 위해 HTTP/1.1 규격에선 ETag를 지원합니다.
http://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html#sec14.19

예)
ETag: "xyzzy"
ETag: W/"xyzzy"

위와 같이 임의의 글자로 만들어진 태그를 붙여서 변경할 때마다 그 값을 바꿔줍니다.

이걸 이용해서 소스관리에서 발생할 수 있는 충돌을 예방할 수 있을 것 같습니다.

Google Data API에서도 이를 이용해서 ETag가 맞지 않는 경우에는 변경을 못 시키게 하고 있는데,
이 경우 다시 Entry 정보를 가져와서 그때 얻은 ETag를 이용해서 고쳐야 합니다.

이것이 소스관리에서 먼저 업데이트를 하고 난 뒤 커밋을 해야 하는 것과 비슷하다는 느낌이 들었습니다.

(ETag가 어떻게 작동하는지 몰라서 잠시 동안 Match가 안 된다는 메시지가 떠서 좀 고생했는데,
이렇게 생각하니 고치기 전엔 먼저 Entry를 받는 것부터 시작해야 한다는 의미가 이해되었습니다^^;)
Posted by net4all