1.反射
反射允许使用变量的值(而不是变量的名)对成员进行访问
反射的功能:
- 设置属性
- 读取属性
- 删除属性
- 判断属性
class A:
class_name="A"
age=18
def show_me(self):
print(class_name)
attr_name="test_case_name"
setattr(A,attr_name,"sex") #设置属性
delattr(A,attr_name) #删除属性
if hasattr(A,attr_name): #判断属性
print(getattr(A,attr_name)) #读取属性
else:
print(f"属性{attr_name}不存在")
2 装饰器
原理:
函数:
- 可以作为参数传递
- 可以作为返回值返回
- 修改名字
- 新的覆盖旧的
函数也是一种变量
装饰器:
- 接受函数,并返回函数的函数
- 是一个函数,参数是函数,返回值是函数
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.创建一个装饰器,用来校验【被装饰函数】收到的参数是否包含关键字参数,如果是,则打印 :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.2.创建一个生成器,用来模拟和代替内置的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)