티스토리 뷰
우리는 자바 애플리케이션을 개발하면서 습관적으로 System.out.println(" ... ");
을 활용한다. 대부분은 작성한 프로그램의 정확성이나 논리적인 오류를 찾아내기 위한 디버깅 과정에서 자주 사용하고, 일부 경우에만 정보를 전달하거나 에러 메시지를 전달하기 위해 사용한다.
하지만 이것은 프로그램의 성능 이슈를 발생시키기 때문에 실 서비스하는 시점에서는 개발 단계에서 곳곳에 추가했던 System.out.println(" ... ");
을 찾아다니면서 제거하는 번거로운 작업을 해야 한다.
여기서 이슈가 한 가지 더 발생한다. 실제 작성한 프로그램을 배포한 후, 추가로 문제가 발생한 상황에서는 다시 디버깅 메시지를 출력해서 해당 로직을 수정하는 작업 등을 해야 한다. 기존에 삭제했던 디버깅 메시지들을 추가해야 하는 상황이 발생할 수도 있다. 이러한 불편함이 있기 때문에 디버깅 메시지를 주석처리 해놓고 개발하기도 하지만, 그다지 매력적이지 않은 방법이다.
포스팅("logback을 사용해야 하는 이유")으로 대신한다.)
로그 레벨
ALL < TRACE < DEBUG < INFO < WARN < ERROR < OFF
logback에서는 위와 같은 단계로 로그 레벨을 구분한다. TRACE 레벨은 TRACE, DEBUG, INFO, WARN, ERROR 레벨을 포함하고, INFO 레벨은 INFO, WARN, ERROR 레벨을 포함한다. 가장 많이 사용하게 될 DEBUG 레벨은 개발 단계에서 디버깅을 위한 메시지를 출력하고, 각 레벨 설정에 따라서 확인할 수 있는 로그 메시지의 종류가 달라진다. 다음 단계에서 더 자세히 살펴본다.
로그 메시지 형식
설정 파일의 configuration 태그를 사용하여 로그 메시지의 형식을 설정할 수 있다. maven의 기본 resources 폴더의 경로는 src/main/resources 이다. 따라서, 해당 경로에 logback.xml 파일을 생성한다.
logback.xml
패턴이 포함하고 있는 내용은 순서대로 시간, 스레드, 로그 레벨, 로거를 포함하고 있는 클래스, 디버그 메시지 이다.
1 2 3 4 5 6 7 8 9 10 11 | <configuration> <appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender"> <encoder> <pattern>%d{HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n</pattern> </encoder> </appender> <root level="DEBUG"> <appender-ref ref="STDOUT" /> </root> </configuration> | cs |
아래의 dependency를 추가한다.
1 2 3 4 5 6 7 8 9 10 11 | <dependencies> <dependency> <groupId>ch.qos.logback</groupId> <artifactId>logback-classic</artifactId> <version>1.2.3</version> </dependency> ... </dependencies> | cs |
예제
사용자가 한글을 입력하는 경우를 대비하여 Encoding 설정을 하는 Filter 클래스에 로깅 프레임워크를 적용하는 예제이다.
LoggerFactory 생성
logger를 사용하기 위해서 LoggerFactory를 생성한다.
org.slf4j.Logger와 LoggerFactory를 import한다.
1 2 3 4 5 6 7 8 | import org.slf4j.Logger; import org.slf4j.LoggerFactory; ... @WebFilter(urlPatterns = { "/*" }) public class CharacterEncodingFilter implements Filter { static final Logger logger = LoggerFactory.getLogger(CharacterEncodingFilter.class); ... | cs |
System.out.println()를 대신하여 logger를 통해서 로그를 출력한다.
debug() 메소드는 로그 레벨을 DEBUG로 설정한다.
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 32 33 34 | package io.namjune.support; import java.io.IOException; import javax.servlet.Filter; import javax.servlet.FilterChain; import javax.servlet.FilterConfig; import javax.servlet.ServletException; import javax.servlet.ServletRequest; import javax.servlet.ServletResponse; import javax.servlet.annotation.WebFilter; import org.slf4j.Logger; import org.slf4j.LoggerFactory; @WebFilter(urlPatterns = { "/*" }) public class CharacterEncodingFilter implements Filter { static final Logger logger = LoggerFactory.getLogger(CharacterEncodingFilter.class); @Override public void init(FilterConfig filterConfig) throws ServletException { logger.debug("character encoding filter init!"); } @Override public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException { request.setCharacterEncoding("UTF-8"); chain.doFilter(request, response); } @Override public void destroy() { } } | cs |
logback.xml 파일에 설정한 configuration 형식대로 로그가 출력된다.
해당 메시지는 DEBUG 로그 레벨 메시지이다.
1 2 3 | ... 23:30:06.266 [localhost-startStop-1] DEBUG i.n.support.CharacterEncodingFilter - character encoding filter init! ... | cs |
개발 단계에서 디버깅을 위해 로그 레벨을 DEBUG로 설정하고 개발을 한다. 그러나 실제 프로그램을 운영하는 단계에서는 DEBUG 레벨의 메시지들이 모두 로그로 찍히면 프로그램의 성능 이슈가 생긴다. 특히, 로그를 파일로 저장한다면 파일 I/O에 대한 비용이 발생하면서 성능에 직접적인 영향을 준다. 이것은 디버그 메시지와 사용자가 많을수록 더 커진다.
따라서, logback 설정파일을 이용하여 로그 레벨을 변경할 수 있다. 위에서 기본으로 설정한 logback.xml 파일을 보면, root 태그에 설정한 로그 레벨이 DEBUG로 되어 있다. 애플리케이션의 default 로그 레벨을 DEBUG가 아닌 INFO 또는 WARN으로 설정하면, 해당 레벨의 메시지만 로그로 남길 수 있다.
1 2 3 4 5 6 7 8 9 10 11 | <configuration> <appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender"> <encoder> <pattern>%d{HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n</pattern> </encoder> </appender> <root level="INFO or WARN"> <appender-ref ref="STDOUT" /> </root> </configuration> | cs |
logger.debug("debug message");
를 통해 출력하는 DEBUG 레벨의 메시지는 출력하지 않는다. 따라서, 개발단계에서는 DEBUG 레벨로 개발하고, 실제로 서비스를 운영할 때에는 default 로그 레벨을 INFO 레벨이나 WARN 레벨로 변경하여 성능 이슈가 발생하지 않도록 한다.
실제로 로깅 프레임워크를 사용하다 보면, 의존하고 있는 라이브러리에 의해서 찍히는 로그들을 상당히 많이 볼 수 있다. Java Bean Validation Check를 하기 위해 Hibernate Validator를 사용하는 경우 해당 라이브러리에서 디버깅을 위해 출력하는 메시지가 자연스레 찍히는 경우가 이에 해당한다. 이렇게 개발자가 원하지 않은 로그 메시지를 출력하지 않기 위해서 패키지별로 Logger를 관리할 수 있다.
아래와 같이 configuration에 logger를 등록하고 관리한다. 다음의 logback 설정은 io.namjune 패키지에 대해서는 DEBUG 로그 레벨의 메시지를 출력하고, 애플리케이션 default 로그 레벨은 WARN으로 설정한다. 이렇게 설정을 할 경우, 개발자가 의도한 패키지의 DEBUG 레벨 로그 메시지만 출력하고, Hibernate Validator 등 의존 라이브러리에서 찍히는 DEBUG 로그 레벨의 메시지는 출력되지 않는다.
1 2 3 4 5 6 7 8 9 10 11 12 13 | <configuration> <appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender"> <encoder> <pattern>%d{HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n</pattern> </encoder> </appender> <logger name="io.namjune" level="DEBUG" /> <root level="WARN"> <appender-ref ref="STDOUT" /> </root> </configuration> | cs |
아래와 같이 해당 패키지의 로그 레벨을 INFO 레벨로 설정했다고 가정하자. io.namjune 패키지에 대해서는 INFO 레벨 이상의 로그 메시지만 출력될 것이다. 그리고 설정된 INFO 레벨 아래의 DEBUG, TRACE 단계의 로그 메시지를 출력하는 로직이 수행될 경우 개발자가 의도하지 않은 성능상 이슈가 발생한다.
1 2 3 4 5 6 7 8 9 10 11 12 13 | <configuration> <appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender"> <encoder> <pattern>%d{HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n</pattern> </encoder> </appender> <logger name="io.namjune" level="INFO" /> <root level="WARN"> <appender-ref ref="STDOUT" /> </root> </configuration> | cs |
아래와 같이 작성된 Servlet 코드를 보자. Java 코드가 동작하는 과정을 생각해보면, logger.debug() 메소드를 실행하기 위해서 String과 String을 더하는 연산 즉, 비용이 발생한다. INFO 레벨의 로그 메시지만 찍히길 바라는 개발자로서는 필요하지 않은 성능 이슈이다.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 | @WebServlet("/users/create") public class CreateUserServlet extends HttpServlet { static final Logger logger = LoggerFactory.getLogger(CreateUserServlet.class); @Override protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { String userId = request.getParameter("userId"); String password = request.getParameter("password"); String name = request.getParameter("name"); String email = request.getParameter("email"); User user = new User(userId, password, name, email); logger.debug("User : " + user); ... } } | cs |
logback에서는 다음과 같이 구현할 수 있다. 아래와 같이 구현하면, 먼저 debug() 메소드가 현재 로그 레벨을 판단하고 로깅 메시지의 출력을 결정한다. 로깅 메시지의 출력 여부가 결정된 다음에, 전달되는 두 개의 인자의 formatting 연산을 통해 메시지를 구성한다. 결과적으로 불필요한 연산의 비용이 발생하지 않는다.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 | @WebServlet("/users/create") public class CreateUserServlet extends HttpServlet { static final Logger logger = LoggerFactory.getLogger(CreateUserServlet.class); @Override protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { String userId = request.getParameter("userId"); String password = request.getParameter("password"); String name = request.getParameter("name"); String email = request.getParameter("email"); User user = new User(userId, password, name, email); logger.debug("User : {}", user); ... } } | cs |
손에 익은 sout + tab 보다는 Logging Framwork을 사용하는 습관을 기르자.
References
Contributor
Namjun Kim<namjunemy@gmail.com>
'ICT Eng > JAVA' 카테고리의 다른 글
소스코드 배포 과정에 대한 자동화 쉘 스크립트 (0) | 2018.03.13 |
---|---|
[JAVA] Stream, 스트림과 병렬처리 (3) | 2018.01.04 |
[JAVA] Generic, 제네릭 (3) | 2017.11.08 |
[JAVA] 자바의 멀티 스레드 (0) | 2017.10.11 |
[JAVA] DI(Dependency Injection)를 이용한 빈 의존성 관리 (0) | 2017.09.14 |
- Total
- Today
- Yesterday
- 순환
- Vue.js
- Spring Boot
- springboot
- Java
- Raspberry Pi
- 한밭대학교
- 시간복잡도
- 정렬
- 한밭이글스
- github
- 알고리즘
- ORM
- 무선통신소프트웨어연구실
- 스프링부트
- vuex
- 라즈베리파이
- 자바
- vuejs
- AWS
- Wisoft
- IT융합인력양성사업단
- JPA
- Algorithm
- Recursion
- 젠킨스
- 인프런
- Spring
- RBT
- 레드블랙트리
일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
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 |