*args和**kwargs传参的注意点 + 装饰器
传参的时候,字典格式的形参2种写法.
如下方示例
def aa(**kwargs): print(kwargs) # 第一种 等式写法 aa(name = "alex") #第二种字典键值对加**写法 aa(**{"name":"alex"})
列表传参 注意点
1 def test(*args): 2 print(args,"参数个数为%s"%len(args)) 3 str = [1, 2, 3, 4] 4 5 test(str) #输出结果([1, 2, 3, 4],) 参数个数为1 6 test(*str) #输出结果(1, 2, 3, 4) 参数个数为4
传形参时候,test(str)是将列表整体作为一个数据传入.而test(*str) 是将str作为一个列表传入.
所以前者用len函数输出为1,而后者作为列表传入则显示有4个元素
一个装饰器例子如下:
1 def zsq(func): 2 def wrapper(*args): 3 import re 4 v = func(*args) 5 ret = [] 6 for i in args: 7 ret.append(re.findall("\d+",i)) 8 return ret 9 10 return wrapper 11 @zsq 12 def a(*args): 13 print(*args) 14 str = ["a88d66e3a","wwa665wwa"] 15 print(a(*str))
-----------------------------------------------------
输出结果为
a88d66e3a wwa665wwa
[['88', '66', '3'], ['665']]
装饰器代码详解
def deco1(func): def outter1(*args, **kwargs): print("start deco1>>outer1") res1 = func(*args, **kwargs) return res1 return outter1 def deco2(func): def outter2(*args, **kwargs): print("start deco2>>outer2") res2 = func(*args, **kwargs) return res2 return outter2 def deco3(x): def outter3(func3): print("deco3传入的参数为%s" % x) def wrapper3(*args, **kwargs): print("start deco3>>outter3>>wrappre3") res3 = func3(*args, **kwargs) return res3 return wrapper3 return outter3 # 多个装饰器加载顺序自下而上,deco3() → deco2 →deco1, # 运行顺序自上而下 deco1 → deco2 →deco3() @deco1# index = deco1(outter2的内存地址) ==> outter1的内存地址 @deco2 # index = deco2(wrapper3的内存地址) ==> outter2 的内存地址 @deco3(5) # 加载 deco3 就是得到outter3地址==>index = outter3(index) = wrapper3 def index(x, y): print("Index函数开始运行,参数为%s和%s" % (x, y)) index("A", "B") # 运行结果 '''deco3传入的参数为5 start deco1>>outer1 start deco2>>outer2 start deco3>>outter3>>wrappre3 Index函数开始运行,参数为A和B'''
装饰器 实现爬虫 错误重试次数
1 import requests 2 3 def time_count(time_count): 4 def wrapper(func): 5 def inner(*args,**kwargs): 6 flg =1 7 while time_count >= flg: 8 try: 9 # print(args,"try里面",kwargs) 10 ret = func(*args,**kwargs) 11 print(ret.status_code) 12 print(func.__name__) 13 return ret 14 except: 15 print("第%s出错了,错误的参数是"%flg,args,kwargs) 16 flg += 1 17 return inner 18 return wrapper 19 20 @time_count(3) 21 def fun(**kwargs): 22 url = kwargs.get("url") 23 timeout = kwargs.get("timeout") 24 print("获取的url参数是%s,time参数是%s"%(url,timeout)) 25 ret = requests.get(url=url,timeout=timeout) 26 print("from fun 访问url >>",url) 27 return ret 28 29 url_list=["https://www.baidu.com/","https://www.b.com/","https://www.c.com/"] 30 31 for i in url_list: 32 fun(**{"url":i,"timeout":3})
输出内容如下
获取的url参数是https://www.baidu.com/,time参数是3
from fun 访问url >> https://www.baidu.com/
200
fun
获取的url参数是https://www.b.com/,time参数是3
第1出错了,错误的参数是 () {'url': 'https://www.b.com/', 'timeout': 3}
获取的url参数是https://www.b.com/,time参数是3
第2出错了,错误的参数是 () {'url': 'https://www.b.com/', 'timeout': 3}
获取的url参数是https://www.b.com/,time参数是3
第3出错了,错误的参数是 () {'url': 'https://www.b.com/', 'timeout': 3}
获取的url参数是https://www.c.com/,time参数是3
第1出错了,错误的参数是 () {'url': 'https://www.c.com/', 'timeout': 3}
获取的url参数是https://www.c.com/,time参数是3
第2出错了,错误的参数是 () {'url': 'https://www.c.com/', 'timeout': 3}
获取的url参数是https://www.c.com/,time参数是3
第3出错了,错误的参数是 () {'url': 'https://www.c.com/', 'timeout': 3}
给类方法添加装饰器
装饰器的基本语法是 func = wrapper(func(*args,**kwargs)),但是如果给类方法添加装饰器,并且还要使用里面实例的self,
在调用的时候采用的是类方法的调用方式而不是对象方法的调用方式,所以是func(self, *args, **kwargs)
(在本例中等价于class.func(self, *args, **kwargs)
)。
装饰器需要改写成
1 def wrapper(func): 2 def inner(self,*args, **kwargs): 3 res = func(self,*args, **kwargs) 4 return res 5 return inner