try:
...
except [발생 오류[as 오류 메시지 변수]]:
...
try 블록 수행 중 오류가 발생하면 except 블록이 수행되고,
try 블록에서 오류가 발생하지 않는다면 except 블록은 수행되지 않는다.
1. try, except만 쓰는 방법
try:
...
except:
...
오류 종류에 상관없이 오류가 발생하면 except 블록을 수행
2. 발생 오류만 포함한 except문
try:
...
except 발생 오류:
...
오류가 발생했을 때 except문에 미리 정해 놓은 오류 이름과 일치할 때만 except 블록을 수행
3. 발생 오류와 오류 메시지 변수까지 포함한 except문
try:
...
except 발생 오류 as 오류 메시지 변수:
...
두 번째 경우에서 오류 메시지의 내용까지 알고 싶을 때 사용하는 방법
ex)
try:
5 / 0
except ZeroDivisionError as e:
print(e)
위처럼 4를 0으로 나누려고 하면 ZeroDivisionError가 발생하여 except 블록이 실행되고 변수 e에 담기는 오류 메시지가 출력
결과값: division by zero
3. 여러개의 오류처리하기
try:
...
except 발생 오류1:
...
except 발생 오류2:
...
ex)
try:
a = [1,2]
print(a[3])
5/0
except ZeroDivisionError:
print("0으로 나눌 수 없습니다.")
except IndexError:
print("인덱싱 할 수 없습니다.")
a는 2개의 요소값을 가지고 있기 때문에 a[3]는 IndexError 발생 "인덱싱할 수 없습니다."라는 문자열이 출력
인덱싱 오류가 먼저 발생했으므로 5/0으로 발생되는 ZeroDivisionError 오류는 발생하지 않았다.
try:
a = [1,2]
print(a[3])
5/0
except ZeroDivisionError as e:
print(e)
except IndexError as e:
print(e)
프로그램을 실행하면 "list index out of range" 오류 메시지가 출력
다음과 같이 ZerroDivisionError와 IndexError를 함께 처리할 수도 있다.
try:
a = [1,2]
print(a[3])
5/0
except (ZeroDivisionError, IndexError) as e:
print(e)
2개 이상의 오류를 동일하게 처리하기 위해서는 위와 같이 괄호를 사용하여 함께 묶어 처리하면 된다.
4.try문에 else절 사용하기
try:
...
except [발생 오류[as 오류 메시지 변수]]:
...
else: # 오류가 없을 경우에만 수행된다.
...
try문 수행중 오류가 발생하면 except절이 수행되고 오류가 없으면 else절이 수행된다.
ex)
try:
age=int(input('수학 점수 입력: '))
except:
print('입력이 정확하지 않습니다.')
else:
if age < 60:
print('재시험이 필요합니다.')
else:
print('합격입니다.')
'수학 점수 입력:' 라는 input에 숫자가 아닌 다른 값을 입력하면 오류가 발생하여 '입력이 정확하지 않습니다.'라는 문장을 출력.
오류가 없을 경우에만 else절이 수행.
5.오류 회피하기
특정 오류가 발생할 경우 그냥 통과시키기
try:
f = open("파일없음", 'r')
except FileNotFoundError:
pass
try문 안에서 FileNotFoundError가 발생할 경우에 pass를 사용하여 오류를 회피함
6.try .. finally
finally절은 try문 수행 도중 예외 발생 여부에 상관없이 항상 수행된다.
보통 finally절은 사용한 리소스를 close해야 할 때에 많이 사용한다.
ex)
f = open('foo.txt', 'w')
try:
# 무언가를 수행한다.
finally:
f.close()
foo.txt 파일을 쓰기 모드로 연 후에 try문을 수행한 후 예외 발생 여부와 상관없이 finally절에서 f.close()로 열린 파일을 닫을 수 있다.
오류 일부러 발생시키기
파이썬은 raise 명령어를 사용해 오류를 강제로 발생시킬 수 있다.
예를 들어 Candy 클래스를 상속받는 자식 클래스는 반드시 eat라는 함수를 구현하도록 만들고 싶은 경우(강제로 그렇게 하고 싶은 경우)
class Candy:
def eat(self):
raise NotImplementedError
위 예제는 Candy 클래스를 상속받는 자식 클래스는 반드시 eat 함수를 구현해야 한다는 의지를 보여 준다. 만약 자식 클래스가 eat 함수를 구현하지 않은 상태로 eat 함수를 호출한다면 어떻게 될까?
※ NotImplementedError는 파이썬 내장 오류로, 꼭 작성해야 하는 부분이 구현되지 않았을 경우 일부러 오류를 일으키기 위해 사용한다.
class snack(Candy):
pass
snack = Snack()
snack.eat()
Snack 클래스는 Candy 클래스를 상속받는다. 그런데 Snack 클래스에서 eat 함수를 구현하지 않았기 때문에 Candy 클래스의 eat 함수가 호출된다. 그리고 raise문에 의해 NotImplemented Error가 발생할 것이다.
※ 상속받는 클래스에서 함수를 재구현하는 것을 메서드 오버라이딩이라고 부른다.
Traceback (most recent call last):
File "...", line 33, in <module>
snack.eat()
File "...", line 26, in eat
raise NotImplementedError
NotImplementedError
NotImplementedError가 발생되지 않게 하려면 다음과 같이 Snack 클래스에 eat 함수를 반드시 구현해야 한다.
class Snack(Candy):
def eat(self):
print("very fast")
snack = Snack()
snack.eat()
위 예처럼 eat 함수를 구현한 후 프로그램을 실행하면 오류 없이 다음 문장이 출력된다.
very fast
예외 만들기
예외는 다음과 같이 파이썬 내장 클래스인 Exception 클래스를 상속하여 만들 수 있다.
class MyError(Exception):
pass
그리고 별명을 출력해 주는 함수를 다음과 같이 작성한다.
def say_nick(nick):
if nick == '바보':
raise MyError()
print(nick)
그리고 다음과 같이 say_nick 함수를 호출해 보자.
say_nick("천사")
say_nick("바보")
저장한 뒤 프로그램을 실행해 보면 다음과 같이 "천사"가 한 번 출력된 후 MyError가 발생한다.
천사
Traceback (most recent call last):
File "...", line 11, in <module>
say_nick("바보")
File "...", line 7, in say_nick
raise MyError()
__main__.MyError
예외 처리 기법을 사용하여 MyError 발생을 예외 처리하기
try:
say_nick("천사")
say_nick("바보")
except MyError:
print("허용되지 않는 별명입니다.")
프로그램을 실행하면 다음과 같이 출력된다.
천사
허용되지 않는 별명입니다.
만약 오류 메시지를 사용하고 싶다면 다음처럼 예외 처리를 하면 된다.
try:
say_nick("천사")
say_nick("바보")
except MyError as e:
print(e)
하지만 프로그램을 실행해 보면 print(e)로 오류 메시지가 출력되지 않는 것을 확인할 수 있다. 오류 메시지를 출력했을 때 오류 메시지가 보이게 하려면 오류 클래스에 다음과 같은 str 메서드를 구현해야 한다. str 메서드는 print(e)처럼 오류 메시지를 print문으로 출력할 경우에 호출되는 메서드이다.
class MyError(Exception):
def __str__(self):
return "허용되지 않는 별명입니다."
프로그램을 다시 실행해 보면 "허용되지 않는 별명입니다."라는 오류메시지가 출력되는 것을 확인할 수 있다.