Java에서 Throwable 객체로 콜스택을 출력할 수 있습니다. 현재 코드 위치까지 어떤 함수들을 거쳐왔는지 보여주는 것을 Stack trace라고 합니다.
Throwable의 다음 메소드들을 이용하여 Trace를 가져오거나 출력할 수 있습니다.
// Throwable.java
public void printStackTrace()
public void printStackTrace(PrintStream s)
public StackTraceElement[] getStackTrace()
Java 코드를 보면 Exception 클래스는 Throwable 클래스를 상속받습니다. 그렇기 때문에 Exception 객체도 Throwable의 메소드를 이용할 수 있습니다.
public class Exception extends Throwable {
...
}
Stack trace 출력
보통 try-catch 패턴으로 예외처리를 합니다.
여기서 Exception.printStackTrace()
를 호출하면 로그에 Stack trace가 출력됩니다.
try {
Exception e = new Exception();
e.initCause(new IOException("No space memory"));
throw e;
} catch(Exception e) {
e.printStackTrace();
}
결과
java.lang.Exception
at example.StackTrace$AAA.ccc(StackTrace.java:21)
at example.StackTrace$AAA.bbb(StackTrace.java:14)
at example.StackTrace$AAA.aaa(StackTrace.java:10)
at example.StackTrace.main(StackTrace.java:47)
Caused by: java.io.IOException: No space memory
at example.StackTrace$AAA.ccc(StackTrace.java:22)
... 3 more
Throwable은 printStackTrace(PrintWriter s)
메소드도 지원합니다.
다음과 같이 PrintWriter를 인자로 전달하여 Trace를 PrintWriter에 추가할 수 있습니다.
try {
Exception e = new Exception();
e.initCause(new IOException("No space memory"));
throw e;
} catch(Exception e) {
StringWriter sw = new StringWriter();
PrintWriter pw = new PrintWriter(sw);
pw.append("+++Start printing trace:\n");
e.printStackTrace(pw);
pw.append("---Finish printing trace");
System.out.println(sw.toString());
}
결과
+++Start printing trace:
java.lang.Exception
at example.StackTrace$AAA.fff(StackTrace.java:50)
at example.StackTrace$AAA.bbb(StackTrace.java:22)
at example.StackTrace$AAA.aaa(StackTrace.java:12)
at example.StackTrace.main(StackTrace.java:75)
Caused by: java.io.IOException: No space memory
at example.StackTrace$AAA.fff(StackTrace.java:51)
... 3 more
---Finish printing trace
Exception 없이 Stack trace 출력
위의 코드들은 Exception이 발생할 때만 사용할 수 있습니다.
그냥 디버깅 목적으로 Trace를 출력하고 싶다면 아래와 같이 Throwable 객체를 생성하고 printStackTrace()
를 호출하면 됩니다.
throw
하는 것이 아니기 때문에 로그만 출력되고 객체는 소멸됩니다.
// print stack trace
(new Throwable()).printStackTrace();
결과
java.lang.Throwable
at example.StackTrace$AAA.eee(StackTrace.java:39)
at example.StackTrace$AAA.bbb(StackTrace.java:16)
at example.StackTrace$AAA.aaa(StackTrace.java:10)
at example.StackTrace.main(StackTrace.java:47)
Trace를 출력하지 않고 String으로 가져오기
Trace를 로그에 출력하지 않고 String 객체로 가져올 수 있습니다.
아래와 같이 getStackTrace()
을 호출하면, StackTraceElement
배열을 리턴합니다.
StackTraceElement 객체는 Trace의 정보를 갖고 있으며, Stack이 많이 쌓일 수록 배열 길이도 길어집니다.
이 객체의 아이템들을 String으로 출력할 수 있습니다.
// get stack trace and print
StackTraceElement[] stacks = (new Throwable()).getStackTrace();
for (StackTraceElement element : stacks) {
System.out.println(element);
}
결과
example.StackTrace$AAA.ddd(StackTrace.java:31)
example.StackTrace$AAA.bbb(StackTrace.java:15)
example.StackTrace$AAA.aaa(StackTrace.java:10)
example.StackTrace.main(StackTrace.java:47)
예제
아래 코드는 위에서 예제로 사용한 코드입니다.
package example;
import java.io.IOException;
import java.io.PrintWriter;
import java.io.StringWriter;
public class StackTrace {
static class AAA {
void aaa() {
bbb();
}
void bbb() {
ccc();
sleep(1000);
ddd();
sleep(1000);
eee();
sleep(1000);
fff();
}
void ccc() {
try {
Exception e = new Exception();
e.initCause(new IOException("No space memory"));
throw e;
} catch(Exception e) {
e.printStackTrace();
}
}
void ddd() {
// get stack trace and print
StackTraceElement[] stacks = (new Throwable()).getStackTrace();
for (StackTraceElement element : stacks) {
System.out.println(element);
}
}
void eee() {
// print stack trace
(new Throwable()).printStackTrace();
}
void fff() {
try {
Exception e = new Exception();
e.initCause(new IOException("No space memory"));
throw e;
} catch(Exception e) {
StringWriter sw = new StringWriter();
PrintWriter pw = new PrintWriter(sw);
pw.append("+++Start printing trace:\n");
e.printStackTrace(pw);
pw.append("---Finish printing trace");
System.out.println(sw.toString());
}
}
void sleep(long ms) {
try {
Thread.sleep(ms);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
public static void main(String args[]) {
AAA a = new AAA();
a.aaa();
}
}
참고
Related Posts
- Java - Unsupported class file major version 61 에러
- Java - String.matches()로 문자열 패턴 확인 및 다양한 예제 소개
- Java - 문자열 공백제거 (trim, replace)
- Java - replace()와 replaceAll()의 차이점
- Java - ArrayList 초기화, 4가지 방법
- Java - 배열 정렬(Sorting) (오름차순, 내림차순)
- Java - 문자열(String)을 비교하는 방법 (==, equals, compare)
- Java - StringBuilder 사용 방법, 예제
- Java - 로그 출력, 파일 저장 방법 (Logger 라이브러리)
- Java IllegalArgumentException 의미, 발생 이유
- Java - NullPointerException 원인, 해결 방법
- Seleninum의 ConnectionFailedException: Unable to establish websocket connection 해결
- Java - compareTo(), 객체 크기 비교
- Java - BufferedWriter로 파일 쓰기
- Java - BufferedReader로 파일 읽기
- Java charAt() 함수 알아보기
- Java - BigInteger 범위, 비교, 연산, 형변환
- Java contains()로 문자(대소문자 X) 포함 확인
- Java - Set(HashSet)를 배열로 변환
- Java - 문자열 첫번째 문자, 마지막 문자 확인
- Java - 문자열 한글자씩 자르기
- Java - 문자열 단어 개수 가져오기
- Java - 1초마다 반복 실행
- Java - 배열을 Set(HashSet)로 변환
- Java - 여러 Set(HashSet) 합치기
- Java - 명령행 인자 입력 받기
- Java - 리스트 역순으로 순회, 3가지 방법
- Java - 특정 조건으로 리스트 필터링, 3가지 방법
- Java - HashMap 모든 요소들의 합계, 평균 계산
- Java - 특정 조건으로 HashMap 필터링
- Java - 싱글톤(Singleton) 패턴 구현
- Java - 숫자 왼쪽에 0으로 채우기
- Java - String 배열 초기화 방법
- Java - 정렬된 순서로 Map(HashMap) 순회
- Java - HashMap에서 key, value 가져오기