기본 콘텐츠로 건너뛰기

작년에 왔던 각설이 같은 자바에서의 Exception 처리 관련 글 소개



잊을만하면 한 번씩 올라오는 자바 Exception에 대한 글을 하나 소개할까 합니다.
작년에 왔던 각설이 죽지도 않고 또 오는 거 보면 정말 자바에서 중요한 부분이지 않을까 합니다.
개인적으로도 Exception에 대해서는 최대한 잘 처리를 하려고 하는 편이지만, Best Practice에 반하는 경우도 많아서 다시 한번 정리를 해 볼 겸 해서 간단히 정리해 봅니다.

원문은 아래 링크를 통해서 확인 가능합니다. 원문 꼭 확인 하시길 추천하는건 다 아시리라 생각하며...



1. Catch 블록에서 Exception을 확인할 수 있는 어떤 처리를 하지 않고 그냥 소비하지 말라고 합니다. 즉, Catch 블록에서 로그를 남기지 않고 그냥 리턴해 버리거나 하는 경우인데요. 저도 가끔은 그냥 이렇게 하는 경우도 있긴 합니다만, Exception이 정말 발생하면 안 되는 경우라면 절대 그러면 안 되겠지요? 해당 Exception이 발생했는지를 모른다면 안 될 테니 말이죠. 적당한 대응을 위해서라도 어떤 Exception이 발생해서 처리를 한 것인지 정도는 알 수 있도록 프로그램 하는 게 맞습니다. 만약 Exception에 대한 대응이 필요 없다면 차라리 10번 항목처럼 처리를 하는 것을 추천합니다.

2. Method에 throws 를 지정할 때 Exception으로 하지 말라고 합니다. 즉, Exception이 아니라 특정 Exception을 지정하라는 것인데요. 개인적으로는 본인이 특정 Exception을 상속받아서 만든 Exception을 throws 하는 것을 추천합니다. 물론 항상 그런 건 아니고요.

3. 2번과 비슷한 맥락에서 catch 를 할 때 Exception 으로 catch 하지 말라는 것입니다. 네. 맞습니다. 당연한 이야기입니다. 근데... 저도 그렇지만 귀찮으면 그냥 그렇게 프로그램 하는 경우가 많지요^^

4. Never catch any Throwable class. 설명이 좀 난해해서 패스.. 어떤 경우를 말하는 것인지 잘 모르겠네요.

5. Catch 블록에서 throw를 하는 경우 2번의 경우처럼 본인이 작업한 Custom Exception을 발생시키는데 이때 에러 정보를 정확히 포함시켜서 보내라는 내용입니다. 즉, message만 만들어서 Exception을 throw 하지 말고 exception 객체를 같이 넘기라는 내용인데요. 원문에는 샘플이 있습니다. 샘플 보시는 것이 한방에 이해돼요.

6. Catch 블록에서 로그를 남기거나 throw를 하게 되는데, 두 가지를 함께 하지 말라는 내용입니다. throw 한다면 throw를 catch 한 쪽이 있을 수 있으니 throw 하면서 Log를 남길 필요는 없겠죠? 예전 회사에서 작업할 때 이런 코드들이 많이 생기곤 했는데... 제가 만든 클래스에서 임시적으로 확인을 위해 Catch를 하고 로그를 남긴 후에 Custom Exception을 throw 했는데 사용하는 쪽에서 당연하게도 Log를 남기는 거죠. 이럼 로그양만 늘어날 뿐이죠. 나중에 확인하고 지우긴 했지만, 아예 두 가지를 같이 쓰지 않는다면 해결되는 문제입니다.

7. finally 블록에서 throw가 일어날 상황을 만들지 말라는 내용입니다. 당연하죠? finally에서 Exception 발생하면 어떻게 될까요?

8. 처리하지 않을 Exception에 대해서 catch 하지 말라는 이야기입니다. 보통 많이 하는 게 catch 블록 내에 해당 Exception을 그대로 throw 하는 경우가 되겠습니다. 그런데 보통... 이렇게 하나요?

9. catch 블록에서 에러에 대해 임시로라도 printStackTrace() Method를 사용하지 말라는 것입니다. 요즘은 보통 많이 안 쓰시지 않나요? 

10. 일반적인 Exception을 처리하기 위해 try를 쓰긴 했는데 어떤 Exception에도 별다른 처리를 하지 않을 경우엔 그냥 finally 블록만 가지고 처리를 하라는 내용입니다. 1번에서 추천했던 방법입니다.

원문을 그대로 옮기지 않았습니다. 자세한 내용은 원문을 꼭 읽어주시길...

사실 이렇게 정리를 해 놓는 이유는 제가 나중에라도 보려 하거나 후배들에게 알려줘야 할 경우에 참고하기 위해서인데... 항상 원문을 훔쳐 온 것 같은 느낌이라... 꼭 원문을 읽어주세요.

[Update, 2020-05-19]

마침 Throwable에 대해 총정리 된 느낌의 Post를 발견했습니다(아래 링크입니다).
Exception을 정의하고 사용하는 방법들에 대한 다양한 부분을 볼 수 있는 글이네요.
꽤 이래저래 알고 있었다고 생각했는데, 보면서 많이 배웠습니다. 역시 고수는 많고 배울 것이 넘쳐나는 세상이라는 사실에 왠지 즐거워집니다. (전투력 재상승!!!)


댓글

이 블로그의 인기 게시물

경력 개발자의 자기소개서에 대해서...

갑자기 뜬금없이 이런 글을 쓰다니 무슨 생각이야? 라고 생각하시는 분들이 있을지도 모르겠네요. 뜬금없음에 대한 변명은 잠시 접어두고 일단 오늘 쓰려고 하는 글을 시작해볼까 합니다. 개발자로 대충 16년을 그럭저럭 보내왔습니다. 시대적 상황으로 5년 차쯤에 대리로 처음 팀장을 시작했으니, 일반 개발자로 산 시간보다는 어쨌건 프로젝트 또는 팀의 리더로 산 시간이 더 많았던 것 같습니다. 그 기간 동안 남들보다 좀 심하게 회사를 많이 옮겨 다니다 보니 꽤 많은 면접을 볼 수 있는 경험이 있었고, 또 옮긴 회사가 대부분 팀을 리빌딩하는 곳이었다 보니 꽤 많은 채용절차에 관여할 기회가 있어서 어린 나이부터 비교적 많은 이력서를 검토했고 면접관으로도 여러 사람을 만날 수 있었습니다. 처음 면접을 보러 다니던 시절의 제 이력서의 자기소개서는 항상 "19XX년 봄 XX업계에 종사하시던 아버님과 집안일에 헌신적인 어머니의 유복한 가정에 1남 1녀의 막내로..." 로 시작되었습니다 (이 문장에 향수를 느끼시는 분들 많으실 거예요. ^^). 경력이 5년이 넘은 어느 날 도대체 이 문장을 왜 써야 하느냐는 의문이 생겨서 조금 바꾸긴 했습니다만, 그 뒤로도 꽤 오랜 세월을 이런 자기소개서가 항상 제 이력서에 붙어있었죠. 요즘 누가 저런 식으로 자기소개서를 써? 라고 생각하시는 분들 많으실 거로 생각해요. (대신 요즘은 대학 시절의 봉사활동이나 해외연수 이력이... 뭐 어차피 그놈이 그놈입니다.) 저런 자기소개서를 써야 한다는 것이 어디서 어떻게 시작된 것인지는 몰라도 회사를 그만두기 전인 2년 전까지도 약간의 표현은 다를지 모르지만 비슷한 문장으로 시작하는 자기소개서를 이력서에 첨부해서 보내는 지원자들을 볼 수 있었습니다. 이제 제가 뜬금없는 이런 글을 쓰게 된 이유를 밝히고 계속 진행해야겠네요. 블로그에 올릴 글을 준비하는 일이 생각보다 힘들어요. 블로그에 올리려고 준비한 주제에 맞는 소스를 작업하고 거기에 글을 입히다 보면 가끔

Springframework 5에서 바뀌는 것들에 대한 간단 정리 및 생각

Spring framework 5 에 대해 많은 분이 기대와 두려움을 가지고 계시지 않을까 생각합니다. 특히 기대를 하고 계신 분들은 Reactive Programming 지원을 기대하고 계시지 않은가 생각이 드는데요. 7월 초에 John Thompson 이란 분이 D-Zone에 아주 깔끔하고 멋지게 정리를 잘해서 글을 쓰셨더라구요. 해당 글은  https://dzone.com/articles/whats-new-in-spring-framework-5 에서 확인을 하실 수 있습니다. 혹시 Spring framework 5에서 달라지는 내용의 좀 더 자세한 내용이 필요하신 분들은 Spring framework github의 wiki 를 참고하시면 됩니다. 본 포스트는 언제나 그렇듯이 윗글에 대한 번역이 아닙니다. 그저 윗글을 다시 정리하면서 제 생각을 한번 정리해 놓은 포스트입니다. Spring framework 5는 현재 5.0.0.RC2(2017.07.23일 기준)까지 릴리즈된 상황입니다. Spring framework 5에서 크게 변화하는 내용을 John Thompson은 8가지로 깔끔하게 정리해주고 있습니다. 1. JDK 지원 버전의 업데이트 5버전은 원래 JDK 9 버전의 지원을 위해서 시작됐던 프로젝트로 알고 있는데 맞는지는 모르겠네요. JDK 9의 Release가 늦어져서 Spring framework 5가 먼저 Release 될 것으로 보이지만, JDK 9가 Release가 되면 언제건 적용할 수 있다고 합니다. 좀 아쉬운 부분은 JDK의 최소 버전은 JDK 8이라는 부분이 아닐까 싶네요. 이 때문에 Spring framework 5에 무관심한 분들도 많으실 거라고 생각합니다. 지금 진행하는 프로젝트는 JDK 8을 기반으로 합니다만, 최근까지 다니던 회사의 경우는 JDK 7까지가 업그레이드 한계였던 회사였습니다. 아마도 JDK 업그레이드를 쉽게 못 하시는 회사들이 많으니 "나랑은 관계없는 얘기군"

자바에서 파일 타입을 확인하는 방법들... 그리고 Apache Tika...

뭐 역시나 내가 직접 조사한 것은 아니고 Java Code Geeks 에서 본 내용에 대한 정리이다. 자세한 사항은 [원문보기]  를 클릭해서 확인하면 된다. 원문에서는 코드와 스크린샷을 포함한 예제를 확인할 수 있으니 내용을 보시고자 하시는 분들은 꼭 원문을 참고하시길... 내가 쓰는 포스트가 대부분 그렇지만, 이 글도 그저 내용을 정리해서 내가 나중에 확인하기 위해 쓰는 글일 뿐... 1. Files.probeContentType(Path)를 이용하는 방법 [JDK 7] 이 방법은 파일 확장자를 이용해서 판단하는 것으로 보인다. 즉, gif 파일을 png라고 확장자를 주면 png 로 인식한다는 것. 또 확장자가 없는 파일의 경우는 null 을 리턴한다. 2. MimetypesFileTypeMap.getContentType(String) 을 이용하는 방법 [JDK 6] 이 방법은 클래스 명에서 유추가 가능한 것 처럼 Mime type 을 이용한다. 앞서의 경우와 같이 gif 파일을 png 라고 하거나 pdf 파일을 txt 로 하거나 하면 확장자로 판단을 한다. 1번 경우와 다른 점을 이야기하자면 기본적으로 텍스트나 이미지 확장자가 아닌 경우에는 xml 파일을 포함해서 application/octet-stream을 반환한다. 또한 확장자가 없는 파일의 경우도 application/octet-stream을 반환한다. 3. URLConnection.getContentType() 이 방법 역시 확장자에 의한 판단. 대신 null 대신 content/unknown을 반환 4. URLConnection.guessContentTypeFromName(String) 역시 확장자. 3번과 다른 것은 null을 반환. 5. URLConnection.guessContentTypeFromStream(InputStream) 예제에서는 전부 null 을 반환. 저자의 설명에 따르면 여기서 매개변수로 주어진 InputStream은 marks를