异常处理

一.关于异常处理

"""
程序错误分为两种: 语法错误 和 异常处理
语法错误:代码没有按照python规定语法去写,发明创造产生的错误
异常处理: 在代码语法正确的前提下,程序报错就是异常
"""


# try ... except... 基础语法  用于解决程序异常问题
# raise 可以主动抛异常,异常类可以自定义

二、认识异常处理

# IndexError                索引超出序列的范围
"""
lst = [1,2,3,4]
print(lst[100])
"""
IndexError
# KeyError                  字典中查找一个不存在的关键字
"""
dic = {"a":1}
dic["asdf"]
"""
KeyError
# NameError                 尝试访问一个不存在的变量
"""
print(name)
"""
NameError
# IndentationError          缩进错误
"""
if 5 == 5:
    print(1)
    print(2)
"""
IndentationError
# AttributeError            尝试访问未知的对象属性
"""
class A():
    a = 10
obj = A()
obj.b
"""
AttributeError
# StopIteration             迭代器没有更多的值
"""
it = iter([1,2,3])
for i in it:
    pass
res = next(it)
"""
StopIteration
# AssertionError             断言语句(assert)失败
"""
断言就是猜的意思
assert 和 if 之间的区别在于
assert 在断言失败时候,是直接报错,抛出异常,后面的代码直接终止了.
if     在判断为False 的时候,不执行代码
"""
assert 3>1
assert 3>1000
print(res)
print("ok")
AssertionError

三、异常处理的基本语法

1、基本语法

"""
try ... except ...
把有问题的代码放到try这个代码块当中
如果出现了异常,会直接执行except这个代码块中的内容
作用:防止异常抛错,终止程序.
"""
try :
    lst = [1,2,3]
    print(lst[90])
except:
    pass
    # print("这里有异常错误~")

2、带有分支的异常处理

class A():

    a = 10
obj = A()


try:
    # IndexError
    # lst = [1,2,3]
    # print(lst[90])
    
    # KeyError
    # dic = {}
    # print(dic["a"])
    
    # print(wangwen)
    obj.b()
    
except IndexError:
    print("索引超出了范围")
    
except KeyError:
    print("字典中没有这个键")

except NameError:
    print("没有这个变量")

except:
    print("有异常错误...")

3、处理迭代器异常错误

def mygen():
    print("start ... ")
    yield 1
    yield 2
    yield 3
    return 4

# 初始化生成器函数 -> 返回生成器对象 -> 简称生成器
gen = mygen()
try:
    res = next(gen)
    print(res)
    res = next(gen)
    print(res)
    res = next(gen)
    print(res)
    res = next(gen)
    print(res)
    
except StopIteration as e :
    # 为当前异常错误类StopIteration的对象起一个别名叫做e
    # 在StopIteration内部有__str__的方法
    # 在打印对象时,直接获取生成器中的return的返回值.
    print(e)
    print("生成器停止迭代")

4、异常处理的其他写法

4.1、try ... finally ...  不论代码正确与否,都必须执行的代码放到finally当中.

"""一报错会终止掉程序,后面的代码不执行,有些必须要走的代码写在finally中"""
print("<===============>")

try:
    lst = [1,2,3]
    print(lst[90])
finally:
    print("被触发了..")

print(123)

4.2、try .. except .. else ...

"""如果try这个代码块没有异常错误,执行else这个分支,反之就不执行"""
print("<===============>")
try:
    lst = [1,2,3]
    print(lst[90])
except:
    pass
else:
    print("正常执行结束...")

4.3、try .. except .. else .. finally .. 

try:
    lst = [1,2,3]
    print(lst[90])
except:
    print("异常处理222")
else:
    print("正常执行结束...")
finally:
    print("我被执行了111")
# (扩展)
# for/while  ... else .. 当循环遇到break异常终止循环时,不执行else分支
for i in range(10):
    if i  == 5:
        break
    # pass
else:
    print("循环结束")

四、raise 主动抛出异常

"""
raise + 异常错误类  or 异常错误类对象

BaseException 所有异常类的父类(基类,超类)
Exception     常见异常类的父类
"""

1、基本语法

try:
    raise BaseException
except BaseException:
    pass
    
# 简写
try:
    raise 
except:
    print("有异常错误")

2、自定义异常类 MyException (务必继承父类 BaseException)

#(了解)系统底层获取行数和文件名的函数( 只有在程序异常时才能触发 ) 
def return_errorinfo(n):
    import sys
    f = sys.exc_info()[2].tb_frame.f_back
    if n==1:        
        return str(f.f_lineno)      #返回当前行数
    elif n == 2:    
        return f.f_code.co_filename #返回文件名    

# 通过get_value 主动抛出异常
def get_value(n):
    try:
        raise
    except:
        return return_errorinfo(n)


class MyException(BaseException):
    def __init__(self,num,msg,line,file):
        # 错误号
        self.num = num
        # 错误信息
        self.msg = msg
        # 错误行号
        self.line = line
        # 错误文件
        self.file = file


sex = "雌雄同体"
try:
    if sex == "雌雄同体":
        # raise + 异常错误类 或者 异常错误类对象
        raise MyException(404,"醒醒吧老弟,人类没有雌雄同体~",get_value(1),get_value(2))

except MyException as e:
    # 对象.属性 
    print(e.num)
    print(e.msg)
    print(e.line)
    print(e.file)

五、反射(针对类对象或者模块)

"""概念: 通过字符串去操作类对象或者模块当中的成员(属性或者方法)"""
class Man():
    pass

class Woman():
    pass

class Children(Man,Woman):
    skin = "绿色"
    
    def eat(self):
        print("小孩一下生就拿了两串大腰子出来了")

    def drink(self):
        print("小孩一下生拿了两瓶勇闯天涯出来了")
    
    def __suger(self):
        print("小孩自个的糖糖是不给别人的...")
        
obj = Children()

1、反射类对象中的成员

#hasattr() 检测对象/类是否有指定的成员
# 对象

res = hasattr(obj,"skin")
print(res)

#
res = hasattr(Children,"eat")
res = hasattr(Children,"__suger")
print(res)

2、getattr() 获取对象/类成员的值

# 对象
res = getattr(obj,"skin")
print(res)

# 通过对象反射出来的方法是个绑定方法
func = getattr(obj,"drink")
func()

#
# 通过类反射出来的方法是个普通方法
func = getattr(Children,"eat")
func(1)

# 反射的成员不存在时,可以设置一个默认值防止报错
res = getattr(obj,"abcdefg1234","该成员不存在")
print(res)

# 综合案例
"""
strvar = input("请输入你要使用的方法")
if hasattr(obj,strvar):
    func = getattr(obj,strvar)
    func()
"""

3、setattr() 设置对象/类成员的值

# 对象
setattr(obj,"eye","白色")
print(obj.eye)
#
setattr(Children,"sex","女性")
print(Children.sex)
print(obj.sex)

#delattr() 删除对象/类成员的值 
# 对象
# delattr(obj,"eye")
# print(obj.eye) error

#
# delattr(Children,"sex")
# Children.sex

4、反射模块中的成员

import sys
print(sys.modules) # 返回的是系统模块的一个字典
# 获取当前本模块的对象 (通过__main__这个键来获取)
print(sys.modules["__main__"])
while True:
    selfmodule = sys.modules["__main__"]
    strvar = input("请输入你要反射的方法")
    if hasattr(selfmodule,strvar):
        func = getattr(selfmodule,strvar)
        func()
    else:
        print("没有这个方法")
原文地址:https://www.cnblogs.com/whc6/p/14292513.html