python反射、装饰和生成器
1.反射
反射的原理:
- 通过字符串对象(一个字符串的变量名)对应类对象中的属性和方法,对实例对象进行修改和访问
- 对类对象进行属性和方法的操作
- 添加
- 删除
- 修改
- 查看
1.1 通过反射添加或覆盖方法
class A: def a(self): print("a方法被调用啦!") def b(self): print("a方法被调用啦!") def c(self): print("a方法被调用啦!") def d(self): print("a方法被调用啦!") x=A() y=A() #对对象添加一个实例方法 # 反射添加的方法只针对被添加的对象,对其他对象并不会造成影响 #x对象添加的方法,y对象去调用就会报错 setattr(x,"e",print) #x.e()=print() x.e("e方法被调用啦!") setattr(x,"f",input) num=x.f("请输入你的密码:") #x.f()=input() x.e(num) #对对象覆盖一个实例方法 setattr(x,"b",print) x.b("这是通过反射覆盖之后的b方法")
1.2 通过反射删除对象方法
1 class A: 2 o=888 3 4 def __init__(self): 5 self.i=666 6 def a(self): 7 print("a方法被调用啦!") 8 def b(self): 9 print("b方法被调用啦!") 10 def c(self): 11 print("c方法被调用啦!") 12 def d(self): 13 print("d方法被调用啦!") 14 15 x=A() 16 17 print(x.i) 18 delattr(x,"i") #只能删除实例属性,不能删除方法 19 #print(x.i) 20 #delattr(x,"a") #不能删除方法 21 x.a() 22 #delattr(x,"o") #不能删除类属性 23 print(x.o)
1.3 通过反射判断对象是否有指定的方法
1 class A: 2 o=888 3 4 def __init__(self): 5 self.i=666 6 def a(self): 7 print("a方法被调用啦!") 8 def b(self): 9 print("b方法被调用啦!") 10 def c(self): 11 print("c方法被调用啦!") 12 def d(self): 13 print("d方法被调用啦!") 14 15 @classmethod 16 def e(cls): 17 print("e方法") 18 19 x=A() 20 21 #可以使用内建函数hasattr()判断对象是否存在该类的类属性或者实例属性,类方法或者实例方法 22 hasattr(x,"a") #True 23 hasattr(x,"b") #True 24 hasattr(x,"v") #False 25 hasattr(x,"o") #True 26 hasattr(x,"i") #True 27 hasattr(x,"e") #True
1.4 通过反射读取方法
1 o=888 2 3 def __init__(self): 4 self.i=666 5 def a(self): 6 print("a方法被调用啦!") 7 def b(self): 8 print("b方法被调用啦!") 9 def c(self): 10 print("c方法被调用啦!") 11 def d(self): 12 print("d方法被调用啦!") 13 14 @classmethod 15 def e(cls): 16 print("e方法") 17 18 x=A() 19 f=getattr(x,"a") # 把a方法的引用传递给f 20 f() #a方法被调用啦 #那么f()=a() 21 22 f=getattr(x,"i") 23 print(f) 24 25 f=getattr(x,"e") 26 f() 27 28 f2=getattr(x,"o") 29 print(f2) 30 31 #通过x对象调用所有方法 32 for i in ["a","b","c","d","e"]: 33 f=getattr(x,i) 34 f()
2 装饰器
作用:装饰器的本质是一个python函数,它可以在不改动其他函数的前提下,对函数的功能进行扩充。
装饰器用于以下场景:
引入日志、函数执行时间统计、执行函数后清理功能、权限校验、缓存
def test1(func):
def test2(root,key):
if root == "root" and key ==123:
print("您的用户名和密码输入正确")
else:
print("您的用户名或密码输入错误")
return test2
@test1
def test3(root,key):
pass
test3("root",1234)
函数:
- 可以作为参数传递
- 可以作为返回值返回
- 修改名字
- 新的覆盖旧的
函数也是一种变量
装饰器:
- 接受函数,并返回函数的函数
- 是一个函数,参数是函数,返回值是函数
2.1 使用装饰器
装饰器的装饰过程:被装饰函数作为参数,传递给装饰器,并且返回值覆盖原函数
def logs(func): #装饰器
def f(*args,**kwargs):
print(level,datetime.datetime.now(),func.__name__,"开始调用了")
func(*args,**kwargs) #转发参数
print(level,datetime.datetime.now(),func.__name__,"调用结束了")
return f
@logs
def add():
print("add is calling")
等同于
add=logs(add)
2.2 装饰器怎么接收参数
被装饰函数有参数怎么办?
装饰器的返回值,接收参数,并传递给被装饰的函数
装饰器怎么接收自己的参数
创建一个函数来接收参数,然后返回原来的装饰器
import datetime
def logs(level):
def _logs(func): #装饰器
def f(*args,**kwargs):
print(level,datetime.datetime.now(),func.__name__,"开始调用了")
func(*args,**kwargs) #转发参数
print(level,datetime.datetime.now(),func.__name__,"调用结束了")
return f
return _logs
#
# def p(y):
# return y
#
# #@p
@logs(level="INFO") #装饰器的使用 logs接收参数 才能完成调用 返回一个返回值
def add(x,y): #被装饰函数
print("add is calling:",f"{x=},{y=}")
def sub(x,y): #被装饰函数
print("sub is calling:",f"{x=},{y=}")
#装饰的过程:被装饰的函数作为参数,传递给装饰器,并且将返回值覆盖原来的函数
#add=logs(add)
add(x=1,y=2)
sub=logs(level="DEBUG")(sub)
sub(x=11,y=22)
3.生成器
如果函数中有yield关键字,其调用结果,则返回一个生成器
生成器是一个可迭代对象,可以被for循环遍历使用
range 就是一个生成器
在遍历时才执行,并计算返回值
生成器,属于迭代器:交给for循环进行使用
def add(a,b):
c=a+b
print(c)
yield 123
return c
c=add(1,2)
print(c) #c是生成器
for i in c: #生成器:在使用数据时,才产生数据
print(f"{i=}")
l=[1,2,3]
iter_l=iter(l) #为列表创建迭代器
for i in iter_l: #for循环是为了迭代器服务的
print(i)
for i in l:
print(i)
4.面试题
4.1 什么是可迭代对象、迭代器、生成器?
1)可迭代对象包含迭代器。
2)如果一个对象拥有__iter__方法,其是可迭代对象;如果一个对象拥有next方法,其是迭代器。
3)定义可迭代对象,必须实现__iter__方法;定义迭代器,必须实现__iter__和next方法。
生成器是一种特殊的迭代器,生成器自动实现了“迭代器协议”(即__iter__和next方法),不需要再手动实现两方法。
生成器在迭代的过程中可以改变当前迭代值,而修改普通迭代器的当前迭代值往往会发生异常,影响程序的执行
具有yield关键字的函数都是生成器,yield可以理解为return,返回后面的值给调用者。不同的是return返回后,函数会释放,而生成器则不会。在直接调用next方法或用for语句进行下一次迭代时,生成器会从yield下一句开始执行,直至遇到下一个yield
4.2.创建一个装饰器,用来校验【被装饰函数】收到的参数是否包含关键字参数,如果是,则打印 :Error:调用本函数是,只能传递位置参数
def check_kwargs(func):
def f(*args,**kwargs):
if kwargs:
print("Error:调用本函数是,只能传递位置参数")
return
#raise ValueError
return func(*args,**kwargs)
return f
@check_kwargs
def add(a,b):
return a+b
print(f"1+1={add(1,1)}")
print(f"2+2={add(a=2,b=2)}")
4.3.创建一个生成器,用来模拟和代替内置的range函数
def my_range(start,end=None,step=None):
if end==None and step==None: #说明只有start接收到了参数
start,end=end,start #start和end互换
if start is None:
start=0
if end is None:
end=0
if step is None:
step=1
if end==0:
return
while True:
yield start
start+=step
if start>=end:
break
print("*"*10)
for i in my_range(5,0,1):
print(i)
print("*"*10)
for i in my_range(5):
print(i)
print("*"*10)
for i in my_range(5,10):
print(i)
print("*"*10)
for i in my_range(1,10,2):
print(i)
浙公网安备 33010602011771号