#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. ubuntu 18.04 安装 flash

    下载源码包, 解压 sudo cp Downloads/flash_player_npapi_linux.x86_64/libflashplayer.so /usr/lib/mozilla/plugi ...

  2. Python之容器类Collections

    容器类Collections 标签(空格分隔): Python进阶 defaultdict counter deque namedtuple defaultdict defaultdict的作用是可以 ...

  3. 170406回顾-SQL Server的smalldatetime类型比较

    在比较SQL Server的类型为smalldatetime字段时出现下面的错误:将 expression 转换为数据类型 smalldatetime 时出现算术溢出错误 正确的比较方法如下:将lon ...

  4. 洛谷P3385负环

    传送门 #include <iostream> #include <cstdio> #include <cstring> #include <algorith ...

  5. 洛谷P2300 合并神犇

    传送门啦 分析: 刚开始读完题后感觉很懵,怎么算都不是3,结果发现题目理解错了.题目要求的是求一个不降的序列,不是递减的(发现自己好傻) 看明白题就好做了吧.经典的区间dp题,合并果子大家应该都做过, ...

  6. Robust Mesh Watermarking

    之前看了一篇题为"Robust Mesh Watermarking"的论文,查阅资料的时候发现了一篇与之很相似的名为"三维模型数字水印系统的设计与实现"的中文论 ...

  7. Centos简介

    Centos作为主流的一种Linux操作系统,以后项目中,比如后期Redis,以及部署一些项目,会把Centos作为服务器操作系统,我们选用Centos,主要是免费,以及稳定. Centos详细介绍, ...

  8. 20155309南皓芯 实验2 Windows口令破解

    在网络界,攻击事件发生的频率越来越高,其中相当多的都是由于网站密码泄露的缘故,或是人为因素导致,或是口令遭到破解,所以从某种角度而言,密码的安全问题不仅仅是技术上的问题,更主要的是人的安全意识问题. ...

  9. PHP的XML Parser(转)

    PHP处理XML文件 一.读取,更新(创建或者操作)一个XML文档,需要XML解析器 .有两种XML parsers: 1. Tree-based parser:将XML文档转化为DOM Tree结构 ...

  10. **CI中使用IN查询(where_in)

    注意别漏了$this->db->get(); /** * 匹配用户手机号,返回匹配的用户列表 * @param $column_str 'user_id, user_name, user_ ...