#coding:utf8

#斐波那契数列,第三项起,每一项都等于前两项之和

def memo(func):
cache = {}#闭包
def wrap(*args):
if args not in cache:
cache[args] = func(*args)
return cache[args]
return wrap @memo
def fibonacci(n):
if n<=1:
return 1
return fibonacci(n-1)+fibonacci(n-2) #上楼梯算法,总过n个台阶,一次只能迈a-b个台阶,不能后退,问有几种走法
@memo
def climb(n,steps):
count = 0
if n==0:
count = 1
elif n>0:
for step in steps:
count += climb(n-step,steps)
return count

装饰器的简单应用还是明白的,但是要理解这两个递归算法。

关于函数的元数据,如:f.__name__,指的是def时指定的名字;f.__doc__,指的是函数文档字符串;f.__moudle__,指的是函数所属模块;f.__default__,指的是函数默认值;f.__dict__,指的是属性字典;f.__closure__,指的是函数的闭包。。。这些元数据都是func的属性。

使用装饰器可能会改变func的元数据。

#coding:utf8
from functools import update_wrapper,wraps,WRAPPER_ASSIGNMENTS,WRAPPER_UPDATES def mydecorator(func):
@wraps(func)#第一种
def wrapper(*args,**kw):
'''wrapper function'''
print 'In wrapper'
func(*args,**kw)
#wrapper.__name__ = func.__name__ 第二种
#update_wrapper(wrapper,func,('__name__','__doc__'),('__dict__',))#第四个参数是将func的属性更新到wrap
#第三种:update_wrapper后两个的默认参数分别是('__module__', '__name__', '__doc__');('__dict__',)
return wrapper @mydecorator
def example():
'''example function'''
print 'In example'

其实就是对处理后的func的属性进行重新赋回原来的值。

定义带参数的装饰器,就是用去判断参数的类型,因为要用过python3的库,所以先略过。

修改装饰器的属性。这样的好处是能够在运行中动态修改,而不是每次都要去装饰器处修改。

给包裹函数增加一个函数,作为包裹函数的属性,使其可以修改闭包中的自由变量。

 import time
from functools import wraps
import logging def warn(timeout):
timeout = [timeout]#将参数设为只有一个数的list
def decorator(func):
@wraps(func)
def wrapper(*args,**kw):
start = time.time()
res = func(*args,**kw)
used = time.time()-start
if used > timeout[0]:
msg = '%s : %s >%s' %(func.__name__,used,timeout[0])
logging.warn(msg)
return res
def setTimeout(k):#注意这里
timeout[0] = k
wrapper.setTimeout = setTimeout
return wrapper
return decorator from random import randint
@warn(0.5)
def test():
print 'In test'
while randint(0,1):
print 'I am sleep'
time.sleep(1) for _ in range(10):
test() test.setTimeout(1)
for _ in range(10):
test()

使用类来定制装饰器。

 #coding:utf8
import logging
from time import localtime,time,strftime,sleep class CallingInfo(object):
def __init__(self,name):
log = logging.getLogger(name)
log.setLevel(logging.INFO)#设置等级
fh = logging.FileHandler(name+'.log')#设置文件处理方法
log.addHandler(fh)#绑定方法到log上
log.info('Start'.center(50,'-'))#添加log文本的头部
self.log= log#绑定到类的属性上
self.formatter = '%(func)s ->[%(time)s - %(used)s -%(ncalls)s]'#定义了输出模板 def info(self,func):#这个就是平常的装饰器
def wrapper(*args,**kw):
wrapper.ncalls +=1
lt = localtime()#返回当地时间
start = time()
res = func(*args,**kw)
used = time() - start info = {}#建立info的字典
info['func'] = func.__name__
info['time'] = strftime('%x %X',lt)
info['used'] = used
info['ncalls'] = wrapper.ncalls msg = self.formatter % info#将字典映射到输入模板上
self.log.info(msg)#输出字符串
return res
wrapper.ncalls = 0#这个语句需要在这个位置的解释有待完善
return wrapper def setFromatter(self,formatter):
self.formatter = formatter def turnOn(self):
self.log.setLevel(logging.INFO) def turnOff(self):
self.log.setLevel(logging.WARN)#级别提高就不在那里输出了 cinfo1 = CallingInfo('mylog1')
cinfo2 = CallingInfo('mylog2') cinfo1.setFromatter('%(func)s ->[%(time)s -%(ncalls)s]')
cinfo2.turnOff()#这里可以看到是很方便地修改装饰器 @cinfo1.info
def f():
print 'in f' @cinfo1.info
def h():
print 'in h' @cinfo2.info
def g():
print 'in g' from random import choice
for _ in xrange(10):
choice([f,g,h])()
sleep(choice([0.5,1,1.5]))

Python168的学习笔记8的更多相关文章

  1. Python168的学习笔记7

    关于多线程操作. 对于IO操作,如访问网站,写入磁盘这种需要时间等待响应的操作,多个cpu也几乎不能提高效率. 对于CPU密集型操作,如这个格式转换,可以通过多个cpu同时去进行. 但是对于pytho ...

  2. Python168的学习笔记6

    如何派生内置不可变类型并修改实例化行为. 个人理解,如何派生出自己想要的类. class IntTuple(tuple): def __new__(cls,iterable): g = (x for ...

  3. Python168的学习笔记5

    关于对csv文件的操作. python标准库中有csv的库,使用非常方便. import csv with open('pingan.csv','rb') as rf: reader = csv.re ...

  4. Python168的学习笔记4

    关于普通文本文件的读写 python2.7中,未注明的字符都是以acsii来编码的,而要让字符能够通用,必须声明为unicode. s=u'你好',s.encode('utf8')就是指用utf8来进 ...

  5. Python168的学习笔记3

    list.extend(),可以拓展list,a=(0,1),b=(2,3) a.extend(b),a就变成(0,1,2,3) 分割字符串(除去字符串中的,\/;之类的),如果用str.split( ...

  6. Python168的学习笔记2

    关于for循环,其实质是利用被循环对象的__iter__,或者__getitem__属性接口,由可迭代对象得到迭代器.for循环就是不断调用.next(),直到最终捕获到stop. import re ...

  7. Python168的学习笔记1

    在对list的条件选择有两种常用方法,直接使用filter函数,就是filter(func,sequence);另外一种就是迭代操作,类似 x for x in sequence func.这两种方法 ...

  8. js学习笔记:webpack基础入门(一)

    之前听说过webpack,今天想正式的接触一下,先跟着webpack的官方用户指南走: 在这里有: 如何安装webpack 如何使用webpack 如何使用loader 如何使用webpack的开发者 ...

  9. PHP-自定义模板-学习笔记

    1.  开始 这几天,看了李炎恢老师的<PHP第二季度视频>中的“章节7:创建TPL自定义模板”,做一个学习笔记,通过绘制架构图.UML类图和思维导图,来对加深理解. 2.  整体架构图 ...

随机推荐

  1. BZOJ4840 NEERC2016 Binary Code

    Problem BZOJ Solution 可能是因为快要省选了,所以最近更博的频率好像高了点_(:зゝ∠)_ 每个字符串最多有两个状态,然后要满足一些依赖关系,考虑2sat.可以先把字符串的结束节点 ...

  2. js如何查看元素类型

    <script type="text/javascript"> //定义变量temp var temp = Object.prototype.toString.appl ...

  3. linux limits研究

    ---------------------------------------------------------------------------------------------------- ...

  4. 二、springcloud之熔断器hystrix

    一.背景 雪崩效应 在微服务架构中通常会有多个服务层调用,基础服务的故障可能会导致级联故障,进而造成整个系统不可用的情况,这种现象被称为服务雪崩效应.服务雪崩效应是一种因“服务提供者”的不可用导致“服 ...

  5. Python模块:Random(未完待续)

    本文基于Python 3.6.5的官文random编写. random模块简介 random为各种数学分布算法(distributions)实现了伪随机数生成器. 对于整数,是从一个范围中均匀选择(u ...

  6. Mysql Limit操作

    oracle : ||   mysql: contact    contact_ws 拼接   Font Size: Large | Medium | Small select * from tabl ...

  7. python网络编程-线程队列queue

    一:线程queu作用 Python中,queue是线程间最常用的交换数据的形式. 队列两个作用:一个是解耦,一个是提高效率 二:语法 1)队列的类 class queue.Queue(maxsize= ...

  8. Filter过滤器-JavaWeb三大组件之一

    Servlet.Filter.Listener是JavaWeb的三大组件,给Web开发提供了很大的便利. 什么是Filter? Filter,过滤器.类似与生活中的净水器.空气净化器. JavaWeb ...

  9. 浅析redux

    一 redux 思想 首先,每一个webApp有且只有一个state tree,为方便管理和跟踪state的变化,也为了减少混乱,redux只允许通过发送(dispatch)action的方式来改变s ...

  10. iOS图片缓存

    iOS的内存管理始终是开发者面临的大问题,内存占用过大时,很容易会被系统kill掉,开发者需要尽可能的优化内存占用问题. 现在的App界面做的越来越精致,里面集成了大量的图片,笔者首先想到的就是如何减 ...