python反射、闭包、装饰器_python之闭包、装饰器、生成器、反射

本文深入探讨Python中的闭包、装饰器和生成器。首先介绍了函数的作用域,包括全局、局部和非局部变量。接着,详细讲解了闭包函数,特别是其在装饰器中的应用。然后,通过实例展示了装饰器如何在不改变原函数功能的基础上添加新功能。最后,简要提及了Python的生成器和反射机制,包括getattr、hasattr、setattr和delattr等内置函数的使用。
摘要由CSDN通过智能技术生成

1. 函数的作用域

1.在函数定义时候就固定,与调用位置无关,在调用的时候返回到函数定义的位置

x=1 #全局作用

def f1(): #整个f1函数是定义阶段

deff2():print(x)returnf2deffoo(func):

x=3func()#func()调用阶段

x=200 #这样定义x又变成全局了

foo(f1()) #执行foo(f1()) 返回到F1函数里找x所以打印是1不是3

生成器

2.名称空间

内置名称空间:在python解释器启动产生的空间,简单理解就是python自带的方法比如:max()len()

全局名称空间 :在全局定义的好的名称空间,文件级别的产生的 比如:

x=1 #全局作用

deftest():

x=2 #局部作用print(x)

test()

if x=1 y=2#全局作用

#在执行test()调用的时候先加载内置查找有没有test这个内置函数,如果去全局里查找test这个函数方法,最后局部

局部名称空间:在调用函数时候产生局部名称空间 y=2,在调用teset()立马产生了局部作用空间

nonlocal x 函数正上方的x的变量

global x 改成全局的变量

3.闭包函数

闭包函数的作用主要在与装饰器

函数式编程里面闭包只是给函数捆绑死一个值或者状态

1.定义在函数内部的函数

2. 包含对外部作用域名字的引用,而不是对全局作用域名字的引用

3.该内部函数就称为闭包函数

importrequests#第一种传参闭包

defre(func):defget():returnrequests.get(func).textreturnget

baidu=re('http://www.baidu.com')

baidu1=re('http://www.souhu.com')print(baidu())#第二种自定义固定值闭包函数

deftest():

url='http://www.baidu.com'

defgett(): #gett()是闭合函数外面包裹这个urlreturnrequests.get(url).textreturngett

func=test()print(func())

def ff():

url=‘nq.com’

func() #此时的func已经一种状态就是url这个百度地址不会

ff() #得到的结果是url='http://www.baidu.com'

4.装饰器

装饰器的定义:1.在不变动主功能函数的前提下,为其添加上新的功能或者状态。

例:比如抓数据时候,我们需要添加计时器,显示抓取用了多少时间,

写一个te函数是计时器功能,这样可以给爬取baidu、搜狐网站添加上这个计时器。

第一种装饰器写法,其实是给闭包函数添加了一个功能

importrequests

improt time#第一种传参闭包,这里当主功能函数

defre(func):defget():returnrequests.get(func).textreturnget

baidu=re('http://www.baidu.com')

baidu1=re('http://www.souhu.com')print(baidu())#装饰器,给获取百度数据添的用时

defte(func):deftimer():

start=time.time()

func()

stop=time.time()print("获取数据的时间%s"%(stop-start))returntimer

baidu=te(baidu)

baidu()

第二种写法

这种方法适合用@装饰器函数名称,上面例子是闭包函数,已经闭包了是不可以种@的来装饰建议使用  函数调用方式

n是表示给传参的函数,以防万一最要用*args,**kwargs的方式

1 importpsutil2 defcount(func):3 deftimer():4 fu=psutil.cpu_count()5 print(fu)6 func(1)7 returntimer8

9

10 #查看cpu使用情况的一个函数

11 @count12 defmain(n):13 res=psutil.cpu_times()14 print(res)15 print(n)16

17 #查看Process的一个函数n

18 @count19 deftest(t):20 fl=psutil.Process21 print(fl)22 print(t)23

24

25 test()26 main()

View Code

装饰器的细节问题:

from functools import wraps 在装饰器中添加显示备注信息,如下代码

importpsutilfrom functools importwrapsdefcount(func):

@wraps(func) #这里需要添加wraps这个装饰器deftimer():

fu=psutil.cpu_count()print(fu)

func(1)returntimer#查看cpu使用情况的一个函数

@countdefmain(n):'''这个是mian函数的'''res=psutil.cpu_times()print(res)print(n)

#return 123 如果函数里有return 打印这个函数结果时候显示空 因为,main显示已经被conunt函数装饰了,所以要在timer里面retrun才对print(main.__doc__) #这里是打印上面的备注信息

生成器

生成器:不断调用和返回值,生成器

迭代器:可以被next()函数调用并不断返回下一个值的对象称为迭代器,可以直接作用于for循环的对象统称为可迭代对象

# def Pycharm(name):

# food_list=[]

# food= yieldfood_list

# print("%s想吃%s" %(name, food))

# food_list.append(food)

# print(food_list)

#

#

# res= Pycharm('alex')

# next(res) #next 会停留在 food=yield这里,send传送后继续执行下面的内容

# res.send('fangfood')

python之反射

反射即想到4个内置函数分别为:getattr、hasattr、setattr、delattr  获取成员、检查成员、设置成员、删除成员下面逐一介绍先看例子

它的核心本质其实就是利用字符串的形式去对象(模块)中操作(查找/获取/删除/添加)成员,一种基于字符串的事件驱动

classfunc():def __init__(self):

self.teacher='teacher'self.student='student'self.age='ee'self.name='name'

deftest01(self):print('反射机智显示test01功能%s' %self.teacher)deftest02(self):print('反射机智显示test02功能%s' %self.student)

func=func()defrun():''':return:通过res用户输入相对应的变量、函数方法

通过反射机制来判断、增删添改'''res=input('>>我想执行里面的函数:').strip()

hasa=hasattr(func,res) #hasattr 可以判断func类中是否存在res传入的变量或者函数方法

res=getattr(func,res,'not find') #getattr 可以获取传入方法或者函数的执行结果,

#得到是内存地址需要通过res()来展示内容

#not find意思 如果没有找到相对应的变量和函数 通过print(res())来打印出notfind

#print(res())

tom=setattr(func,res,18) #setattr 可以修改传入变量的结果 比如self age=17 通过setattr可以修改成18

print(func.age)

delattr(func,'age') #delattr 删除age这个变量

print(func.age)#setattr(func,res,18)

run()

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值