Python生成器、装饰器、异常

简介: 【10月更文挑战第15天】

【10月更文挑战第15天】
生成器的定义方式:在函数中使用yield

yield值:将值返回到调用处

我们需要使用next()进行获取yield的返回值

yield的使用以及生成器函数的返回的接收next()

def test():
    yield 1,2,3

t=test()
print(t)
#<generator object test at 0x01B77A48>
#生成器对象的意思


#我们在函数中使用这个yield关键字,那么这个函数就是一个生成器函数

#t存放的是生成器对象的信息
#yield的作用类似于return 将值返回到调用的地方

#我们如何获取这个1呢?

#next(生成器对象)
print(next(t))
#(1, 2, 3)

#yied返回的内容我们需要通过next()进行一个获取的操作

有yiled的函数被称为生成器函数

image.png

对于性质一来说的话,yield会将后面的数据进行返回,返回到调用处

对于性质二的话,我们运行完yield之后,这个函数的运行位置就会被记录下来了

然后我们在交互模式再次进行这个next()的使用,进行返回值的获取那么就会从上次函数中结束的位置进行开始寻找数据然后进行返回的操作

然后后面如果没有yield的话,有个print('abc')

那么这个函数会将abc进行返回的,但是最终会进行报错的

就像下图所示

image.png

从下面的图片我们可以看的出什么呢?

image.png

def test():
    yield 1,2,3
    print('abc')
    yield 'a'
t=test()
print(t)

print(next(t))
#(1, 2, 3)
print(next(t))
'''
abc
a
'''

我们在编辑模式第一次调用next()的时候打印出的返回值是1 2 3

我们在函数中又添加了一个yield关键字

然后我们在交互模式再次进行next的调用

这次的返回值是abc a

我们在调用next()的时候,这个我们会回到上一次yield结束的后面的一个位置

然后从那里开始寻找关键字yield进行数据的返回的操作

next()的作用就是获取yield后面的内容的

我们每次调用的时候就会回到上次yield结束的位置,从那个位置开始

yield和return的区别就是return会直接将函数进行结束,但是yield会保留此次的位置,下次调用的时候就从这个位置开始进行

yield只会中断,但是不会进行结束的操作

def testa():
    for i in range(1,10):
        yield i

t1=testa()
#获取yield返回值的方式:next()
print(next(t1))
#每次获取一个值,有多少个值就获取多少次
print(next(t1))
print(next(t1))
print(next(t1))
print(next(t1))
print(next(t1))
print(next(t1))
'''
1
2
3
4
5
6
7
'''
#如果我们超出了了的话就是会报错了
#因为最后一个yield后面没有数据了



t2=testa()
#获取yield返回值的方法二
for n in t2:
    print(n)

#我们通过for循环能够一次性拿完

#我们通过next()的时候我们需要的时候就可以进行调用,想拿几个就拿几个

print(next(t1))
#8
#我们需要用的时候就进行调用一下,不用就放着

对于获取yield我们有两种方法的,第一种就是进行Next函数的调用

第二种就是利用for循环,直接将对象当做条件进行循环,将这个函数中所有的yield后面的值进行返回

我们对于第一种的话,想什么时候用就什么时候用,随时能够进行调用的操作

image.png

装饰器本质上是一个Python函数(其实就是闭包),它可以让其他函数在不需要做任何代码变动的前提下增加额外功能,装饰器的返回值也是一个函数对象

image.png

image.png


'''在不改动函数的情况下,给函数添加内容
装饰器的定义:
1.嵌套函数
2.外函数返回内函数名
3.外函数中定义一个形参,形参用来接受被装饰的函数名信息
4.要添加的额外功能,写在内函数中

调用函数:函数名()

使用装饰器:@装饰器名(外部函数名)
'''
#定义装饰器
def funa(name):
    def funb():
        print('---开始执行函数---')
        #在这个中间执行被装饰的函数
        name()#这个形参是用来接受这个被装饰的函数名的信息
        '''
        name=testc
        name()=testc()
        '''

        print('---函数执行完毕---')
    return funb

#被装饰的函数
@funa
def testc():
    print('执行testc函数')

testc()

'''
---开始执行函数---
执行testc函数
---函数执行完毕---
'''
@funa
def testd():
    print('执行testd函数')
testd()
'''
---开始执行函数---
执行testd函数
---函数执行完毕--
'''

我们先进行装饰器的定义操作

1.嵌套函数

2.外函数返回内函数名

3.外函数中定义一个形参,形参用来接受被装饰的函数名信息

4.要添加的额外功能,写在内函数中

5.在内部函数中调用被装饰的函数,即外函数的函数名

我们的外函数有个形参name就是用来接受被装饰函数的函数名信息的

方便我们在内函数中进行调用

我们在被装饰的函数的定义上面加上 @外部函数名

那么就说明我们这个函数就已经被装饰好了

那么我们对testc进行调用的操作,那么就会运行我们之前在内部函数中做的装饰代码

总结:我们先会执行这个 @funa

@后面跟的是一个装饰器函数

然后就直接将这个testc的内容给到了name

给到name 之后我们就往后面走

执行内部函数,这个name=装饰的函数的函数名

我们先运行到@funa

然后就运行到def testc()

然后就def funa(name)

然后就运行到def funb()

运行到这个内部函数的时候

我们会直接返回这个funb()返回到被装饰的函数

就是返回到testc这里

那么到这里的话装饰就完成了

然后就直接跳到了testc()的带调用处

然后进行testc的调用的时候

我们就会直接调用装饰器内部函数

我们跳到testc()的地方的时候我们直接进行装饰器的内部函数的调用操作了

对于函数装饰的代码我们写在内部函数中

外部函数一定要定义形参,接受被装饰函数的函数名

不然我们在内部函数中无法进行被装饰函数的调用

那么装饰器的作用:在不改变原函数的情况下对函数进行一系列的装饰操作

就是一个外包操作的升级版本

def log(u):
    def aaa(name,pwd):
        d={'123456':{'pwd':'1234'},
              '1234567':{'pwd':'12345'}
             }
        if name in d:
            if pwd==d[name]['pwd']:
                print("登录成功")
                #调用被装饰的函数
                u(name,pwd)#外部参数的形参是u
            else:
                print("密码错误")

        else:
            print("用户不存在")
    return aaa


#取款
@log
def get_money(name,pwd):
    print(f"取款1000")
#查询
def set_money(name,pwd):
    print(f"存款1000")



get_money('1256','1234')

#通过装饰器我们能减少代码冗余的效果

不改变函数代码的情况下,对函数增加一系列的操作

image.png

相关文章
|
3天前
|
SQL 人工智能 安全
【灵码助力安全1】——利用通义灵码辅助快速代码审计的最佳实践
本文介绍了作者在数据安全比赛中遇到的一个开源框架的代码审计过程。作者使用了多种工具,特别是“通义灵码”,帮助发现了多个高危漏洞,包括路径遍历、文件上传、目录删除、SQL注入和XSS漏洞。文章详细描述了如何利用这些工具进行漏洞定位和验证,并分享了使用“通义灵码”的心得和体验。最后,作者总结了AI在代码审计中的优势和不足,并展望了未来的发展方向。
|
10天前
|
编解码 Java 程序员
写代码还有专业的编程显示器?
写代码已经十个年头了, 一直都是习惯直接用一台Mac电脑写代码 偶尔接一个显示器, 但是可能因为公司配的显示器不怎么样, 还要接转接头 搞得桌面杂乱无章,分辨率也低,感觉屏幕还是Mac自带的看着舒服
|
17天前
|
存储 人工智能 缓存
AI助理直击要害,从繁复中提炼精华——使用CDN加速访问OSS存储的图片
本案例介绍如何利用AI助理快速实现OSS存储的图片接入CDN,以加速图片访问。通过AI助理提炼关键操作步骤,避免在复杂文档中寻找解决方案。主要步骤包括开通CDN、添加加速域名、配置CNAME等。实测显示,接入CDN后图片加载时间显著缩短,验证了加速效果。此方法大幅提高了操作效率,降低了学习成本。
2688 8
|
12天前
|
存储 缓存 关系型数据库
MySQL事务日志-Redo Log工作原理分析
事务的隔离性和原子性分别通过锁和事务日志实现,而持久性则依赖于事务日志中的`Redo Log`。在MySQL中,`Redo Log`确保已提交事务的数据能持久保存,即使系统崩溃也能通过重做日志恢复数据。其工作原理是记录数据在内存中的更改,待事务提交时写入磁盘。此外,`Redo Log`采用简单的物理日志格式和高效的顺序IO,确保快速提交。通过不同的落盘策略,可在性能和安全性之间做出权衡。
1576 12
|
5天前
|
人工智能 关系型数据库 Serverless
1024,致开发者们——希望和你一起用技术人独有的方式,庆祝你的主场
阿里云开发者社区推出“1024·云上见”程序员节专题活动,包括云上实操、开发者测评和征文三个分会场,提供14个实操活动、3个解决方案、3 个产品方案的测评及征文比赛,旨在帮助开发者提升技能、分享经验,共筑技术梦想。
701 94
|
1月前
|
弹性计算 人工智能 架构师
阿里云携手Altair共拓云上工业仿真新机遇
2024年9月12日,「2024 Altair 技术大会杭州站」成功召开,阿里云弹性计算产品运营与生态负责人何川,与Altair中国技术总监赵阳在会上联合发布了最新的“云上CAE一体机”。
阿里云携手Altair共拓云上工业仿真新机遇
|
18天前
|
人工智能 Serverless API
AI助理精准匹配,为您推荐方案——如何快速在网站上增加一个AI助手
通过向AI助理提问的方式,生成一个技术方案:在网站上增加一个AI助手,提供7*24的全天候服务,即时回答用户的问题和解决他们可能遇到的问题,无需等待人工客服上班,显著提升用户体验。
1466 9
|
5天前
|
SQL 存储 人工智能
【产品升级】Dataphin V4.3重大升级:AI“弄潮儿”,数据资产智能化
DataAgent如何助理业务和研发成为业务参谋?如何快速低成本的创建行业数据分类标准?如何管控数据源表的访问权限?如何满足企业安全审计需求?
355 0
【产品升级】Dataphin V4.3重大升级:AI“弄潮儿”,数据资产智能化
|
2天前
|
人工智能 自然语言处理 程序员
提交通义灵码创新实践文章,重磅好礼只等你来!
通义灵码创新实践征集赛正式开启,发布征文有机会获得重磅好礼+流量福利,快来参加吧!
194 7
|
16天前
|
人工智能 Rust Java
10月更文挑战赛火热启动,坚持热爱坚持创作!
开发者社区10月更文挑战,寻找热爱技术内容创作的你,欢迎来创作!
874 29