异常处理、生成式对象

1.异常常见类型

SyntaxError:语法错误
NameError:名字错误,一般由于变量名未定义造成
IndexError:索引错误,列表的索引值超过了范围
KeyError:字典键错误,字典的键找不到
IndentationError:缩进错误

2.异常处理语法结构

1.基本语法结构:
    try:
        待监测的代码(可能会出错的代码)
    except 错误类型:
        针对上述错误类型指定的方案
        
try:
    print(name)
except NameError:
    print('变量名没有定义就使用了')  # 变量名没有定义就使用了
"""
如果except后面的错误类型正确,那么就不会报错,而是直接执行except的子代码,错误类型不对还是会报错
"""    
        
2.查看错误信息:
    try:
        待监测的代码(可能会出错的代码)
    except 错误类型 as e:
        针对上述错误类型指定的方案
try:
    print(name)
except NameError as e:
    print('变量名没有定义就使用了')
    print(e)  # 变量名没有定义就使用了 name 'name' is not defined
"""
加一个参数e可以查看具体错误,也可以加上其他代码
"""

3.针队不同的错误类型制定不同的解决方案
	 try:
        待监测的代码(可能会出错的代码)
    except 错误类型1 as e:
        针对上述错误类型指定的方案
    except 错误类型2 as e:
        针对上述错误类型指定的方案
    except 错误类型3 as e:
        针对上述错误类型指定的方案
        
try:
    print(name)
    l1 = [1, 2, 3, 4]
    l1[5]
except IndexError as e:
    print(e)
except NameError as e:
    print(e)
except KeyError as e:
    print(e)  # name 'name' is not defined
"""
如果不确定是哪种错误类型,可以采用上面的格式。但是该形式找到地一个错误类型之后就会停止运行,也就是说运行一次只能找到一处错误
"""

4.万能异常
try:
    待监测代码(可能会出现的代码)
except Exception as e:
    针队各种常见的错误类型全部统一处理

try:
    print(name)
    l1 = [1, 2, 3, 4]
    l1[5]
except Exception as e:
    print(e)  # name 'name' is not defined
"""
一次只能检测一处错误
"""

5.结合else使用:被监测代码没有错误时执行else子代码
try:
    待监测的代码(可能会出现错误的代码)
except Exception as e:
    针队各种常见的错误类型全部统一处理
else:
    try的子代码正常运行结束没有任何的报错后 再执行else子代

try:
    # print(name)
    l1 = [1, 2, 3, 4]
    # l1[5]
except Exception as e:
    print(e)
else:
    print('没有其他错误了')  # 没有其他错误了
    
6.结合finally使用:
    try:
        待监测的代码(可能会出现错误的代码)
    except Exception as e:
        针队各种常见的错误类型全部统一处理
    else:
        try的子代码正常运行结束没有任何的报错后 再执行else子代
	 finally:
        无论try子代码是否报错,最后都要执行finally子代码 
try:
    print(name)
except Exception as e:
    print(e)
else:
    print('没有错误了')
finally:
    print('嘿嘿嘿')  # name 'name' is not defined 嘿嘿嘿

try:
    name = 'jason'
    print(name)
except Exception as e:
    print(e)
else:
    print('没有错误了')
finally:
    print('嘿嘿嘿')  # jason 没有错误了 嘿嘿嘿

3.异常处理补充

1.断言:关键字assert,后面的代码如果是对的那么会执行下面的代码,是错的直接会报错
info = 'jerry'
assert isinstance(info, str)
print('看我是否会执行')  # 看我是否会执行

info = 'jerry'
assert isinstance(info, int)
print('看我是否会执行')  # 报错	

2.主动抛异常:如果不满足条件则会执行else分支结构,满足条件则直接报错
name = 'max'
if name == 'max':
    raise Exception('如果是max则执行我')
else:
    print('name不是max才执行我')  # 执行结果:报错
    
name = 'jason'
if name == 'max':
    raise Exception('如果是max则执行我')
else:
    print('name不是max才执行我')  # name不是max才执行我

4.异常处理实战应用

1.异常处理能尽量少用就少用
2.被try监测的代码能尽量少就尽量少
3.当代码中可能会出现一些无法控制的情况报错才应该考虑使用
	eg: 使用手机访问网络软件 断网
      编写网络爬虫程序请求数据 断网

练习:使用while循环+异常处理+迭代器对象 完成for循环迭代取值的功能
l1 = [11, 22, 33, 44, 55, 66, 77, 88, 99]
new_l1 = l1.__iter__()
while True:
    try:
        print(new_l1.__next__())
    except Exception as e:
        break

5.生成器对象

1.本质:
    内置有__iter__和__next__的迭代器对象
    
2.区别:
    迭代器对象是解释器自动提供的,由数据类型,文件对象转变为迭代器对象。
    生成器对象是程序员编写出来的
    
3.创建生成器的基本语法:函数团体代码中填写yield关键字
"""
1.函数体代码中如果有yield关键字,那么函数名加括号不会执行函数体代码,而是会生成一个生成器对象(迭代器对象)
"""
def my_iter():
    print('生成器就是迭代器的一种')
    yield 
my_iter()
"""
2.把加括号的结果赋值给一个变量名,变量名调用__next__才会执行函数体代码
"""
def my_iter():
    print('生成器就是迭代器的一种')
    yield
my_iter()
res = my_iter()
res.__next__()
"""
3.多加一个res.__next___()会报错,因为执行一次代码会停在yield后面(类似鼠标的移动),下次基于该位置再继续往下找第二个yield
"""
def my_iter():
    print('生成器就是迭代器的一种')
    yield
    print('生成器就是迭代器的一种111')
    yield
    print('生成器就是迭代器的一种222')
    yield
my_iter()
res = my_iter()
res.__next__()
res.__next__()
res.__next__()  # 生成器就是迭代器的一种 生成器就是迭代器的一种111 生成器就是迭代器的一种222
"""
4.yield类似return,可以有返回值
"""
def my_iter():
    print('生成器就是迭代器的一种')
    yield 0
    print('生成器就是迭代器的一种111')
    yield 111
    print('生成器就是迭代器的一种222')
    yield 222
my_iter()
res = my_iter()
res1 = res.__next__()
print(res1)
res2 = res.__next__()
print(res2)
res3 = res.__next__()  
print(res3)  # 生成器就是迭代器的一种 0 生成器就是迭代器的一种111 111 生成器就是迭代器的一种222 222

6.自定义生成器range()功能

1.两个参数:
def index(start_num, end_num):
    while start_num < end_num:
        yield start_num
        start_num += 1

for i in index(1, 10):
    print(i)
"""
深层原理:
res = index(1, 10).__iter__()
while True:
    try:
        print(res.__next__())
    except Exception:
        break
"""

        
2.一个参数
"""
for i in range(10)代表10是end_num,所以要首先把10传给第二个参数,再把第一个参数变成0
"""
def my_range(start_num, end_num=None):
    if not end_num:
        end_num = start_num
        start_num = 0
    while start_num < end_num:
        yield start_num
        start_num += 1

for i in my_range(10):
    print(i)
    
3.三个参数
def my_len(start_num, end_num=None, step=1):
    if not end_num:
        end_num = start_num
        start_num = 0
    while start_num < end_num:
        yield start_num
        start_num += step


for i in my_len(1, 10, 2):
    print(i)

7.yield冷门用法

如果yield后面有值,函数体代码会执行到yield行并且返回yield后面的值,如果yield前面有变量名和赋值符号,通过send给此变量名传值,调用一次相当于调用一次双下next
def eat(name, food=None):
    print(f'{name}准备用餐')
    while True:
        food = yield
        print(f'{name}正在吃{food}')

res = eat('jason')  # 变成一个生成器
res.__next__()  # 执行函数体代码
res.send('汉堡')
res.send('包子')

8.生成器表达式

列表生成式:
l1 = [i*3 for i in range(1, 10)]
print(l1)  # [3, 6, 9, 12, 15, 18, 21, 24, 27]

元组生成器:组成一个新元组时直接打印变量名是一个内存地址,用for循环才能得到元组中的值
t1 = (i*3 for i in range(1, 10))
print(t1)  # <generator object <genexpr> at 0x00000284A176F190>
for i in t1:
    print(i)
    
"""
面试题:
"""
def add(n, i):  # 1.普通函数 返回两个数的和  求和函数
    return n + i
def test():  # 2.有yield关键字,定义生成器
    for i in range(4):
        yield i
g = test()  # 3.激活生成器
for n in [1, 10]:  # 4.第一次n=1  5.第二次n=10
    g = (add(n, i) for i in g)  # 5.第一次g = (add(1, i) for i in g)
    #  6.g = (add(10, i) for i in (add(10, i) for i in g))
    """
    第一次for循环
        g = (add(n, i) for i in g)
    第二次for循环
        g = (add(10, i) for i in (add(10, i) for i in g))
    """
res = list(g)  # 7.list(g)相当于for循环
print(res)
"""
1.add(n, i)就是一个普通的求和函数,test()中含有yield关键字,所以g = test()产生了一个生成器对象。
2.生成器对象的特点就是惰性,不进行迭代就不运算。迭代有以下几种方式: 
    2.1for循环,for循环的本质就是调用了__iter__和__next__方法进行了迭代
    2.2调用__next__方法
    2.3调用send方法
    2.4数据类型强制转换,比如使用list()强制转换(相当于是对其进行for循环)
    
3.对n进行for循环时,第一次n=1,生成器没有迭代,所以i没有值,第二次n=10,同样i没有值。
第一次:g = (add(n, i) for i in g)(拿不到值就不会执行,所以n=1放进去没有意义,n就是一个变量名),g=test(),所以g = (add(n, i) for i in test())
第二次:g = (add(n, i) for i in (add(n, i) for i in test())),此时for循环结束,结束时n=10

4.此时list(g),生成器对象开始执行,test()的返回值i循环了4次,分别是0,1,2,3,这4个值被g表达式中的i接收,所以add(n,i)的值是10,11,12,13

5.此时for i in (add(n, i) for i in test()),i拿到的值是20,21,22,23。对其循环拿到的值就是20,21,22,23。
"""
posted @ 2023-04-24 20:31  ERROR404Notfound  阅读(35)  评论(0编辑  收藏  举报
Title